免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
楼主: duanjigang
打印 上一主题 下一主题

源码阅读第一期:axel和wget [复制链接]

论坛徽章:
0
101 [报告]
发表于 2011-10-03 22:57 |只看该作者
axel_open之后就该打开网络连接了,当然是axel_start了。呵呵

  1. void axel_start( axel_t *axel )
  2. {
  3.         int i;
  4.        
  5.         /* HTTP might've redirected and FTP handles wildcards, so
  6.            re-scan the URL for every conn                                */
  7.         for( i = 0; i < axel->conf->num_connections; i ++ )
  8.         {
  9.                 conn_set( &axel->conn[i], axel->url->text );
  10.                 axel->url = axel->url->next;
  11.                 axel->conn[i].local_if = axel->conf->interfaces->text;
  12.                 axel->conf->interfaces = axel->conf->interfaces->next;
  13.                 axel->conn[i].conf = axel->conf;
  14.                 if( i ) axel->conn[i].supported = 1;
  15.         }
  16.        
  17.         if( axel->conf->verbose > 0 )
  18.                 axel_message( axel, _("Starting download") );
  19.        
  20.         //遍历连接,创建多线程
  21.         for( i = 0; i < axel->conf->num_connections; i ++ )
  22.         //在范围之内
  23.         if( axel->conn[i].currentbyte <= axel->conn[i].lastbyte )
  24.         {
  25.                 if( axel->conf->verbose >= 2 )
  26.                 {
  27.                         axel_message( axel, _("Connection %i downloading from %s:%i using interface %s"),
  28.                                       i, axel->conn[i].host, axel->conn[i].port, axel->conn[i].local_if );
  29.                 }
  30.                
  31.                 axel->conn[i].state = 1;
  32.                 //给每个连接创建一个线程
  33.                 if( pthread_create( axel->conn[i].setup_thread, NULL, setup_thread, &axel->conn[i] ) != 0 )
  34.                 {
  35.                         axel_message( axel, _("pthread error!!!") );
  36.                         axel->ready = -1;
  37.                 }
  38.                 else
  39.                 {
  40.                         axel->conn[i].last_transfer = gettime();
  41.                 }
  42.         }
  43.        
  44.         /* The real downloading will start now, so let's start counting        */
  45.         axel->start_time = gettime();
  46.         axel->ready = 0;
  47. }
复制代码
然后看看连接线程setup_thread

  1. /* Thread used to set up a connection                                        */
  2. void *setup_thread( void *c )
  3. {
  4.         conn_t *conn = c;
  5.         int oldstate;
  6.        
  7.         /* Allow this thread to be killed at any time.                        */
  8.         pthread_setcancelstate( PTHREAD_CANCEL_ENABLE, &oldstate );
  9.         pthread_setcanceltype( PTHREAD_CANCEL_ASYNCHRONOUS, &oldstate );
  10.        
  11.         if( conn_setup( conn ) )
  12.         {
  13.                 conn->last_transfer = gettime();
  14.                 if( conn_exec( conn ) )
  15.                 {
  16.                         conn->last_transfer = gettime();
  17.                         conn->enabled = 1;
  18.                         conn->state = 0;
  19.                         return( NULL );
  20.                 }
  21.         }
  22.        
  23.         conn_disconnect( conn );
  24.         conn->state = 0;
  25.         return( NULL );
  26. }
复制代码

论坛徽章:
0
102 [报告]
发表于 2011-10-03 22:59 |只看该作者
最后剩下axel_do下载主体部分了,明天搞,基本上就over了

