- 论坛徽章:
- 15
|
本帖最后由 yulihua49 于 2015-11-16 20:39 编辑
虽然AIO有某些好处,但是程序难写。
你可以用下面的程序进行接口。其中sc.h是我专用的,你换成你的即可,里边就是函数原型。那里边有个:
typedef int (*T_YIELD)(int socket,int rwflg,int timeout);
你的yield功能按这个接口。
要想发送就:SendNet(socket,buf,size,MTU);参数的含义你懂的。
要想接收就: RecvNet(socket,buf,size,timeout);
一般情况下他是同步阻塞的。
如果你能提供协程的yield功能,把yield设置好,他就变成非阻塞的了。
你可以去掉ShowLog或改成你的
- cat AIO_tcp.c
- /**********************************************
- * @(#) SDBC 7.1 TCP RECV/SEND Tools *
- * to suport coroutine
- **********************************************/
- #include <sys/socket.h>
- #include <sys/time.h>
- #include <unistd.h>
- #include <fcntl.h>
- #include <sc.h>
- #ifndef MIN
- #define MIN(a,b) ((a)<(b))?(a):(b)
- #endif
- static T_YIELD yield=NULL;
- T_YIELD get_yield()
- {
- return yield;
- }
- T_YIELD set_yield(T_YIELD new_yield)
- {
- T_YIELD oldyield=yield;
- yield=new_yield;
- return oldyield;
- }
- //timeout for second
- int RecvNet(int socket,char *buf,int n,int timeout)
- {
- int bcount=0,br,ret;
- int i,repeat=0;
- int fflag=-1;
- if(socket<0) return SYSERR;
- if(!buf && n<0) return 0;
- if(yield) {
- fflag=fcntl(socket,F_GETFL,0);
- if(fflag!=-1) fcntl(socket,F_SETFL,fflag|O_ASYNC|O_NONBLOCK); //异步操作
- }
- *buf=0;
- br=0;
- while(bcount<n){
- if((br=read(socket,buf,n-bcount))>0){
- bcount+=br;
- buf+=br;
- repeat=0;
- continue;
- }
- if(fflag==-1 && errno==EAGAIN) return TIMEOUTERR;
- if(br<=0 && errno && errno != EAGAIN){
- if(errno!=ECONNRESET)
- ShowLog(1,"%s:br=%d,err=%d,%s",__FUNCTION__,br,errno,strerror(errno));
- break;
- }
- if(bcount < n && fflag!=-1) { //切换任务
- if(repeat++>3) return -errno;
- ShowLog(5,"%s:tid=%lX,socket=%d,yield to schedle bcount=%d/%d",__FUNCTION__,pthread_self(),socket,bcount,n);
- i=yield(socket,0,timeout);
- if(i<0) {
- if(timeout>0) {
- struct timeval tmout;
- tmout.tv_sec=timeout;
- tmout.tv_usec=0;
- ret=setsockopt(socket,SOL_SOCKET,SO_RCVTIMEO,(char *)&tmout,sizeof(tmout));
- }
- fcntl(socket,F_SETFL,fflag);
- fflag=-1;
- if(i==-11) return -11;
- }
- }
- }
- if(fflag!=-1) fcntl(socket,F_SETFL,fflag);
- return bcount==0?-1:bcount;
- }
- int SendNet(int socket,char *buf,int n,int MTU)
- {
- int bcount,bw;
- int sz,i=0;
- int fflag=-1;
- size_t SendSize;
- if(socket<0) return SYSERR;
- if(yield) {
- fflag=fcntl(socket,F_GETFL,0);
- if(fflag != -1) fcntl(socket,F_SETFL,fflag|O_NONBLOCK); //异步操作
- }
- bcount=0;
- bw=0;
- if(MTU>500) SendSize=MTU;
- else SendSize=n;
- while(bcount<n){
- sz=MIN(n-bcount,SendSize);
- if((bw=write(socket,buf,sz))>0){
- bcount+=bw;
- buf+=bw;
- }
- if(bw<0&&errno!=EAGAIN) {
- ShowLog(1,"%s:err=%d,%s",__FUNCTION__,errno,strerror(errno));
- break;
- }
- if(bw < sz && fflag != -1) { //切换任务
- ShowLog(5,"%s:tid=%lX,socket=%d,yield bw=%d/%d",__FUNCTION__,pthread_self(),socket,bw,sz);
- i=yield(socket,1,0);
- if(i<0) {
- fcntl(socket,F_SETFL,fflag);
- fflag = -1;
- }
- }
- }
- if(fflag != -1) fcntl(socket,F_SETFL,fflag);
- return bcount==0?-1:bcount;
- }
复制代码 |
|