织梦CMS - 轻松建站从此开始!

罗索

socket bad address 错误的解决

落鹤生 发布于 2011-03-08 14:48 点击:次 
落鹤生:实际上这个问题是我在用boost的ASIO来实现一个服务器程序时碰到的,当我在服务器端要发送的Buffer很大时,经常会出现发了一半ASIO就返回错误的问题。查来查去,结果问题原因却是很简单,同下文一样,到头来还是因为内存。
TAG:

不明白,为什么在读取socket缓冲的时候出现bad address 错误?发送端write返回值正常,接收端read却返回0,并且打印bad address 错误信息。

接收端代码:receive.c

#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdio.h>
#include <unistd.h>

/**初始化,得到sock_fd**/

int receive_init(int *sock_fd,int port);

int main()
{
int server_fd;
int client_fd;
int port=8000;
char *buff=”";
client_fd=receive_init(&server_fd,port);
if(client_fd<0){
printf(”receive init failed!\n”);
return -1;
}

while (1)
{   read(client_fd,buff,1);
printf(”receive : %s\n”,buff);

sleep(2);
}

return 0;

}

发送端代码:send.c

#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdio.h>
#include <unistd.h>

/**初始化send,返回client_fd**/
int send_init(int *sock_fd,char *host_addr,int port);

int main()
{
int err;
int port=8000;
char *buff=”ABCD”;
char *host_addr=”127.0.0.1″;
int sock_fd;
err=send_init(&sock_fd,host_addr,port);

if(err<0){
printf(”send init failed!\n”);
return -1;
}
while (1)
{
write(sock_fd,buff,2);
printf(”send: %s\n”,buff);
sleep(2);
}
return 0;

}
在调试的时候注意到,当把receive.c 中的buff改为 char型,也就是说,每次写、读都是1个字符的时候,读正常。这时为什么哩?内存地址错误的原因何在?因为没有分配空间?

网上有解释如下:(来源:这里)

在Linux中可以用strerror(errno)去得到错误代码的字符串描述,在涉及到内存操作的时候,一不小心就 strerror(errno)就返回“Bad address”的错误,当然更多的时候是Killed和Segmentation fault。何解呢?绝大多数情况肯定是内存地址已经错误了,我在工作中已经碰到了以下两种“Badaddress”:

第一、文件读写:

FILE *fp = NULL;

size_t file_length = 0;

……//打开文件并求取文件长度

size_t bytes_read = 0;

unsigned char *ptemp = NULL;

……//忘记给ptemp 赋值或者没有给它分配内存。

while (0 == feof(fp))

{

byres_read = fread(ptemp, 1, file_length, fp);

if (0 != ferror(fp))

{

printf(”fread error[%s]\n”, strerror(errno));

……

}

ptemp += byres_read;

}

这时候返回的错误就是:fread error[Bad address]

原因:ptemp 为空(NULL)

问题:

1. 我如果在ptemp += byres_read;后再加一句file_length -=byres_read;程序会陷入死循环,请兄弟参照微软的 MSDN有关fread的文档查找原因。

第二,recv,send,sendto,recvfrom

有时候调用上述四个函数进行数据收发时也容易发生 “Bad address”的错误,那99%的可能是你函数中有一指针指向无法存取的内存空间。小心哦,兄弟!我们在实际工作中已经碰到过了。

recvfrom可能的错误代码:

1. EBADF 参数s非合法的socket处理代码
2. EFAULT 参数中有一指针指向无法存取的内存空间。
3. ENOTSOCK 参数s为一文件描述词,非socket。
4. EINTR 被信号所中断。
5. EAGAIN 此动作会令进程阻断,但参数s的socket为不可阻断。
6. ENOBUFS 系统的缓冲内存不足
7. ENOMEM 核心内存不足
8. EINVAL 传给系统调用的参数不正确。

附注:
send(经socket传送数据)
相关函数:sendto,sendmsg,recv,recvfrom,socket
#include<sys/types.h>
#include<sys/socket.h>
定义函数 int send(int s,const void * msg,int len,unsigned int falgs);
参数flags一般设0,其他数值定义如下:
MSG_OOB 传送的数据以out-of-band 送出。
MSG_DONTROUTE 取消路由表查询
MSG_DONTWAIT 设置为不可阻断运作
MSG_NOSIGNAL 此动作不愿被SIGPIPE信号中断。

注:果然是没有为指针分配空间的原因。把char *buff 改为char 数组即可,char buff[MAX_SIZE],或者干脆memset (buff,0,MAX_SIZE)。之所以出现 bad address 是因为,指针指向了无法访问的空间。

(ets2008)
本站文章除注明转载外,均为本站原创或编译欢迎任何形式的转载,但请务必注明出处,尊重他人劳动,同学习共成长。转载请注明:文章转载自:罗索实验室 [http://www.rosoo.net/a/201103/11038.html]
本文出处:百度博客 作者:ets2008
顶一下
(1)
100%
踩一下
(0)
0%
------分隔线----------------------------
发表评论
请自觉遵守互联网相关的政策法规,严禁发布色情、暴力、反动的言论。
评价:
表情:
用户名: 验证码:点击我更换图片
栏目列表
将本文分享到微信
织梦二维码生成器
推荐内容