论坛徽章:
0
103 [报告]
发表于 2011-10-04 14:15 |只看该作者
下载的主循环:

  1.         signal( SIGINT, stop );
  2.         signal( SIGTERM, stop );
  3.         //没收下载完或者收到停止信号前,一直下载
  4.         //axel_start中已经把axel->ready置为0了
  5.         //stop信号处理中会把run置为0
  6.         while( !axel->ready && run )
  7.         {
  8.                 long long int prev, done;
  9.                
  10.                 prev = axel->bytes_done;
  11.                 //进入下载函数,进行数据接收
  12.                 axel_do( axel );
  13.                
  14.                 //配置了-a方式打印信息的话
  15.                 if( conf->alternate_output )
  16.                 {                       
  17.                         //有信息需要打印而且有进度产生
  18.                         if( !axel->message && prev != axel->bytes_done )
  19.                                 print_alternate_output( axel );
  20.                 }
  21.                 else//否则,用wget打印的方式去打印信息
  22.                 {
  23.                         /* The infamous wget-like 'interface'.. ;)                */
  24.                         //转换成K,下载增量
  25.                         done = ( axel->bytes_done / 1024 ) - ( prev / 1024 );
  26.                         //下载不为0K,而且要打印
  27.                         if( done && conf->verbose > -1 )
  28.                         {
  29.                                 for( i = 0; i < done; i ++ )
  30.                                 {
  31.                                         i += ( prev / 1024 );
  32.                                         //50个字符一换行
  33.                                         if( ( i % 50 ) == 0 )
  34.                                         {
  35.                                                 //大于1K
  36.                                                 if( prev >= 1024 )
  37.                                                         printf( "  [%6.1fKB/s]", (double) axel->bytes_per_second / 1024 );
  38.                                                 //打印进度。。百分比
  39.                                                 //文件小于10M
  40.                                                 if( axel->size < 10240000 )
  41.                                                         printf( "\n[%3lld%%]  ", min( 100, 102400 * i / axel->size ) );
  42.                                                 else//大于等于10M
  43.                                                         printf( "\n[%3lld%%]  ", min( 100, i / ( axel->size / 102400 ) ) );
  44.                                         }//10个字符一个空格
  45.                                         else if( ( i % 10 ) == 0 )
  46.                                         {
  47.                                                 putchar( ' ' );
  48.                                         }
  49.                                         putchar( '.' );
  50.                                         i -= ( prev / 1024 );
  51.                                 }
  52.                                 fflush( stdout );
  53.                         }
  54.                 }
  55.                
  56.                 if( axel->message )
  57.                 {
  58.                         if(conf->alternate_output==1)
  59.                         {
  60.                                 /* clreol-simulation */
  61.                                 putchar( '\r' );
  62.                                 for( i = 0; i < 79; i++ ) /* linewidth known? */
  63.                                         putchar( ' ' );
  64.                                 putchar( '\r' );
  65.                         }
  66.                         else
  67.                         {
  68.                                 putchar( '\n' );
  69.                         }
  70.                         print_messages( axel );
  71.                         if( !axel->ready )
  72.                         {
  73.                                 if(conf->alternate_output!=1)
  74.                                         print_commas( axel->bytes_done );
  75.                                 else
  76.                                         print_alternate_output(axel);
  77.                         }
  78.                 }
  79.                 else if( axel->ready )
  80.                 {
  81.                         putchar( '\n' );
  82.                 }
  83.         }
  84.         //下载完成,打印信息
  85.         strcpy( string + MAX_STRING / 2,
  86.                 size_human( axel->bytes_done - axel->start_byte ) );
  87.         //打印下载速度
  88.         printf( _("\nDownloaded %s in %s. (%.2f KB/s)\n"),
  89.                 string + MAX_STRING / 2,
  90.                 time_human( gettime() - axel->start_time ),
  91.                 (double) axel->bytes_per_second / 1024 );
  92.        
  93.         i = axel->ready ? 0 : 2;
  94.         //关闭连接
  95.         axel_close( axel );
复制代码

