linux內(nèi)核中的get_user和put_user
CPU平臺(tái):arm
本文引用地址:http://www.ex-cimer.com/article/201611/319994.htm在內(nèi)核空間和用戶空間交換數(shù)據(jù)時(shí),get_user和put_user是兩個(gè)兩用的函數(shù)。相對(duì)于copy_to_user和copy_from_user(將在另一篇博客中分析),這兩個(gè)函數(shù)主要用于完成一些簡(jiǎn)單類型變量(char、int、long等)的拷貝任務(wù),對(duì)于一些復(fù)合類型的變量,比如數(shù)據(jù)結(jié)構(gòu)或者數(shù)組類型,get_user和put_user函數(shù)還是無(wú)法勝任,這兩個(gè)函數(shù)內(nèi)部將對(duì)指針指向的對(duì)象長(zhǎng)度進(jìn)行檢查,在arm平臺(tái)上只支持長(zhǎng)度為1,2,4,8的變量。下面我具體分析,首先看get_user的定義(linux/include/asm-arm/uaccess.h):
- externint__get_user_1(void*);
- externint__get_user_2(void*);
- externint__get_user_4(void*);
- externint__get_user_8(void*);
- externint__get_user_bad(void);
- #define__get_user_x(__r2,__p,__e,__s,__i...)
- __asm____volatile__(
- __asmeq("%0","r0")__asmeq("%1","r2")//進(jìn)行判斷(#define__asmeq(x,y)".ifnc"x","y";.err;.endifnt")
- "bl__get_user_"#__s//根據(jù)參數(shù)調(diào)用不同的函數(shù),此時(shí)r0=指向用戶空間的指針,r2=內(nèi)核空間的變量
- :"=&r"(__e),"=r"(__r2)
- :"0"(__p)
- :__i,"cc")
- #defineget_user(x,p)
- ({
- constregistertypeof(*(p))__user*__pasm("r0")=(p);//__p的數(shù)據(jù)類型和*(p)的指針數(shù)據(jù)類型是一樣的,__p=p,且存放在r0寄存器中
- registertypeof(*(p))__r2asm("r2");//__r2的數(shù)據(jù)類型和*(p)的數(shù)據(jù)類型是一樣的,且存放在r2寄存器中
- registerint__easm("r0");//定義__e,存放在寄存器r0,作為返回值
- switch(sizeof(*(__p))){//對(duì)__p所指向的對(duì)象長(zhǎng)度進(jìn)行檢查,并根據(jù)長(zhǎng)度調(diào)用響應(yīng)的函數(shù)
- case1:
- __get_user_x(__r2,__p,__e,1,"lr");
- break;
- case2:
- __get_user_x(__r2,__p,__e,2,"r3","lr");
- break;
- case4:
- __get_user_x(__r2,__p,__e,4,"lr");
- break;
- case8:
- __get_user_x(__r2,__p,__e,8,"lr");
- break;
- default:__e=__get_user_bad();break;//默認(rèn)處理
- }
- x=__r2;
- __e;
- })
上面的源碼涉及到gcc的內(nèi)聯(lián)匯編,不太了解的朋友可以參考前面的博客(http://blog.csdn.net/ce123/article/details/8209702)。繼續(xù),跟蹤__get_user_1等函數(shù)的執(zhí)行,它們的定義如下(linux/arch/arm/lib/getuser.S)。
- .global__get_user_1
- __get_user_1:
- 1:ldrbtr2,[r0]
- movr0,#0
- movpc,lr
- .global__get_user_2
- __get_user_2:
- 2:ldrbtr2,[r0],#1
- 3:ldrbtr3,[r0]
- #ifndef__ARMEB__
- orrr2,r2,r3,lsl#8
- #else
- orrr2,r3,r2,lsl#8
- #endif
- movr0,#0
- movpc,lr
- .global__get_user_4
- __get_user_4:
- 4:ldrtr2,[r0]
- movr0,#0
- movpc,lr
- .global__get_user_8
- __get_user_8:
- 5:ldrtr2,[r0],#4
- 6:ldrtr3,[r0]
- movr0,#0
- movpc,lr
- __get_user_bad_8:
- movr3,#0
- __get_user_bad:
- movr2,#0
- movr0,#-EFAULT
- movpc,lr
- .section__ex_table,"a"
- .long1b,__get_user_bad
- .long2b,__get_user_bad
- .long3b,__get_user_bad
- .long4b,__get_user_bad
- .long5b,__get_user_bad_8
- .long6b,__get_user_bad_8
- .previous
這段代碼都是單條匯編指令實(shí)現(xiàn)的內(nèi)存操作,就不進(jìn)行詳細(xì)注解了。如果定義__ARMEB__宏,則是支持EABI的大端格式代碼(http://blog.csdn.net/ce123/article/details/8457491),關(guān)于大端模式和小端模式的詳細(xì)介紹,可以參考http://blog.csdn.net/ce123/article/details/6971544。這段代碼在.section __ex_table, "a"之前都是常規(guī)的內(nèi)存拷貝操縱,特殊的地方在于后面定義“__ex_table”section 。
標(biāo)號(hào)1,2,...,6處是內(nèi)存訪問指令,如果mov的源地址位于一個(gè)尚未被提交物理頁(yè)面的空間中,將產(chǎn)生缺頁(yè)異常,內(nèi)核會(huì)調(diào)用do_page_fault函數(shù)處理這個(gè)異常,因?yàn)楫惓0l(fā)生在內(nèi)核空間,do_page_fault將調(diào)用search_exception_tables在“__ex_table”中查找異常指令的修復(fù)指令,在上面這段帶面的最后,“__ex_table”section 中定義了如下數(shù)據(jù):
- .section__ex_table,"a"
- .long1b,__get_user_bad//其中1b對(duì)應(yīng)標(biāo)號(hào)1處的指令,__get_user_bad是1處指令的修復(fù)指令。
- .long2b,__get_user_bad
- .long3b,__get_user_bad
- .long4b,__get_user_bad
- .long5b,__get_user_bad_8
- .long6b,__get_user_bad_8
put_user用于將內(nèi)核空間的一個(gè)簡(jiǎn)單類型變量x拷貝到p所指向的用戶空間。該函數(shù)可以自動(dòng)判斷變量的類型,如果執(zhí)行成功則返回0,否則返回-EFAULT。下面給出它們的定義(linux/include/asm-arm/uaccess.h)。
- externint__put_user_1(void*,unsignedint);
- externint__put_user_2(void*,unsignedint);
- externint__put_user_4(void*,unsignedint);
- externint__put_user_8(void*,unsignedlonglong);
- externint__put_user_bad(void);
- #define__put_user_x(__r2,__p,__e,__s)
- __asm____volatile__(
- __asmeq("%0","r0")__asmeq("%2","r2")
- "bl__put_user_"#__s
- :"=&r"(__e)
- :"0"(__p),"r"(__r2)
- :"ip","lr","cc")
- #defineput_user(x,p)
- ({
- constregistertypeof(*(p))__r2asm("r2")=(x);
- constregistertypeof(*(p))__user*__pasm("r0")=(p);
- registerint__easm("r0");
- switch(sizeof(*(__p))){
- case1:
- __put_user_x(__r2,__p,__e,1);
- break;
- case2:
- __put_user_x(__r2,__p,__e,2);
- break;
- case4:
- __put_user_x(__r2,__p,__e,4);
- break;
- case8:
- __put_user_x(__r2,__p,__e,8);
- break;
- default:__e=__put_user_bad();break;
- }
- __e;
- })
- .global__put_user_1
- __put_user_1:
- 1:strbtr2,[r0]
- movr0,#0
- movpc,lr
- .global__put_user_2
- __put_user_2:
- movip,r2,lsr#8
- #ifndef__ARMEB__
- 2:strbtr2,[r0],#1
- 3:strbtip,[r0]
- #else
- 2:strbtip,[r0],#1
- 3:strbtr2,[r0]
- #endif
- movr0,#0
- movpc,lr
- .global__put_user_4
- __put_user_4:
- 4:strtr2,[r0]
- movr0,#0
- movpc,lr
- .global__put_user_8
- __put_user_8:
- 5:strtr2,[r0],#4
- 6:strtr3,[r0]
- movr0,#0
- movpc,lr
- __put_user_bad:
- movr0,#-EFAULT
- movpc,lr
- .section__ex_table,"a"
- .long1b,__put_user_bad
- .long2b,__put_user_bad
- .long3b,__put_user_bad
- .long4b,__put_user_bad
- .long5b,__put_user_bad
- .long6b,__put_user_bad
- .previous
評(píng)論