免费注册 查看新帖 |

Chinaunix

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

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

论坛徽章:
0
131 [报告]
发表于 2011-10-09 11:15 |只看该作者
我有一个疑问,就是除开text.c中有个main函数外,serch.c中也有一个mian函数虽然它是条件编译的,但如果不满足条件就不编译那个main函数,那还写它干嘛呢?有什么意义?

论坛徽章:
0
132 [报告]
发表于 2011-10-09 12:03 |只看该作者
我有一个疑问,就是除开text.c中有个main函数外,serch.c中也有一个mian函数虽然它是条件编译的,但如果不满 ...
guohuatao 发表于 2011-10-09 11:15



    呵呵,测试用啊,给度代码的人测试,帮助理解。
作者开发时也用到咯,测试他的search功能对不。如何等等。
当然这可能需要你手动键入编译命令,添加指定宏定义,比如:

  1. gcc -D_XXX -DXXX search.c
复制代码
等,这样单独search.c编出的exe就能测试

论坛徽章:
1
双子座
日期:2013-11-06 17:18:01
133 [报告]
发表于 2011-10-09 14:38 |只看该作者
回复 128# duanjigang
请问:

    search_makelist中对conn进行了初始化,
  1. memset( conn, 0, sizeof( conn_t ) );
  2.        
  3.        
复制代码
然后

  1.        
  2.         conn->conf = results->conf;
复制代码
接着调用conn_set 再调用conn_init,这期间都没有涉及到conn->local_if,也就是说这个local_if应该还是NULL ,但在conn_init中为什么就直接赋给conn->http->local_if(或是conn->ftp->local_if)了呢?
  1. conn->http->local_if = conn->local_if;
复制代码

  1. http_connect函数接着再被调用,但至此都没有看到conn->local_if的赋值情况,它按理应该还是NULL
  2. if( !http_connect( conn->http, conn->proto, proxy, conn->host, conn->port, conn->user, conn->pass ) )
  3.                 {
  4.                         conn->message = conn->http->headers;
  5.                         conn_disconnect( conn );
  6.                         return( 0 );
  7.                 }
复制代码

论坛徽章:
0
134 [报告]
发表于 2011-10-09 17:38 |只看该作者
本帖最后由 wangzhen11aaa 于 2011-10-10 15:17 编辑