论坛徽章:
0
104 [报告]
发表于 2011-10-04 15:12 |只看该作者
axel_do主体部分,尝试从多个连接select方式去读取数据,如果读取失败或者连接超时就重新连接。
觉得这里有一点似乎不妥,除了一些致命的错误,比如写文件,select出错,axel会退出外,别的情况,比如连接资源超时,下载失败,都会重新开启线程建立连接,重新下载。
个人觉得这种重复尝试也应该有个超时限制,比如重复了10次或者10分钟,没成功,就认为失败,这样似乎合理些,但是Axel却是一直在try.不过,从另外一方面,也能解释通,因为前面的过程,建立连接还有search,其实都测试过了,如果能走到这一步,站点都是可靠能访问的,这样一直重试似乎也是可行的。
呵呵,不知道别的人怎么想。
下面是代码分析.

  1. //下载的主循环
  2. void axel_do( axel_t *axel )
  3. {
  4.         fd_set fds[1];
  5.         int hifd, i;
  6.         long long int remaining,size;
  7.         struct timeval timeval[1];
  8.        
  9.         /* Create statefile if necessary                                */
  10.         //如果到了保存状态的时间,保存当前状态到状态文件
  11.         if( gettime() > axel->next_state )
  12.         {
  13.                 //把连接的个数,当前下载量,每个连接当前的下载进度保存起来,
  14.                 //如果状态文件已经存在,就清空重写,目的是只保存一份
  15.                 //这样的话,如果程序异常退出,下次开启任务时,就能从状态文件中
  16.                 //重新加载最近的下载状态(不是100%准确的状态,因为保存状态是有周期的),
  17.                 //接着下载
  18.                 save_state( axel );
  19.                 //计算下一次状态保存时间
  20.                 axel->next_state = gettime() + axel->conf->save_state_interval;
  21.         }
  22.        
  23.                 //采用多线程做连接,但是还是单线程select做传输
  24.         /* Wait for data on (one of) the connections                        */
  25.         FD_ZERO( fds );
  26.         hifd = 0;
  27.         //经典的select方式,把有用的socket描述符加入到集合中
  28.         //而且,限于文件下载的特殊任务方式,基本上每个建立好的连接
  29.         //都不会太空,有恒定持续的数据传输,因此在这里select的效率并不弱于epoll
  30.         for( i = 0; i < axel->conf->num_connections; i ++ )
  31.         {
  32.                 if( axel->conn[i].enabled )
  33.                         FD_SET( axel->conn[i].fd, fds );
  34.                 hifd = max( hifd, axel->conn[i].fd );
  35.         }
  36.         //没有任何连接,等待。。。重试
  37.         if( hifd == 0 )
  38.         {
  39.                 /* No connections yet. Wait...                                */
  40.                 usleep( 100000 );
  41.                 goto conn_check;
  42.         }
  43.         else
  44.         {
  45.                 //超时0.1秒
  46.                 timeval->tv_sec = 0;
  47.                 timeval->tv_usec = 100000;
  48.                 /* A select() error probably means it was interrupted
  49.                    by a signal, or that something else's very wrong...        */
  50.                 //select等待数据到来
  51.                 if( select( hifd + 1, fds, NULL, NULL, timeval ) == -1 )
  52.                 {
  53.                         //值位ready为-1,这样,下载的主循环将退出,下载失败
  54.                         axel->ready = -1;
  55.                         return;
  56.                 }
  57.         }
  58.        
  59.         /* Handle connections which need attention                        */
  60.         //循环读取每个socket的数据
  61.         for( i = 0; i < axel->conf->num_connections; i ++ )
  62.         if( axel->conn[i].enabled ) {
  63.         //检测,如果该socket有数据到来,就读取
  64.                 if( FD_ISSET( axel->conn[i].fd, fds ) )
  65.         {
  66.                 //更新最后一次读取数据的时间
  67.                 axel->conn[i].last_transfer = gettime();
  68.                 //尝试读取数据
  69.                 size = read( axel->conn[i].fd, buffer, axel->conf->buffer_size );
  70.                 //读取失败
  71.                 if( size == -1 )
  72.                 {
  73.                         if( axel->conf->verbose )
  74.                         {
  75.                                 axel_message( axel, _("Error on connection %i! "
  76.                                         "Connection closed"), i );
  77.                         }
  78.                         //关闭当前连接,并不等于放弃,还可能重新连接。。
  79.                         axel->conn[i].enabled = 0;
  80.                         conn_disconnect( &axel->conn[i] );
  81.                         continue;
  82.                 }//当前连接的数据读取结束
  83.                 else if( size == 0 )
  84.                 {
  85.                         if( axel->conf->verbose )
  86.                         {
  87.                                 /* Only abnormal behaviour if:                */
  88.                                 if( axel->conn[i].currentbyte < axel->conn[i].lastbyte && axel->size != INT_MAX )
  89.                                 {
  90.                                         axel_message( axel, _("Connection %i unexpectedly closed"), i );
  91.                                 }
  92.                                 else
  93.                                 {
  94.                                         axel_message( axel, _("Connection %i finished"), i );
  95.                                 }
  96.                         }
  97.                         //如果是不支持并发分片下载(也就是说是单连接下载),表明下载完成
  98.                         if( !axel->conn[0].supported )
  99.                         {
  100.                                 axel->ready = 1;
  101.                         }
  102.                         axel->conn[i].enabled = 0;
  103.                         conn_disconnect( &axel->conn[i] );
  104.                         continue;
  105.                 }
  106.                 /* remaining == Bytes to go                                        */
  107.                 remaining = axel->conn[i].lastbyte - axel->conn[i].currentbyte + 1;
  108.                 //需要填充的小于读取的,下载完成
  109.                 if( remaining < size )
  110.                 {
  111.                         if( axel->conf->verbose )
  112.                         {
  113.                                 axel_message( axel, _("Connection %i finished"), i );
  114.                         }
  115.                         axel->conn[i].enabled = 0;
  116.                         //关闭连接
  117.                         conn_disconnect( &axel->conn[i] );
  118.                         //修改需要的数据量大小,比如需要20字节,下载了30字节,那么就只要20字节
  119.                         size = remaining;
  120.                         /* Don't terminate, still stuff to write!        */
  121.                 }
  122.                 /* This should always succeed..                                */
  123.                 //调整偏移,写文件
  124.                 lseek( axel->outfd, axel->conn[i].currentbyte, SEEK_SET );
  125.                 if( write( axel->outfd, buffer, size ) != size )
  126.                 {
  127.                         //写失败,退出
  128.                         axel_message( axel, _("Write error!") );
  129.                         axel->ready = -1;
  130.                         return;
  131.                 }
  132.                 //修改偏移
  133.                 axel->conn[i].currentbyte += size;
  134.                 axel->bytes_done += size;
  135.         }
  136.         else //当前socket描述符不在select中,检查超时
  137.         {
  138.                 //传输超时,关闭连接
  139.                 if( gettime() > axel->conn[i].last_transfer + axel->conf->connection_timeout )
  140.                 {
  141.                         if( axel->conf->verbose )
  142.                                 axel_message( axel, _("Connection %i timed out"), i );
  143.                         conn_disconnect( &axel->conn[i] );
  144.                         axel->conn[i].enabled = 0;
  145.                 }
  146.         } }
  147.         //如果一切ok,就返回
  148.         if( axel->ready )
  149.                 return;
  150.        
  151. conn_check:
  152.         /* Look for aborted connections and attempt to restart them.        */
  153.         //检查有问题的连接,如果未下载完,并且出错了,重新启动线程,开始传输
  154.         for( i = 0; i < axel->conf->num_connections; i ++ )
  155.         {
  156.                 //连接无效并且未下载完
  157.                 if( !axel->conn[i].enabled && axel->conn[i].currentbyte < axel->conn[i].lastbyte )
  158.                 {
  159.                         //状态为0,表明setup_thread是成功执行了的,并且已经执行问,因此,连接的初始化没问题,调用join回收
  160.                         if( axel->conn[i].state == 0 )
  161.                         {       
  162.                                 // Wait for termination of this thread
  163.                                 pthread_join(*(axel->conn[i].setup_thread), NULL);
  164.                                
  165.                                 conn_set( &axel->conn[i], axel->url->text );
  166.                                 axel->url = axel->url->next;
  167.                                 /* axel->conn[i].local_if = axel->conf->interfaces->text;
  168.                                 axel->conf->interfaces = axel->conf->interfaces->next; */
  169.                                 if( axel->conf->verbose >= 2 )
  170.                                         axel_message( axel, _("Connection %i downloading from %s:%i using interface %s"),
  171.                                                       i, axel->conn[i].host, axel->conn[i].port, axel->conn[i].local_if );
  172.                                
  173.                                 //状态设置为1,表示setup_thread开始执行,设置为0,表示setup_thread线程执行结束
  174.                                 axel->conn[i].state = 1;
  175.                                 if( pthread_create( axel->conn[i].setup_thread, NULL, setup_thread, &axel->conn[i] ) == 0 )
  176.                                 {
  177.                                         axel->conn[i].last_transfer = gettime();
  178.                                 }
  179.                                 else
  180.                                 {
  181.                                         axel_message( axel, _("pthread error!!!") );
  182.                                         axel->ready = -1;
  183.                                 }
  184.                         }
  185.                         else //setup_thread线程还未执行完,也就是说连接建立过程还未完成,需要检查连接超时
  186.                         {
  187.                                 //超时了,就取消她...
  188.                                 if( gettime() > axel->conn[i].last_transfer + axel->conf->reconnect_delay )
  189.                                 {
  190.                                         pthread_cancel( *axel->conn[i].setup_thread );
  191.                                         axel->conn[i].state = 0;
  192.                                 }
  193.                         }
  194.                 }
  195.         }

  196.         /* Calculate current average speed and finish_time                */
  197.         //计算平均速度
  198.         axel->bytes_per_second = (int) ( (double) ( axel->bytes_done - axel->start_byte ) / ( gettime() - axel->start_time ) );
  199.         //估算结束时间
  200.         axel->finish_time = (int) ( axel->start_time + (double) ( axel->size - axel->start_byte ) / axel->bytes_per_second );

  201.         /* Check speed. If too high, delay for some time to slow things
  202.            down a bit. I think a 5% deviation should be acceptable.        */
  203.         //速度调整
  204.         if( axel->conf->max_speed > 0 )
  205.         {
  206.                 //如果超速了
  207.                 if( (float) axel->bytes_per_second / axel->conf->max_speed > 1.05 )
  208.                         axel->delay_time += 10000;
  209.                 //速度太低,少休息会儿
  210.                 else if( ( (float) axel->bytes_per_second / axel->conf->max_speed < 0.95 ) && ( axel->delay_time >= 10000 ) )
  211.                         axel->delay_time -= 10000;
  212.                 //速度太低,干脆不休息
  213.                 else if( ( (float) axel->bytes_per_second / axel->conf->max_speed < 0.95 ) )
  214.                         axel->delay_time = 0;
  215.                 usleep( axel->delay_time );
  216.         }
  217.        
  218.         /* Ready?                                                        */
  219.         //下载完了?
  220.         if( axel->bytes_done == axel->size )
  221.                 axel->ready = 1;
  222. }
