在linux的kernel space编程时用到了copy_from_user这个函数,在编译的时候出现了这样的警告:
Warning: the frame size of xxx bytes is larger than 1024 bytes
我调用copy_from_user的方法是这样的:
copy_from_user(dest_pointer, src_buffer. n);
copy_from_user的作用就是将src_buffer的内容拷贝n个byte到dest_pointer这个目标地址去.
因为我当时的dest_pointer指向的结构体的大小是1028,GCC的提示是1032是大于1024byte的,
看了kernel的源码才知道,copy_from_user这个函数是要自己8byte,16byte,32byte补齐的,而1032就是8的倍数.也就是说,copy_from_user这个函数是会判断第一个参数的类型的大小的.
对这点做了以下的测试:
1.将src_buffer改为函数的返回值,-->不行
2.将src_buffer再封装2遍,->因为GCC在编译的时候会将所有的函数展开,所以仍旧是无效的.
3.将n这个变量改写为常量->这个警告仍然还是会出现的.
因为之前也怀疑过是不是linux的page size不是4k而是1k导致的这个警告,但是经过代码的追踪,发现page size还真就是4k,所以应该排除这个可能.那么也就是说:
copy_from_user是会判断第一个参数的类型的大小,如果超过1k的话,就会报出一个警告,
解决的方法是:
因为第一个参数是自己传入的,所以就不要传入那么大的结构体就可以了,如果确实要获取如此大的结构体,那么你可以先获取1k左右的大小,传入一个大结构的子结构,之后再偏移1k左右,再将后面的数据获取出来.
举个例子:
我要获取的结构体是:
这里sizeof(struct Big) == 2048;
sizeof(struct subA) == 1024;
sizeof(struct subB) == 1024;
你如果直接调用copy_from_user(&big, src_buffer, 2048);
那么就会出现这个警告,你可以用以下的方法来避开这个问题:
这是一个折衷的办法,对一些不允许出现经过的程序而言,此法有效.