码迷,mamicode.com
首页 > 其他好文 > 详细

39 网络相关函数(七)——live555源码阅读(四)网络

时间:2015-08-17 17:15:43      阅读:227      评论:0      收藏:0      [点我收藏+]

标签:

39 网络相关函数(七)——live555源码阅读(四)网络

本文由乌合之众 lym瞎编,欢迎转载 blog.cnblogs.net/oloroso
本文由乌合之众 lym瞎编,欢迎转载 my.oschina.net/oloroso

简介

网络相关函数是一系列用于操作网络数据的函数。在多个文件中都有相关的函数的定义。还有一些函数是系统socket API相关函数,就不提了。
   这一系列的函数大多有一个特点,需要一个UsageEnvironmet&型的参数。
   这些方法大多在live555sourcecontrol\groupsock\include\GroupsockHelper.hh中声明。

14)readSocket从套接口读取数据

readSocket函数从套接口socket读取数据到buffer,并捕获数据发送源的地址到fromAddress
函数返回读取到的字节数,出错时返回0并调用socketErr(env, "recvfrom() error: ")来设置套接口错误消息。

// 从套接口读数据
int readSocket(UsageEnvironment& env,
    int socket, unsigned char* buffer, unsigned bufferSize,
struct sockaddr_in& fromAddress) {
    SOCKLEN_T addressSize = sizeof fromAddress;
    // ssize_t recvfrom(int sockfd,void *buf,int len,unsigned int flags, struct sockaddr *from,socket_t *fromlen);
    // 读取主机经指定的socket传来的数据,并把数据传到由参数buf指向的内存空间,参数len为可接收数据的最大长度。flag一般设置为0。from是来源地址,fromlen传出来源长度
    // 如果正确接收返回接收到的字节数,失败返回-1.
    int bytesRead = recvfrom(socket, (char*)buffer, bufferSize, 0,
        (struct sockaddr*)&fromAddress,
        &addressSize);
    if (bytesRead < 0) {
        //##### HACK to work around bugs in Linux and Windows:
        int err = env.getErrno();
        if (err == 111 /*ECONNREFUSED (Linux) 连接请求被服务器拒绝*/
#if defined(__WIN32__) || defined(_WIN32)
            // What a piece of crap Windows is.  Sometimes
            // recvfrom() returns -1, but with an ‘errno‘ of 0.
            // This appears not to be a real error; just treat
            // it as if it were a read of zero bytes, and hope
            // we don‘t have to do anything else to ‘reset‘
            // this alleged error:
            // 垃圾的Windows。有时recvfrom()返回- 1,但是有errno为0。
            // 这似乎不是一个真正的错误;只是把它当作一个读取零字节,并希望我们不需要做什么“reset”
            // 这所谓的错误:
            || err == 0 || err == EWOULDBLOCK
#else
            || err == EAGAIN
#endif
            || err == 113 /*EHOSTUNREACH (Linux)*/) { // Why does Linux return this for datagram sock?
            fromAddress.sin_addr.s_addr = 0;
            return 0;
        }
        //##### END HACK
        socketErr(env, "recvfrom() error: ");
    }

    return bytesRead;
}

recv/recvfrom 函数

功能描述:
从套接字上接收一个消息。对于recvfrom,可同时应用于面向接的和无连接的套接字。recv一般只用在面向连接的套接字,几乎等同于recvfrom,只要将recvfrom的第个参数设置NULL
如果消息太大,无法完整存放在所提供的缓冲区,根据不同的套接字,多余的字节会丢弃
假如套接字上没有消息可以读取,除了套接字已被设置为非阻塞模式,否则接收调用会等待消息的到来。

函数原型:

#include <sys/types.h>
#include <sys/socket.h>
ssize_t recv(int sock, void *buf, size_t len, int flags);
ssize_t recvfrom(int sock, void *buf, size_t len, int flags,
struct sockaddr *from, socklen_t *fromlen);

参数说明:

  • sock:索引将要从其接收数据的套接字。
  • buf:存放消息接收后的缓冲区。
  • len:buf所指缓冲区的容量。
  • flags:是以下一个或者多个标志的组合体,可通过or操作连在一起
    MSG_DONTWAIT:操作不会被阻塞。
    MSG_ERRQUEUE:指示应该从套接字的错误队列上接收错误值,依据不同的协议,错误值以某种辅佐性消息的方式传递进来,使用者应该提供足够大的缓冲区。导致错误的原封包通过msg_iovec作为一般的数据来传递。导致错误的数据报原目标地址作为msg_name被提供。错误以sock_extended_err结构形态被使用,定义如下
#define SO_EE_ORIGIN_NONE    0
#define SO_EE_ORIGIN_LOCAL   1
#define SO_EE_ORIGIN_ICMP    2
#define SO_EE_ORIGIN_ICMP6   3
struct sock_extended_err
{
    u_int32_t ee_errno;   /* error number */
    u_int8_t ee_origin; /* where the error originated */
    u_int8_t ee_type;    /* type */
    u_int8_t ee_code;    /* code */
    u_int8_t ee_pad;
    u_int32_t ee_info;    /* additional information */
    u_int32_t ee_data;    /* other data */
    /* More data may follow */
};
说明
MSG_PEEK 指示数据接收后,在接收队列中保留原数据,不将其删除,随后的读操作还可以接收相同的数据。
MSG_TRUNC 返回封包的实际长度,即使它比所提供的缓冲区更长, 只对packet套接字有效。
MSG_WAITALL 要求阻塞操作,直到请求得到完整的满足。然而,如果捕捉到信号,错误或者连接断开发生,或者下次被接收的数据类型不同,仍会返回少于请求量的数据。
MSG_EOR 指示记录的结束,返回的数据完成一个记录。
MSG_TRUNC 指明数据报尾部数据已被丢弃,因为它比所提供的缓冲区需要更多的空间。
MSG_CTRUNC 指明由于缓冲区空间不足,一些控制数据已被丢弃。
MSG_OOB 指示接收到out-of-band数据(即需要优先处理的数据)。
MSG_ERRQUEUE 指示除了来自套接字错误队列的错误外,没有接收到其它数据。
  • from:指向存放对端地址的区域,如果为NULL,不储存对端地址。
  • fromlen:作为入口参数,指向存放表示from最大容量的内存单元。作为出口参数,指向存放表示from实际长度的内存单元。

返回说明:

成功执行时,返回接收到的字节数。另一端已关闭则返回0。失败返回-1errno被设为以下的某个值

  • EAGAIN:套接字已标记为非阻塞,而接收操作被阻塞或者接收超时
  • EBADF:sock不是有效的描述词
  • ECONNREFUSE:远程主机阻绝网络连接
  • EFAULT:内存空间访问出错
  • EINTR:操作被信号中断
  • EINVAL:参数无效
  • ENOMEM:内存不足
  • ENOTCONN:与面向连接关联的套接字尚未被连接上
  • ENOTSOCK:sock索引的不是套接字

39 网络相关函数(七)——live555源码阅读(四)网络

标签:

原文地址:http://www.cnblogs.com/oloroso/p/4736913.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!