复制代码

论坛徽章:
0
105 [报告]
发表于 2011-10-04 15:15 |只看该作者
到此为止,axel基本分析完成,从文件,函数,数据结构,到核心代码。不同层面做了展示。
如果哪些部分有错误或者含糊不清的地方,欢迎诸路朋友跟帖交流,呵呵,尽量避免站内消息,那样不方便交流,而且对问题跟踪也不利,在帖子里讨论,更多的人能受益或者发表意见。


休息两天,开始curl代码的分析

论坛徽章:
0
106 [报告]
发表于 2011-10-04 20:50 |只看该作者
貌似,calltree -gb -np -m 这个命令是有局限的。把我带进了test.c中的main()入口函数。如今我遇到其他的几个文件中也有main()函数,有点让我迷糊。不过还是听一个前辈lengyuex的话,从main.c中读吧。前面的有点跑题,测试不知道在哪里进行的。难道. . . . . .还是不瞎猜了。知识有限,只能认为是多进程的。。。。。。。

论坛徽章:
12
2015年辞旧岁徽章
日期:2015-03-03 16:54:1515-16赛季CBA联赛之同曦
日期:2017-03-17 19:13:162016科比退役纪念章
日期:2016-11-07 08:28:12luobin
日期:2016-06-17 17:46:36wusuopu
日期:2016-06-17 17:43:4515-16赛季CBA联赛之福建
日期:2016-01-14 12:49:22程序设计版块每日发帖之星
日期:2015-12-13 06:20:00程序设计版块每日发帖之星
日期:2015-06-08 22:20:00程序设计版块每日发帖之星
日期:2015-06-08 22:20:002015年亚洲杯之科威特
日期:2015-03-24 14:21:272015年迎新春徽章
日期:2015-03-04 09:57:092016科比退役纪念章
日期:2018-04-10 16:20:18
107 [报告]
发表于 2011-10-04 21:07 |只看该作者
如果配置文件里边的参数是分组的怎么办?
举个例子,ipsec-tools的配置文件:
remote anonymous
{

        exchange_mode main,aggressive;
        doi ipsec_doi;
        situation identity_only;

        my_identifier asn1dn;
        certificate_type x509 "my.cert.pem" "my.key.pem";

        nonce_size 16;
        initial_contact on;
        proposal_check strict;        # obey, strict, or claim

        proposal {
                encryption_algorithm 3des;
                hash_algorithm sha1;
                authentication_method rsasig;
                dh_group 2;
        }
}

