分类: LINUX
2008-11-17 13:38:52
如何找到生成的动态库有3种方式:
1)把库拷贝到/usr/lib和/lib目录下。
(2)在LD_LIBRARY_PATH环境变量中加上库所在路径。
例如动态库libhello.so在/home/example/lib目录下:
$export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/home/example/lib
(3) 修改/etc/ld.so.conf文件,把库所在的路径加到文件末尾,并执行ldconfig刷新。这样,加入的目录下的所有库文件都可见。
当静态库和动态库同名时, gcc命令将优先使用动态库。
--- rd_old.c 2008-11-18 14:50:26.000000000+0800 +++ rd_new.c 2008-11-18 14:49:44.000000000 +0800 @@ -1,10 +1,56 @@ #include #include #include -#include #include #include "rd.h" +//校验和算法 +/*Reliable data defines*/ + +#define VALID_BUF_STAMP 0xCAFECAFE +#define CHECKSUM 0xb36e1207 +#define RD_FORMAT_VER 0x00001017 +#define IMEI_TYPE 0xCAFE0010 +#define NVM_TYPE 0xCAFE0020 +#define END_BUF_STAMP 0xDADADADA--------------------全局变量放到.c文件中 +#define CREATE_SIZE 128*1024 +#define COUNTER_MAX 100 + +#define IMEI_FILE "imei1234.bin" +#define RF_FILE "GsmCalData.nvm" +#define CREATE_FILE "rd.bin" + +/* Data structor */ + +typedef struct tag_buffer_flash_header +{ + unsigned long validBufferStamp; + unsigned long bufCheckSum; + unsigned long version; + unsigned long reserve1; + unsigned long reserve2; +}FLASH_BUFFER_HEADER; + + +typedef struct tag_flash_entry_header +{ + unsigned long entryType; + unsigned long entrySize; + unsigned char fileName[128]; + //unsigned long *pFileToWrite; +}FLASH_ENTRY_HEADER; + +typedef struct tag_reliable_data +{ + FLASH_BUFFER_HEADER header; + FLASH_ENTRY_HEADER entry; +}RD; +/*Global variables*/ + +FLASH_ENTRY_HEADER *entry[COUNTER_MAX]; +FLASH_BUFFER_HEADER buffer_header; +unsigned long end_stamp = END_BUF_STAMP; +int counter=0;// entry counter void buffer_header_initial() @@ -18,6 +64,7 @@ } +/*function: get file size */ static unsigned long get_file_size( const char *filename ) { @@ -56,6 +103,7 @@ #endif } +/*function : read file contents to buffer */ static int read_file( char *filename, char *buffer, unsigned long length) { FILE *fp; @@ -82,7 +130,7 @@ return -1; } } - +/* write buffer to file */ static int write_file( const char *filename, char *buffer, unsigned long length) { FILE *fp; @@ -109,6 +157,7 @@ return -1; } } +/* free entry pointer */ void free_entry() { int i; @@ -117,18 +166,21 @@ free( entry[i] ); } } +/* add an entry */ int add_entry( const char* entry_name) { + unsigned long read_size; if( counter < COUNTER_MAX ) { - - entry[counter] = (FLASH_ENTRY_HEADER *)malloc( entry_size ); + //alloc memory + entry[counter] = (FLASH_ENTRY_HEADER *)malloc( sizeof(FLASH_ENTRY_HEADER) ); if( entry[counter] == NULL ) { printf( "when add entry[%d] alloc memory failed\n" ,counter); free_entry(); return -1; } + //entry type if( counter == 0 ) { entry[counter]->entryType = IMEI_TYPE; @@ -137,11 +189,28 @@ { entry[counter]->entryType = NVM_TYPE ; } + //entry name memset(entry[counter]->fileName, 0, 128 ); - strcpy(entry[counter]->fileName, entry_name ); - if( (entry_size = get_file_size(entry_name) ) != -1) + if( strlen(entry_name) <= 128) + { + strcpy(entry[counter]->fileName, entry_name ); + }-------如果entry_name大于128时可能要失败,所以加判断 + else + { + printf( "Add entry name is too long!\n" ); + free_entry(); + return -1; + } + //entry size + if( (read_size = get_file_size(entry_name) ) != -1) { - entry[counter]->entrySize = entry_size; + entry[counter]->entrySize = read_size; + } + else + { + printf( "Read file:%s failed!\n",entry_name ); + free_entry(); + return -1; } counter++; return 0; @@ -154,29 +223,42 @@ } } +/* create the reliable data as a file */ int create_rd( const char* rd_name) { int i; char *create_buffer; //save the whole contens which used to create the reliale data char *file_buffer;//save contents of entry file - unsigned long length = header_size;//record the position to copy contents; - + unsigned long length = sizeof(FLASH_BUFFER_HEADER);//record the position to copy contents; + + //create header buffer_header_initial(); - //alloc memory + + //alloc whole memory buffer create_buffer = (char *)malloc( CREATE_SIZE* sizeof(char) ); if (create_buffer == NULL) { printf( "Fail to alloc memory to create reliable data buffer, errno is: %d, %s \n", errno, strerror( errno ) ); return -1; } + //initial whole buffer memset( create_buffer, 0xff, CREATE_SIZE ); - memcpy( create_buffer, &buffer_header, header_size);//buffer header + memcpy( create_buffer, &buffer_header, sizeof(FLASH_BUFFER_HEADER));//buffer header // copy entry contents to create_buffer for( i=0; i< counter; i++) { - memcpy( create_buffer+length, entry[i], entry_size); - length += entry_size; + if( entry[i] != NULL ) + { + memcpy( create_buffer+length, entry[i], sizeof(FLASH_ENTRY_HEADER)); + } ---------如果entry[i]为空,则崩溃,所以要加判断,为了安全,用memcpy时一定要小心。当在add_entry里第一个空时,则这个程序可能就崩,所要一定要加判断。 + else + { + printf( "Entry[%d] pointer is NULL or add_entry() return failed!\n" , counter); + free_entry(); + return -1; + } + length += sizeof(FLASH_ENTRY_HEADER); file_buffer = (char *)malloc( entry[i]->entrySize); if (file_buffer == NULL) { @@ -200,17 +282,33 @@ length += entry[i]->entrySize; free( file_buffer ); } - //add end stamp + //add end stamp memcpy( create_buffer+length, &end_stamp, sizeof( unsigned long) ); // create the reliable data if(write_file( rd_name, create_buffer, CREATE_SIZE) == -1) { printf( "Failed to create reliable data file.\n" ); free_entry(); + return -1;
free_entry(); return 0; } |
||||||||
linux中有两类函数库,分别是静态库和动态库。
静态库是一些目标文件的集合,通常为后缀位.o的文件。通过ar工具打包而成,这类库的名字一般是libxxx.a;其中xxx为给定的静态库的文件名,如libm,
为数学函数库,用户也可以创建自己的静态库,在创建可执行程序的过程中,静态库同时被链接到程序代码,被主程序调用的函数目标文件连同主程序组合成单一的
可执行程序,他的优点就显而易见了,即编译后的执行程序不需要外部的函数库的支持,因为所有的函数都已经被编译进去了。当然这也会成为他的缺点,因为如果
静态函数库改变了,那么你的程序必学重新编译。
动态函数库,这类库的名字一般是libxxx.so;
相对于静态函数库,动态函数库在编译的时候并没有被编译进目标代码中,你的程序执行到相关函数时才调用该函数库里的相应函数,因此动态函数库所产生的可执
行文件比较小,由于函数库没有被整合到你的程序,而是程序运行时动态的申请并调用。所以程序的运行环境中必须提供相应的库。动态函数库的改变并不影响你的
程序。使用动态库创建执行程序,分为两个阶段:
链接阶段,即通过ld创建执行程序时,链接编译器会在指定的动态库搜索、解析被主程序调用的函数及其其他变量等,如引用被找到,则在执行程序的XCOFF头结构的loader区域,建立包含引用的动态库的影像,反之,如在指定的动态库中没有找到此引用的定义,编译器会给出类似未定义的符号引用错误,这不同于静态库,包含引用的目标文件并不和执行程序链接在一起。
另一个阶段为运行阶段,即在执行程序运行时,程序运行时,系统相关模块将读取定义执行程序的XCOFF头结构中的信息,查找并加载相关的动态库,假设,所有被应用的动态库都被定位且加载后,程序将开始运行
==================================================
3、查看库中的符号
有时候可能需要查看一个库中到底有哪些函数,nm命令可以打印出库中的涉及到的所有符号。库既可以是静态的也可以是动态的。nm列出的符号有很多,常见的有三种:
一种是在库中被调用,但并没有在库中定义(表明需要其他库支持),用U表示;
一种是库中定义的函数,用T表示,这是最常见的;
另外一种是所谓的“弱 态”符号,它们虽然在库中被定义,但是可能被其他库中的同名符号覆盖,用W表示。
例如,假设开发者希望知道上文提到的hello库中是否定义了 printf():
$nm libhello.so |grep printf U
其中printf U表示符号printf被引用,但是并没有在函数内定义,由此可以推断,要正常使用hello库,必须有其它库支持,再使用ldd命令查看hello依赖于哪些库:
$ldd hello libc.so.6=>/lib/libc.so.6(0x400la000) /lib/ld-linux.so.2=>/lib/ld-linux.so.2 (0x40000000) 从上面的结果可以继续查看printf最终在哪里被定义,有兴趣可以go on
4、生成库
第一步要把源代码编绎成目标代码。
以下面的代码为例,生成上面用到的hello库:
/* hello.c */
#include <>
void sayhello()
{
printf("hello,world ");
}
用gcc编绎该文件,在编绎时可以使用任何全法的编绎参数,例如-g加入调试代码等: gcc -c hello.c -o hello.o
(1)连接成静态库 连接成静态库使用ar命令,其实ar是archive的意思
$ar cqs libhello.a hello.o
(2)连接成动态库 生成动态库用gcc来完成,由于可能存在多个版本,因此通常指定版本号:
$gcc -shared -Wl,-soname,libhello.so.1 -o libhello.so.1.0 hello.o
另外再建立两个符号连接:
$ln -s libhello.so.1.0 libhello.so.1
$ln -s libhello.so.1 libhello.so
这样一个libhello的动态连接库就生成了。最重要的是传gcc -shared 参数使其生成是动态库而不是普通执行程序。 -Wl 表示后面的参数也就是-soname,libhello.so.1直接传给连接器ld进行处理。实
际上,每一个库都有一个soname,当连接器发现它正在查找的程序库中有这样一个名称,连接器便会将soname嵌入连结中的二进制文件内,而不是它正
在运行的实际文件名,在程序执行期间,程序会查找拥有 soname名字的文件,而不是库的文件名,换句话说,soname是库的区分标志。
这样做的目的主要是允许系统中多个版本的库文件共存,习惯上在命名库文件的时候通常与soname相同 libxxxx.so.major.minor
其中,xxxx是库的名字,major是主版本号,minor 是次版本号 =========================================================== 旧的东西:
|