从13层接着说
完成了 模糊匹配后,将所有的模糊相等的都链入链表中。
然后

  1. _getopt_internal_r()
  2. {
  3. . . . . .

  4. 542
  5. 543               FILE *fp = open_memstream (&buf, &buflen);
  6. 544               if (fp != NULL)
  7. 545                 {
  8. 546                   fprintf (fp,
  9. 547                            _("%s: option '%s' is ambiguous; possibilities:"     ),  /*打开此文件*/
  10. 548                            argv[0], argv[d->optind]);
  11. 549
  12. 550                   do         
  13. 551                     {
  14. 552                       fprintf (fp, " '--%s'", ambig_list->p->name);  /*将刚才的那些模糊匹配的打印进此“文件中"*/
  15. 553                       ambig_list = ambig_list->next;
  16. 554                     }  /*
  17. 555                   while (ambig_list != NULL); /*
  18. 556
  19. 557                   fputc_unlocked ('\n', fp);
  20. 558
  21. 559                   if (__builtin_expect (fclose (fp) != EOF, 1)) /* 这是一个expect函数,提前控制分支这里是预测关闭文件失败,博客有分析*/
  22. 560                     {
  23. 561                       _IO_flockfile (stderr); /*
  24. 562
  25. }
复制代码
这有个open_memstream(&buf, &buflen);的函数。
The open_memstream() opens a stream for writing to a buffer. The buffer is dynamically allocated (as with malloc(3)), and automatically grows as required. After closing the stream, the caller should free(3) this buffer.

When the stream is closed (fclose(3)) or flushed (fflush(3)), the locations pointed to by ptr and sizeloc are updated to contain, respectively, a pointer to the buffer and the current size of the buffer. These values remain valid only as long as the caller performs no further output on the stream. If further output is performed, then the stream must again be flushed before trying to access these variables.

A null byte is maintained at the end of the buffer. This byte is not included in the size value stored at sizeloc.
翻译吧: 这个open_memstream()为写入缓冲区打开一个流。这个缓冲区是动态分配的(用malloc()),随要求动态增长的。当关闭这个流时,调用者应该用free()来释放这个缓冲区。
当这个流被fclose()或者fflush()函数关闭,ptr指向的位置和sizeloc被更新来包含相应的指向buffer的指针和当前的缓冲区的大小。这个值依然是有效的只要调用者没有向这个流输出。如果输出,这个流一定被再次冲刷,然后尝试访问这些变量。
一个 \0 字节放在缓冲区的最后。这个字节不被包含到sizeloc的大小中。
意思是,创建一个文件,写入时就写入内存中。
int fprintf ( FILE * stream, const char * format, ... );

Write formatted output to stream

  向特定的流写入以一定格式格式化的数据。
#include <stdio.h>

int fclose(FILE *stream);
DESCRIPTION

    [CX] [Option Start] The functionality described on this reference page is aligned with the ISO C standard. Any conflict between the requirements described here and the ISO C standard is unintentional. This volume of IEEE Std 1003.1-2001 defers to the ISO C standard. [Option End]
   这个参考页面上描述的功能和ISO C 标准上一致。在这里描述的功能和ISO C 标准上的冲突都是无意的。 IEEE Std 1000.1-2001 的卷服从ISO C标标准。
   
    The fclose() function shall cause the stream pointed to by stream to be flushed and the associated file to be closed. Any unwritten buffered data for the stream shall be written to the file; any unread buffered data shall be discarded. Whether or not the call succeeds, the stream shall be disassociated from the file and any buffer set by the setbuf() or setvbuf() function shall be disassociated from the stream. If the associated buffer was automatically allocated, it shall be deallocated.

  fclose函数会引起stream 执行的stream被冲刷,相关文件被关闭。任何没有写入缓存的数据会被写入这个文件;任何没有读出缓存的数据会被丢弃。这个调用是否成功,这个流应该和文件脱钩,任何被setbuf()和setvbuf()函数设置的数据应该和流文件脱离。

    [CX] [Option Start] The fclose() function shall mark for update the st_ctime and st_mtime fields of the underlying file, if the stream was writable, and if buffered data remains that has not yet been written to the file. The fclose() function shall perform the equivalent of a close() on the file descriptor that is associated with the stream pointed to by stream. [Option End]
     flose()函数应该对st_ctime和st_mtime底册文件的这些域更新,如果这个流文件是可写的,如果这个缓存的数据还没有被写入文件。fclose()函数应该和close()函数在和流文件的文件描述符上是等同的。
    After the call to fclose(), any use of stream results in undefined behavior.
  调用fclose()函数后,每一个流文件的使用都会处于未定义状态。

    Upon successful completion, fclose() shall return 0; otherwise, it shall return EOF [CX] [Option Start]  and set errno to indicate the error. [Option End]
  返回值:如果成功完成,fclose()会返回0,其他情况,它会返回EOF,来预示出错。

论坛徽章:
0
135 [报告]
发表于 2011-10-09 21:12 |只看该作者
本帖最后由 wangzhen11aaa 于 2011-10-11 10:54 编辑

接着上面的分析:

  1. 559                   if (__builtin_expect (fclose (fp) != EOF, 1))
  2. 560                     {
  3. 561                       _IO_flockfile (stderr);  /*这里好复杂*/
  4. 先说这个吧。
复制代码

  1. 1 #include <libio/libio.h>
  2. . . . .
  3. /* 先做判断,如果没有就定义一个*/
  4.    14 # undef _IO_flockfile
  5.    15 # define _IO_flockfile(_fp) \
  6.    16   if (((_fp)->_flags & _IO_USER_LOCK) == 0)                \/*这个判断是否被锁,这里是没有锁*1、___-->*/
  7.    17      _IO_lock_lock (*(_fp)->_lock)    /  *重要的在这里,这里就是加锁了2、________------->* / \

复制代码
1、_____________------------->

  1. 我都有点怀疑,为什么简单的判断命令会出现这种神奇的东西,不废话了。接着走:

  2. 判断是干什么的
  3. 判断stderr这个文件这个是个错误输出文件,#define stderr 2 :重要性在这里。stderr,可以在编译阶段输出
  4. 这个_IO_USER_LOCK是   
  5. #define _IO_USER_LOCK 0x8000  这是个魔数 应该代表锁的意思。
  6.   114 /* Magic numbers and bits for the _flags field.
  7.   115    The magic numbers use the high-order bits of _flags;
  8.   116    the remaining bits are available for variable flags.
  9.   117    Note: The magic numbers must all be negative if stdio
  10.   118    emulation is desired. */
  11.   魔数和位是为_flags域。魔数用_flags的高序位;剩下的位对于其他不同的flags可用。
复制代码

  1. 2、________------->
  2.    40 #define _IO_lock_lock(_name) \  /*对此文件加锁!*/
  3.    41   do {                                                                        \
  4.    42     void *__self = THREAD_SELF;     /*这是宏定义获得此线程的地址3、______-------->*/                                           \
  5.    43     if ((_name).owner != __self)        /*重要的分支从这里开始了,如果正在使用加锁的不是本线程*/                                     \
  6.    44       {                                                                       \
  7.    45         lll_lock ((_name).lock, LLL_PRIVATE);         /*如果不是这个线程的5、______----------->检查锁的状态*/                        \
  8.    46         (_name).owner = __self;                                /那么加上锁的线程就可以是本线程了*/               \
  9.    47       }                                                                       \
  10.    48     ++(_name).cnt;      /*本锁的引用数目加1*/                                                      \
  11.    49   } while (0)
  12.    50
复制代码
3、_________-------->

  1. nptl/sysdeps/i386/tls.h, line 262
  2. 262 # define THREAD_SELF \   
  3.   263   ({ struct pthread *__self;     /*建立一个数据结构*/                                           \
  4.   264      asm ("movl %%gs:%c1,%0" : "=r" (__self)  /*输出到某个寄存器 :c 的作用是去掉返回来的$符号,gcc 扩展*/                              \
  5.   265           : "i" (offsetof (struct pthread, header.self)));  /*获得偏移地址4 、__--->                  \
  6.   266      __self;})

  7. 4、_____--------->
  8. #define offsetof(type, field)   __offsetof(type, field)
  9. #define __offsetof(type, field)  __builtin_offsetof(type, field)

  10. 这属于gcc的扩展语法:

  11. inuxcompiler-gcc4.h文件
  12. #undef offsetof
  13. #ifdef __compiler_offsetof
  14. #define offsetof(TYPE,MEMBER) __compiler_offsetof(TYPE,MEMBER)
  15. #else
  16. #define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)
  17. 现在有个函数叫做list_entry()貌似其中就有这个。我原来分析过。
复制代码
header我认为应该是个全局变量。

  1. 5、____-----------> nptl/sysdeps/unix/sysv/linux/i386/lowlevellock.h, line[code]
  2.   286 #define lll_lock(futex, private) \                                             /*mutex为互斥量,这里的futex指文件互斥量*/
  3.   287   (void)                                                                      \/*(void)类型
  4.   288     ({ int ignore1, ignore2;                                                  \
  5.   289        if (__builtin_constant_p (private) && (private) == LLL_PRIVATE)        \  /*如果在运行时,private为给定的常数,这里显然是0因为在i386环境下LLL_PRIVATE所有架构都是0,说明没有此线程,好了,那么就出现了后面的创建和系统调用。*/
  6.   290          __asm __volatile (__lll_lock_asm_start                               \6、_____----->/*开始为了获得zf的值*/
  7.   291                            "jnz _L_lock_%=\n\t"                               \
  8.   292                            ".subsection 1\n\t"                                \ /*段号是1"*/
  9.   293                            ".type _L_lock_%=,@function\n"                     \ /*_L_lock属性是函数*/
  10.   294                            "_L_lock_%=:\n"                                    \
  11.   295                            "1:\tleal %2, %%ecx\n"                             \  /*取long类型的futex的,(futex是一个线程)偏移,放入了ecx寄存器*/
  12.   296                            "2:\tcall __lll_lock_wait_private\n"               \  /*调用__lll_lock_wait_private()8、____---->*/
  13.   297                            "3:\tjmp 18f\n"                                    \ /*跳转*/
  14.   298                            "4:\t.size _L_lock_%=, 4b-1b\n\t"                  \  /*设置名称为_L_lock%大小为3字节*/
  15.   299                            ".previous\n"                                      \  /*本命令交换当前段(及其子段)和最近访问过的段(及其子段)。多个连续的.previous命令将使当前位置两个段(及其子段)之间反复切换。用段堆栈的术语来说,本命令使当前段和堆顶段交换位置*/
  16.   300                            LLL_STUB_UNWIND_INFO_3                             \ 7、_________------>

  17.   301                            "18:"                                              \
  18.   302                            : "=a" (ignore1), "=c" (ignore2), "=m" (futex)     \   [eax寄存器是在__lll_lock_asm_start()函数中返回的eax寄存器的值,或者是else中的那个__status值。*/
  19.   303                            : "" (0), "1" (1), "m" (futex),                   \ /*""代表是内存变量,内存增量寻址,,ecx初始化为 0.*/
  20.   304                              "i" (MULTIPLE_THREADS_OFFSET)                    \ /*"i"代表为立即数,所以要加上P来消除$符号*/
  21.   305                            : "memory");                                       \
  22.   306        else                                                                   \
  23.   307          {                                                                    \
  24.   308            int ignore3;                                                       \
  25.   309            __asm __volatile (__lll_lock_asm_start                             \/*这里进行比较*/
  26.   310                              "jnz _L_lock_%=\n\t"                             \ /*这里和zf有什么干系呢?*/
  27.   311                              ".subsection 1\n\t"                              \  
  28.   312                              ".type _L_lock_%=,@function\n"                   \  /*如果zf为0,就是
  29.   313                              "_L_lock_%=:\n"                                  \
  30.   314                              "1:\tleal %2, %%edx\n"                           \  /*将futex的有效地址放入edx寄存器*/
  31.   315                              "0:\tmovl %8, %%ecx\n"                           \  /*将private的值存入 ecx寄存器*/
  32.   316                              "2:\tcall __lll_lock_wait\n"                     \  /*调用锁等待*/
  33.   317                              "3:\tjmp 18f\n"                                  \
  34.   318                              "4:\t.size _L_lock_%=, 4b-1b\n\t"                \
  35.   319                              ".previous\n"                                    \  /*段之间可以方便切换*/
  36.   320                              LLL_STUB_UNWIND_INFO_4                           \
  37.   321                              "18:"                                            \
  38.   322                              : "=a" (ignore1), "=c" (ignore2),                \
  39.   323                                "=m" (futex), "=&d" (ignore3)                  \
  40.   324                              : "1" (1), "m" (futex),                          \
  41.   325                                "i" (MULTIPLE_THREADS_OFFSET), "" (0),        \
  42.   326                                "g" (private)                                  \
  43.   327                              : "memory");                                     \
  44.   328          }                                                                    \
  45.   329     })
复制代码

  1. 6、___________---------> nptl/sysdeps/unix/sysv/linux/i386/lowlevellock.h
  2.   278 # define __lll_lock_asm_start LOCK_INSTR "cmpxchgl %1, %2\n\t"  
  3.   279 #else
  4.   280 # define __lll_lock_asm_start "cmpl $0, %%gs:%P6\n\t"                         \ /*去掉$符号,这个%%gs: %P6 ,gs是执行t*/
  5.   281                               "je 0f\n\t"                                     \ /*如果是0,那么跳标记0
  6.   282                               "lock\n"                                        \
  7.   283                               "0:\tcmpxchgl %1, %2\n\t"          /*比较并交换指令见博客分析,最后获得的值等效于%2的值存入eax寄存器返回,%2就是futex即是锁的值,但是重点在如果eax中的数如果和%2中的相等,会将zf置1,并且将%1里面的 1 赋给futex。并且会把锁的值清零。否则,就是zf清零,直接将%2的值装入eax(那里代表着ignore1)。这将影响到后面的分支。*/
  8.   284#endif
  9. 7、______________---------->
  10.   174 #define LLL_STUB_UNWIND_INFO_3 \
  11.   175 LLL_STUB_UNWIND_INFO_START                                      \
  12.   176 "10:\t" ".byte  0x40 + (2b-1b) # DW_CFA_advance_loc\n\t"        \
  13.   177 LLL_STUB_UNWIND_INFO_END
复制代码
这里地形复杂:很抱歉把大家带进了AT&T和CPU指令集的地方。不过没有办法。

  1. 这里少什么呢?                  
  2.                  [color=Blue] 就是创建这是在创建锁,文件名是lowlevellock.h不过我没有弄明白这个格式*/[/color]
  3. 122 #define LLL_STUB_UNWIND_INFO_START \                     
  4.   123         ".section       .eh_frame,\"a\",@progbits\n"            \   /*当目标格式为ELF时,.section命令应如下使用:
  5. .section name [, "flags"[, @type]].eh_frame为段名称  中间的a代表allocated,progbits代表包含数据的段*/
  6.   124 "5:\t"  ".long  7f-6f   # Length of Common Information Entry\n" \  /*长度为标号7到标号6之间的长度*/
  7.   125 "6:\t"  ".long  0x0     # CIE Identifier Tag\n\t"               \  /*这里是赋值标记*/
  8.   126         ".byte  0x1     # CIE Version\n\t"                      \ /*还有版本!*/
  9.   127         ".ascii \"zR\\\"       # CIE Augmentation\n\t"         \  /*增强字符*/
  10.   128         ".uleb128 0x1   # CIE Code Alignment Factor\n\t"        \ /*unsigned little endian base 128 (低地址结尾的无符号128位基数)。这是一个紧凑的,变长的数字表示方法,当使用DWARF符号调试格式时使用,这里看来是目标代码对齐。*/
  11.   129         ".sleb128 -4    # CIE Data Alignment Factor\n\t"        \signed little endian base 128”(低地址结尾的带符号128位基数)。这是一个紧凑的,变长的数字表示方法,当使用DWARF符号调试格式时使用,这里是数据对齐*/
  12.   130         ".byte  0x8     # CIE RA Column\n\t"                    \/*编译成下一个字节,是其本身0x8*/
  13.   131         ".uleb128 0x1   # Augmentation size\n\t"                \ /*段长增加长度,是可变化的*/
  14.   132         ".byte  0x1b    # FDE Encoding (pcrel sdata4)\n\t"      \ /*FED编码,google Frequency Domain Ecoding*/
  15.   133         ".byte  0xc     # DW_CFA_def_cfa\n\t"                   \ 334 #define DW_CFA_def_cfa          0x0c
  16.   134         ".uleb128 0x4\n\t"                                      \
  17.   135         ".uleb128 0x0\n\t"                                      \
  18.   136         ".align 4\n"                                            \/*以4字节对齐,这些数据应该是通用的*/
  19.   137 "7:\t"  ".long  17f-8f  # FDE Length\n"                         \  /PDF长度*/
  20.   138 "8:\t"  ".long  8b-5b   # FDE CIE offset\n\t"                   \  /偏移地址,就是前面的值8-5之间的长度*/
  21.   139         ".long  1b-.    # FDE initial location\n\t"             \  /*初始化位置前面的标记1到 一个标记为.的位置*/
  22.   140         ".long  4b-1b   # FDE address range\n\t"                \ /*地址长度范围在3字节范围之内*/
  23.   141         ".uleb128 0x0   # Augmentation size\n\t"                \  
  24.   142         ".byte  0x16    # DW_CFA_val_expression\n\t"            \ /*这是个和上面都是指令框架操作码*/
  25.   143         ".uleb128 0x8\n\t"                                      \
  26.   144         ".uleb128 10f-9f\n"                                     \ /*由于
  27.   145 "9:\t"  ".byte  0x78    # DW_OP_breg8\n\t"                      \ /*同上*/
  28.   146         ".sleb128 3b-1b\n"
  29.   147 #define LLL_STUB_UNWIND_INFO_END \
  30.   148         ".byte  0x16    # DW_CFA_val_expression\n\t"            \
  31.   149         ".uleb128 0x8\n\t"                                      \
  32.   150         ".uleb128 12f-11f\n"                                    \
  33.   151 "11:\t" ".byte  0x78    # DW_OP_breg8\n\t"                      \
  34.   152         ".sleb128 3b-2b\n"                                      \
  35.   153 "12:\t" ".byte  0x40 + (3b-2b-1) # DW_CFA_advance_loc\n\t"      \
  36.   154         ".byte  0x16    # DW_CFA_val_expression\n\t"            \
  37.   155         ".uleb128 0x8\n\t"                                      \
  38.   156         ".uleb128 16f-13f\n"                                    \
  39.   157 "13:\t" ".byte  0x78    # DW_OP_breg8\n\t"                      \
  40.   158         ".sleb128 15f-14f\n\t"                                  \
  41.   159         ".byte  0x0d    # DW_OP_const4s\n"                      \
  42.   160 "14:\t" ".4byte 3b-.\n\t"                                       \
  43.   161         ".byte  0x1c    # DW_OP_minus\n\t"                      \
  44.   162         ".byte  0x0d    # DW_OP_const4s\n"                      \
  45.   163 "15:\t" ".4byte 18f-.\n\t"                                      \
  46.   164         ".byte  0x22    # DW_OP_plus\n"                         \
  47.   165 "16:\t" ".align 4\n"                                            \
  48.   166 "17:\t" ".previous\n"   /*这里真是水平有限了,好多伪代码。像是在构建一个.elf文件吧*/
  49.   167
复制代码

  1. 8、__________-------->
  2. 28 __lll_lock_wait_private (int *futex)   /*前面创建创建锁,判断如果futex互斥锁值为1,那么old就返回1*/
  3.    29 {
  4.    30   do
  5.    31     {  
  6.    32       int oldval = atomic_compare_and_exchange_val_acq (futex, 2, 1); 9、________------>/*
  7.    33       if (oldval != 0)  /*如果futex所指向的那个值不等于0那么old就不是0*/
  8.            /*
  9.    34         lll_futex_wait (futex, 2, LLL_PRIVATE);  10、____________----------->/*如果不是0,锁等待。*/
  10.    35     }
  11.    /*如果返回0,就说明futex没有被锁,就给它上锁为2,并退出循环*/
  12.    36   while (atomic_compare_and_exchange_bool_acq (futex, 2, 0) != 0); 11、__________---------->
  13.    37 }
复制代码

  1. 9和11在下面
  2.    27 /* The only basic operation needed is compare and exchange.  */
  3.    28 #define atomic_compare_and_exchange_val_acq(mem, newval, oldval) \  /*mem是int *类型*/
  4.    29   ({ __typeof (mem) __gmemp = (mem);                                  \ /*定义个__gmemp类型是int类型*/
  5.    30      __typeof (*mem) __gret = *__gmemp;                               \ /*将mem的值赋值给__gret*/
  6.    31      __typeof (*mem) __gnewval = (newval);                            \ /*定义__gnewval为int 类型,数值为newval,就是2*/
  7.    32                                                                       \
  8.    33      if (__gret == (oldval))       /*如果当前锁的值和old向等,那么就说明上锁了。返回1。并改变锁的值为2。此时futex指向的锁变成了2 ,后得,wait锁打开,接着循环。*/              \
  9.    34        *__gmemp = __gnewval; /*gmemp: pointer*/                                         \
  10.    35      __gret; })    /* gret:*/
  11.    36
  12.    37 #define atomic_compare_and_exchange_bool_acq(mem, newval, oldval) \
  13.    38   ({ __typeof (mem) __gmemp = (mem);                                  \ /*gmemp为int *类型*/
  14.    39      __typeof (*mem) __gnewval = (newval);                            \ /*将新值newval 赋给__gnewval*/
  15.    40                                                                       \
  16.    41      *__gmemp == (oldval) ? (*__gmemp = __gnewval, 0) : 1; })  futex锁值为0时,会给他上锁,并停止循环*/
  17.    42
  18.    43 #endif  
  19. 看样子是用
复制代码
  1. 10、_______________--------------->
  2.   198 #define lll_futex_wait(futex, val, private) \  /*根据前面的LLL_PRIVATE:private定义为0*/
  3.   199   lll_futex_timed_wait (futex, val, NULL, private)
  4.   200
  5.   201
  6.   202 #define lll_futex_timed_wait(futex, val, timeout, private) \    val为2,timeout : 0, private  ; 0;
  7.   203   ({                                                                          \
  8.   204     int __status;                                                             \
  9.   205     register __typeof (val) _val asm ("edx") = (val);                         \  /*_val是edx内的变量*/
  10.   206     __asm __volatile (LLL_EBX_LOAD                                            \  
  11.   207                       LLL_ENTER_KERNEL                                        \
  12.   208                       LLL_EBX_LOAD                                            \
  13.   209                       : "=a" (__status)                                       \
  14.   210                       : "" (SYS_futex), LLL_EBX_REG (futex), "S" (timeout),  \
  15.   211                         "c" (__lll_private_flag (FUTEX_WAIT, private)),       \
  16.   212                         "d" (_val), "i" (offsetof (tcbhead_t, sysinfo))       \
  17.   213                       : "memory");                                            \
  18.   214     __status;      /*返回的是最后的这个__status值,如果系统调用成功那么就会返回0*/    \
  19.   215   })
  20.   216
复制代码

  1.   100 #ifdef PIC 如果定义了PIC微指令控制器
  2.   101 # define LLL_EBX_LOAD   "xchgl %2, %%ebx\n"  /*将寄存器D中的futex装载到ebx寄存器*/
  3.   102 # define LLL_EBX_REG    "D"  
  4.   103 #else
  5.   104 # define LLL_EBX_LOAD
  6.   105 # define LLL_EBX_REG    "b"  /*否则就就直接装载到ebx寄存器*/
  7.   106 #endif
复制代码

  1.   108 #ifdef I386_USE_SYSENTER /*I386*/
  2.   109 # ifdef SHARED    是否定义了共享 ,如果是就调用sysinfo的偏移地址进入进程空间
  3.   110 #  define LLL_ENTER_KERNEL      "call *%%gs:%P6\n\t"
  4.   111 # else
  5.   112 #  define LLL_ENTER_KERNEL      "call *_dl_sysinfo\n\t" /*12、_______------>直接调用*/
  6.   113 # endif
  7.   114 #else
  8.   115 # define LLL_ENTER_KERNEL       "int $0x80\n\t"
  9.   116 #endif
复制代码
[code]
12、______------>
142 uintptr_t _dl_sysinfo = DL_SYSINFO_DEFAULT;

   50 # define DL_SYSINFO_DEFAULT (uintptr_t) _dl_sysinfo_int80 /*想往系统调用里面走,我就不怕了*/
   49 extern void _dl_sysinfo_int80 (void) attribute_hidden;
这里就进入系统调用int80,标准系统调用。

论坛徽章:
0
136 [报告]
发表于 2011-10-09 22:03 |只看该作者
本帖最后由 duanjigang 于 2011-10-09 22:07 编辑
回复  duanjigang
请问:

    search_makelist中对conn进行了初始化,然后接着调用conn_set 再调用conn_ ...
seufy88 发表于 2011-10-09 14:38



    嗯,你说的是对的,在这两处local_if都是NULL
但它并不影响网络连接的。因为本地地址绑定不绑定是可选的。
tcp.c的tcp_connect函数中可以看到:

  1. if( local_if && *local_if )
  2.         {   
  3.                 local.sin_family = AF_INET;
  4.                 local.sin_port = 0;
  5.                 local.sin_addr.s_addr = inet_addr( local_if );
  6.                 if( bind( fd, (struct sockaddr *) &local, sizeof( struct sockaddr_in ) ) == -1 )
  7.                 {   
  8.                         close( fd );
  9.                         return( -1 );
  10.                 }   
  11.         }   
复制代码
而tcp_connect则是ftp_connect和http_connect的底层实现。因此,你所指出的两处,确实是空的local_if

我又翻了下代码,local_if有用是在axel.c中的

axel_new和axel_start函数中

  1. axel->conn[0].local_if = axel->conf->interfaces->text;
复制代码

  1. axel->conn[i].local_if = axel->conf->interfaces->text;
复制代码

conf->interfaces->text
确实是 conf_init  函数中的函数 conf_loadfile  来初始化的,代码段

  1. if( strcmp( key, "interfaces" ) == 0 )
  2.                         st = parse_interfaces( conf, value );
复制代码

论坛徽章:
1
午马
日期:2013-09-10 11:03:08
137 [报告]
发表于 2011-10-09 22:39 |只看该作者
支持,学习,提高

论坛徽章:
0
138 [报告]
发表于 2011-10-10 09:16 |只看该作者
回复  duanjigang


    但是不与的话不也是8位的吗?而且和1相与还是没变呀?
inknk 发表于 2011-10-09 09:50



    直接打印p[1],p[2]是32位的整数。
因此能看到这样的输出:

  1. MAC: FFFFFFF0:4D:FFFFFFA2:2A:80:BE==>FFFFFFFF:FFFFFFFF:FFFFFFFF:FF:FF:FF
复制代码
用0xFF进行&运算,就变成了一个8位的整数。打印为

  1. MAC: F0:4D:A2:2A:80:BE==>FF:FF:FF:FF:FF:FF
复制代码
或者不&运算,直接printf的参数为

  1. (unsigned char)p[1], (unsigned char)p[2]
复制代码
也行

论坛徽章:
0
139 [报告]
发表于 2011-10-10 09:50 |只看该作者
这个活动太好了,谢谢斑竹!

论坛徽章:
0
140 [报告]
发表于 2011-10-10 09:52 |只看该作者
这个活动太好了,谢谢斑竹!
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

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

清除 Cookies - ChinaUnix - Archiver - WAP - TOP