有颜色的是关键字 其他的是配置的值。
它的解决办法是用yacc&lex。能不能有个更简单的办法呢?

论坛徽章:
0
108 [报告]
发表于 2011-10-04 21:57 |只看该作者
到此为止,axel基本分析完成,从文件,函数,数据结构,到核心代码。不同层面做了展示。
如果哪些部分有错 ...
duanjigang 发表于 2011-10-04 15:15

弱弱的问一句,tcp.c文件中的int fd = socket( PF_INET, SOCK_DGRAM, IPPROTO_IP );其中这些参数的设置有什么意义,我平时看到的都是int fd = socket( PF_INET, SOCK_DGRAM, 0 );
为什么protocol参数是IPPROTO_IP,这和取网卡的信息是否有联系?然后楼主能不能举一反三的讲解下参数的搭配问题与适用情形,本人初学网络编程没什么经验,谢谢先!

论坛徽章:
12
2015年辞旧岁徽章
日期:2015-03-03 16:54:1515-16赛季CBA联赛之同曦
日期:2017-03-17 19:13:162016科比退役纪念章
日期:2016-11-07 08:28:12luobin
日期:2016-06-17 17:46:36wusuopu
日期:2016-06-17 17:43:4515-16赛季CBA联赛之福建
日期:2016-01-14 12:49:22程序设计版块每日发帖之星
日期:2015-12-13 06:20:00程序设计版块每日发帖之星
日期:2015-06-08 22:20:00程序设计版块每日发帖之星
日期:2015-06-08 22:20:002015年亚洲杯之科威特
日期:2015-03-24 14:21:272015年迎新春徽章
日期:2015-03-04 09:57:092016科比退役纪念章
日期:2018-04-10 16:20:18
109 [报告]
发表于 2011-10-04 22:21 |只看该作者
回复 108# ccnufufan


    socket( PF_INET, SOCK_DGRAM, IPPROTO_IP );
  
      PF_INET   :指定协议族,这里是IP协议族
    SOCK_DGRAM :指定套接口类型,这里指定udp的
    IPPROTO_IP :指定协议,这里是Ip协议。

    就是说,定一个运用IP协议族中ip协议的UDP累套接字。不知道这么说会不会很绕,呵呵。unpv1讲解很详细,推荐看看。

