IPMsg(飞鸽传书)文件发送源码分析

本文分析了飞鸽传输核心传送过程。

DWORD  WINAPI TMainWin::SendFileThread( void  *_sendFileObj)
{
    SendFileObj *obj = (SendFileObj *)_sendFileObj;
    fd_set      fds;
    fd_set      *rfds = NULL, *wfds = &fds;
    timeval     tv;
     int          sock_ret;
     BOOL         ret = FALSE, completeWait = FALSE;
     // 这里SendFileFunc根据command类型自动选择两种函数 : send file or send directory
     BOOL         (TMainWin::*SendFileFunc)(SendFileObj *obj) =
                obj->command == IPMSG_GETDIRFILES ? TMainWin::SendDirFile : TMainWin::SendFile;

    FD_ZERO(&fds);
    FD_SET(obj->conInfo->sd, &fds);
     // 这里for条件引入了一个简单的超时机制
     // 正常情况下,只要文件未传送完,循环不会退出
     for  ( int  waitCnt=0; waitCnt < 180 && obj->hThread != NULL; waitCnt++)
    {
        tv.tv_sec = 1, tv.tv_usec = 0;
         // 这里select有什么用途呢? 对于select功能我还不是完全明白
         // 根据我的分析,这里主要是利用了select函数的等待功能
         // 如果sd描述符没有就绪,则在select中最久等待1秒
         // 如此反复等待最多180次,也就是3分钟,超过三分钟后,for循环结束
         if  ((sock_ret = ::select(obj->conInfo->sd + 1, rfds, wfds, NULL, &tv)) > 0)
        {
             // 套接字可用,清除等待
            waitCnt = 0;

             //下面的代码是一个有限状态机
             /*
            * 控制变量 completeWait, obj->status
            * 状态机迁移过程(一般状态下):
            * (1)  if ((mainWin->*SendFileFunc)(obj) != TRUE)
            * (2)  if (obj->status == FS_COMPLETE)
            * (3)  if (completeWait)
            *
            * 1首先被反复执行,直到文件发送完毕。
            * 在没有错误的情况下,1总是等价于if(false)
            * 于是其后的2每次都会被执行,判断是否完成
            * 一旦完成,completeWait被设置为True,
            * 在下一次循环里,进入3
            * 在3的语句体内执行recv函数,等待对方应答
            * 
            */
             if  (completeWait)
            {
                 // 本分支在文件发送完后执行
                 if  (::recv(obj->conInfo->sd, ( char  *)&ret,  sizeof (ret), 0) >= 0)
                    ret = TRUE;
                 break ;
            }
             else   if  ((mainWin->*SendFileFunc)(obj) != TRUE)
            {
                 //本分支仅在发送出错时进行
                 break ;  
            }
             else   if  (obj->status == FS_COMPLETE)
            {
                 // 本分支在发送完成后执行
                completeWait = TRUE, rfds = &fds, wfds = NULL;
                 if  (obj->fileSize == 0) { ret = TRUE;  break ; }
            }
        }
         else   if  (sock_ret == 0) {
             // select超时,重置fds
            FD_ZERO(&fds);
            FD_SET(obj->conInfo->sd, &fds);
        }
         else   if  (sock_ret == SOCKET_ERROR) {
             // select错误,算了,离去吧~
             break ;
        }
    }

     // 如果发送的是文件夹,还需要擦一下屁股
     if  (obj->isDir)
    {
        mainWin->CloseSendFile(obj);
         while  (--obj->dirCnt >= 0)
            ::FindClose(obj->hDir[obj->dirCnt]);
    }

     // ret是对方发回的返回值,告知发送方是否完成接收
    obj->status = ret ? FS_COMPLETE : FS_ERROR;
     // 发送TCPEVENT消息,关闭句柄
     // 消息处理流程: EventUser->TcpEvent->EndSendFile
    mainWin->PostMessage(WM_TCPEVENT, obj->conInfo->sd, FD_CLOSE);
     // 退出发送线程
    ::ExitThread(0);
     return   0;
}

上面传送数据最重要的一句是:

           else   if  ((mainWin->*SendFileFunc)(obj) != TRUE)

SendFileFunc的实际内容是什么呢?由函数开始赋值的指针知道:

BOOL  TMainWin::SendFile(SendFileObj *obj)
{
     if  (obj == NULL || obj->hFile == INVALID_HANDLE_VALUE)   //判断文件句柄是否合法
         return   FALSE;

     int      size = 0;
    _int64  remain = obj->fileSize - obj->offset;    //取得还需要传递的总字节数

     //传数据
     if  (remain > 0 && (size = ::send(obj->conInfo->sd, obj->mapAddr + (obj->offset % cfg->ViewMax), remain > cfg->TransMax ? cfg->TransMax : ( int )remain, 0)) < 0)
         return   FALSE;

     // 根据本次成功发送的数据量,调整offset
    obj->offset += size;

     // 如果offset等于文件大小了,那么设置obj状态为完成
     // 由于存在传文件夹模式和传文件模式,所以状态分情况设置
     if  (obj->offset == obj->fileSize)
        obj->status = obj->command == IPMSG_GETDIRFILES ? FS_ENDFILE : FS_COMPLETE;
     else   if  ((obj->offset % cfg->ViewMax) == 0) //没有完成,但是已经传送完成了本部分数据映射,需要调整映射窗口
    {
        ::UnmapViewOfFile(obj->mapAddr);     // 删除旧映射
        remain = obj->fileSize - obj->offset;    // 计算新的剩余量
         // 映射下一块,一次8M ,如果只剩下最后一点了,则少于8M (remain)
        obj->mapAddr = ( char  *)::MapViewOfFile(obj->hMap, FILE_MAP_READ, ( int )(obj->offset >> 32), ( int )obj->offset, ( int )(remain > cfg->ViewMax ? cfg->ViewMax : remain));
    }
     // 更新总消耗时间
    obj->conInfo->lastTick = ::GetTickCount();

     return   TRUE;
}

很多朋友向我要飞鸽带注释的源码,实在很抱歉,我只注释了这么多,其余的也没有深入地看。如果你对带注释的源码感兴趣,不妨来这里看看: http://code.google.com/p/ipigeon/ 这是我在GoogleCode上开的一个项目,大家一起来注释飞鸽源码吧!
评论 25
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值