论坛徽章:
0
110 [报告]
发表于 2011-10-04 22:53 |只看该作者
本帖最后由 wangzhen11aaa 于 2011-10-05 09:00 编辑

进入main.c
初始化这次按照code中本来的调用。
首先调用
int main( int argc, char **argv)
{
. . . . . .
program_name = argv[0] ; /*这就是wget*/
struct ptimer* timer = ptimer_new(); +++++++++------>/src/ptimer// /*1、_________-------->
double start_time = ptimer_measure(timer );/*8、___----->src/primer.c*/
i18n_initialize();/*3__---->
. . . . .
}

  1. struct ptimer{

  2. ptimer_system_time start; /*起始时间:typedef struct timespec ptimer_system_time; typedef struct timeval ptimer_system_time timespec和timeval是POSIX的精度秒以下的时间类型,前者精确到纳秒,后者精确到微秒。定义如下:
  3. struct timespec {
  4. __kernel_time_t tv_sec;                 /* seconds */
  5. long            tv_nsec;                /* nanoseconds */
  6. };
  7. struct timeval {
  8. __kernel_time_t         tv_sec;         /* seconds */
  9. __kernel_suseconds_t    tv_usec;        /* microseconds */
  10. };  定时*/

  11. double elapsed_last;  /*逝去时间,由ptimer_measure()计算*/
  12. double elapsed_pre_start; /*时间矫正,在measure调用和运行之间的误差*/
  13. }
  14. 1、____________------------>
  15. [code]
  16. 315 struct ptimer *
  17. 316 ptimer_new (void)
  18. 317 {  
  19. 318   struct ptimer *pt = xnew0 (struct ptimer); /*2_____________---------->src/utils.h*/
  20. 319 #ifdef IMPL_init
  21. 320   static bool init_done;
  22. 321   if (!init_done)
  23. 322     {
  24. 323       init_done = true;
  25. 324       IMPL_init ();  /*6、_____------------>/src/primer.c*/
  26. 325     }
  27. 326 #endif
  28. 327   ptimer_reset (pt);
  29. 328   return pt;
  30. 329 }  
复制代码
2、__________--------------->
#define xnew0(type) (xcalloc (1, sizeof (type)))/*____------>3、______----->/xmalloc.c*/
3、________------>
/*检测成功,分配空间并返回此地址*/
95 xcalloc (size_t n, size_t s)
96 {
97   void *p;
98   /* Test for overflow, since some calloc implementations don't have
99      proper overflow checks.  But omit overflow and size-zero tests if
100      HAVE_GNU_CALLOC, since GNU calloc catches overflow and never
101      returns NULL if successful.  因为calloc不进行溢出检测。但是如果定义了GUN calloc 可以捕捉到溢出。*/
102   if ((! HAVE_GNU_CALLOC && xalloc_oversized (n, s)) /*4________---------->*/
103       || (! (p = calloc (n, s)) && (HAVE_GNU_CALLOC || n != 0)))5、______------->/C 函数库中完成的*/
104     xalloc_die ();
105   return p;
106 }
4、___________------------->
35 # define xalloc_oversized(n, s) \
36     ((size_t) (sizeof (ptrdiff_t) <= sizeof (size_t) ? -1 : -2) / (s) < (n)) /*ptrdiff_t和size_t代表看我的博客里面有解释,如果没有定义就进行检测如果整个地址空间分配的数目小于n那么就会出现错误。*/
5、________-------->
calloc 和malloc的区别在于第一个对分配空间进行初始化0,第二个不初始化*/
6、____--------->
#define IMPL_init posix_init /*7、__--->src/primer.c*/
7、__________---------->
在此函数中会遇到几个clock.
/*
CLOCK_REALTIME
System-wide realtime clock. Setting this clock requires appro-
priate privileges. 如果权限很高可以设定;

CLOCK_MONOTONIC
Clock that cannot be set and represents monotonic time since
some unspecified starting point. 不可以设定,这是对时钟中断进行的计数*/
函数调用sysconf是返回系统中的某些常数值。
clock_getres()是获得clock的精确值,并将clock.id对应的clock值赋值给&r
8、______________________________________--------------------------->
double ptimer_measure(struct ptimer *pt)
{
. . . . . .
IMPL_measure(&now); /*9_________---------->src/ptimer.c* 这里是初始化时钟/
elapsed = pt->elapsed_pre_start + IMPL_diff (&now, &pt->start); /*这里就是计算真正逝去的时间。1、______------->src/ptimer.c*/
386   if (elapsed < pt->elapsed_last)  /*这是检测CLOCK_REALTIME其他的时钟不会出现这种情况*/
387     {
388       pt->start = now;
389       pt->elapsed_pre_start = pt->elapsed_last;
390       elapsed = pt->elapsed_last;
391     }
392
393   pt->elapsed_last = elapsed;
394   return elapsed; /*返回逝去时间*/
395 }

9、_______________----------->
#define IMPL_measure posix_measuer /*10、______________-------->src/ptimer.c*/
10、_____________------------------->
static inline void posix_measuer(ptimer_system_time *pst)
{
clock_gettime(posix_clock_id, pst); /*11、__________-----------> clock_gettime returns the current timespec value of tp for the specific  clock */
}

11、__________------>
#define IMPL_diff posix_diff /*ptimer.c 12、____________--------->*/
12、__________________----------------------------->
175 posix_diff (ptimer_system_time *pst1, ptimer_system_time *pst2)
176 {
177   return ((pst1->tv_sec - pst2->tv_sec)
178           + (pst1->tv_nsec - pst2->tv_nsec) / 1e9);  /*时间差*/
179 }
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP