成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

調試記錄 | Linux 內核靜態庫封裝問題

系統 Linux
對于靜態庫的封裝,大多數情況在應用層應用的封裝的比較多,用起來比較熟悉。不過,在嵌入式開發中,有些時候,需要將一些私有修改隱藏起來,特別是,內核中的一些修改。

[[410536]]

本文轉載自微信公眾號「漫談嵌入式」,作者Vinson 。轉載本文請聯系漫談嵌入式公眾號。

背景

對于靜態庫的封裝,大多數情況在應用層應用的封裝的比較多,用起來比較熟悉。不過,在嵌入式開發中,有些時候,需要將一些私有修改隱藏起來,特別是,內核中的一些修改。

此時需要在內核態制作靜態庫,然后鏈接到整個內核文件中。

對于一般(沒有復雜的內核依賴關系)的內核靜態庫的封裝,直接安裝應用層封裝即可。

對于內核中一些高級驅動的私有修改,在進行封裝時,就需要格外注意了,包括正確編譯,頭文件交叉引用,如果正確被鏈接到內核中,而不是被編譯器忽略掉了。

封裝問題

我們以 usb_f_uvc.ko 這個uvc function driver為例,來分析,內核靜態庫的封裝(假設,以下文件有修改或者定制)。最終,將usb_f_uvc.ko 打包成一個 靜態庫,鏈接到內核里面。

  1. # kernel/drivers/usb/gadget/function/Makefile 
  2. usb_f_uvc-y    := f_uvc.o uvc_queue.o uvc_v4l2.o uvc_video.o uvc_configfs.o 
  3. obj-$(CONFIG_USB_F_UVC)  += usb_f_uvc.o 

編譯

我們將需要的文件,復雜到一個目錄下,修改Makefile

  1. # Makefile 
  2.  
  3. # 可換成自己的工具鏈 
  4. CROSS_COMPILE ?= arm-linux-gnu-  
  5. CC := $(CROSS_COMPILE)gcc 
  6. LD := $(CROSS_COMPILE)ld 
  7. AR := $(CROSS_COMPILE)ar 
  8. CP := cp 
  9. RM := rm 
  10.  
  11. # 修改正確的kernel 路徑 
  12. KERNEL_PATH := xxxx/kerenl 
  13.  
  14. # 獲取gcc 版本 
  15. CC_PATH := ${shell which $(CC)} 
  16. CROSS_COMPILE_PATH := ${shell dirname $(CC_PATH)} 
  17.  
  18. CFLAGS := -nostdinc -isystem $(CROSS_COMPILE_PATH)/../lib/gcc/arm-linux-gnu/7.2.0/include 
  19.  
  20. # 頭文件順序很重要,換成自己平臺的 
  21. INCLUDE = -I$(KERNEL_PATH)/arch/arm/include \ 
  22.         -I$(KERNEL_PATH)/arch/arm/include/generated/uapi \ 
  23.         -I$(KERNEL_PATH)/arch/arm/include/generated \ 
  24.         -I$(KERNEL_PATH)/include \ 
  25.         -I$(KERNEL_PATH)/arch/arm/include/uapi \ 
  26.         -I$(KERNEL_PATH)/include/uapi \ 
  27.         -I$(KERNEL_PATH)/include/generated/uapi/ \ 
  28.         -include $(KERNEL_PATH)/include/linux/kconfig.h 
  29.  
  30. INCLUDE += -I$(KERNEL_PATH)/arch/arm/xxxx/core/include \ 
  31.         -I$(KERNEL_PATH)/arch/arm/xxxx/soc-xxx/include \ 
  32.         -I$(KERNEL_PATH)/arch/arm/include/asm/mach-generic 
  33.          
  34. #CFLAGS += -fno-delete-null-pointer-checks -Wno-maybe-uninitialized -Wno-frame-address -Wno-format-truncation \ 
  35.         #-Wno-format-overflow -Wno-int-in-bool-context -Os --param=allow-store-data-races=0 -DCC_HAVE_ASM_GOTO \ 
  36.         #-Wframe-larger-than=1024 -fno-stack-protector -Wno-unused-but-set-variable -Wno-unused-const-variable \ 
  37.         #-fomit-frame-pointer -fno-var-tracking-assignments -Wdeclaration-after-statement \ 
  38.         #-Wno-pointer-sign -fno-strict-overflow -fconserve-stack -Werror=implicit-int \ 
  39.         #-Werror=strict-prototypes -Werror=date-time 
  40.   
  41. CFLAGS += -DEXPORT_SYMTAB 
  42.  
  43. # 這個一定要加 
  44. CFLAGS += -D__KERNEL__  
  45.  
  46. CFLAGS += $(INCLUDE) 
  47.  
  48. OBJS := uvc_queue.o uvc_v4l2.o uvc_video.o f_uvc.o uvc_configfs.o 
  49.  
  50. ARFLAG := -rcs 
  51.  
  52. LIB_TARGET := libxxx.a 
  53. TARGET := libxxx.hex 
  54.  
  55. all: $(TARGET) 
  56.  
  57. %.o:%.c 
  58.         $(CC) $(CFLAGS) -o $@ -c $^ 
  59.  
  60. $(TARGET): $(LIB_TARGET) 
  61.         $(CP) $(LIB_TARGET) $(TARGET) 
  62.         $(CP) -vf $(TARGET) $(KERNEL_PATH)/drivers/usb/gadget/function
  63.  
  64. $(LIB_TARGET): $(OBJS) 
  65.         $(AR) $(ARFLAG) $@ $^ 
  66.  
  67. clean: 
  68.         find . -name "*.o" | xargs rm -r 
  69.         $(RM) -vf $(LIB_TARGET) $(TARGET) 
  70.  
  71. install: 
  72.         $(CP) -vf $(TARGET) $(KERNEL_PATH)/drivers/usb/gadget/function

Makefile 參數和頭文件如何來?

事實上,整個內核打包的過程,筆者認為,編譯是最難的一步,特別是第一次接觸的時候。

對于驅動中的各符號和宏的定義,以及頭文件包含是層層套娃,根據錯誤信息定位,簡直要崩潰。

在這里,筆者建議,先參考【內核編譯參數選項】,然后在逐一刪減無關選項,這樣會方便很多。

具體操作如下:

  • 正常編譯內核:
  • touch 修改 f_uvc.c:
  • 重新編譯內核:make uImage V=1 > build.txt
  • vim build.txt 搜索f_uvc 即可看到編譯信息

使用 make V=1 參數將編譯的詳細信息輸出,包括頭文件包含順序,gcc 編譯參數選項等,然后將其添加到我們的Makefie上。最后在對我們的Makfile 做刪減。

添加到內核

  1. #kernel/drivers/usb/gadget/function/Makefile 
  2. usb_f_uvc-y    := libxxx.a                                           
  3. #obj-$(CONFIG_USB_F_UVC)  += usb_f_uvc.o 
  4. obj-y += usb_f_uvc.o 
  5. # 防止Make distclean 把所有 .a都清掉了 
  6. $(obj)/libxxx.a: $(obj)/libxxx.hex 
  7.     cp $(obj)/libxxx.hex $(obj)/libxxx.a 

編譯內核

重新編譯內核,將.a 鏈接到內核。然后燒到板子運行。

運行

實際運行,發現根本沒有鏈到板子去。

原因分析

查看 EXPORT_SYMBOL

打開 Module.symvers 發現,uvc 相關的接口并沒有導出來,猜測有可能沒有成功鏈到內核。

  1. vim Module.symvers 

objdump 反匯編

使用objdump 將所有的符號表都輸出來,然后在搜索查看,進一步確認鏈接是否正確。結果發現也找不到任何符號信息

  1. arm-linux-gnu-objdump -Dz vmlinux > kernel.dump 

此時一個大膽的想法出現了,是否是被編譯器給優化掉了?因為是靜態庫,對于庫文件來說,其本身只是一些接口,自身不能執行調用過程。如果接口沒有人調用,那么所有相關的符號是否自動被忽略了?考慮一波對編譯鏈接的理解

分析源碼

  1. //f_uvc.c 
  2. DECLARE_USB_FUNCTION_INIT(uvc, uvc_alloc_inst, uvc_alloc); 
  3. MODULE_LICENSE("GPL"); 
  4. MODULE_AUTHOR("Laurent Pinchart"); 

這里的 DECLARE_USB_FUNCTION_INIT 很重要。我們,具體展開。

  1. #define DECLARE_USB_FUNCTION_INIT(_name, _inst_alloc, _func_alloc) \ 
  2.  DECLARE_USB_FUNCTION(_name, _inst_alloc, _func_alloc)  \ 
  3.  static int __init _name ## mod_init(void)   \ 
  4.  {        \ 
  5.   return usb_function_register(&_name ## usb_func); \ 
  6.  }        \ 
  7.  static void __exit _name ## mod_exit(void)   \ 
  8.  {        \ 
  9.   usb_function_unregister(&_name ## usb_func);  \ 
  10.  }        \ 
  11.  module_init(_name ## mod_init);     \ 
  12.  module_exit(_name ## mod_exit) 

這里看到 module_init 應該很熟悉了,對于我們上面封裝的庫來說,本質上也是一個驅動,是驅動就有對應的入口和出口。

對于內核,所有的入口都被放在 .text.init 處,加載到內核中后會按照相應順序進行初始化。

如果我們,把整個驅動封裝成一個靜態庫,DECLARE_USB_FUNCTION_INIT 屬于庫的接口,本身不會自己調用。所以內核在鏈接的過程中,發現沒有調用關系,就自然而然會忽略掉libxxx.a的相關符號。

知道了原因,解決方法就很簡單了。在內核中一定要存在有調用DECLARE_USB_FUNCTION_INIT的地方。

  • 方法1:手動調用。不推薦
  • 方法2:自動調用。沿用內核驅動模型。將 DECLARE_USB_FUNCTION_INIT 從靜態庫中剝離出來,其他文件打包成一個庫。

修改如下:

  1. // entry.c 
  2. #include <linux/kernel.h> 
  3. #include <linux/module.h> 
  4. #include <linux/device.h> 
  5. #include <linux/errno.h> 
  6. #include <linux/list.h> 
  7. #include <linux/mutex.h> 
  8. #include <linux/string.h> 
  9. #include <linux/usb/ch9.h> 
  10. #include <linux/usb/gadget.h> 
  11. #include <linux/usb/video.h> 
  12.  
  13. #include "u_uvc.h" 
  14. #include "f_uvc.h" 
  15.  
  16. static struct usb_function_instance *uvc_alloc_inst(void) 
  17.     return uvc_alloc_inst_callback(); 
  18.  
  19. static struct usb_function *uvc_alloc(struct usb_function_instance *fi) 
  20.     return uvc_alloc_callback(fi); 
  21.  
  22. DECLARE_USB_FUNCTION_INIT(uvc, uvc_alloc_inst, uvc_alloc); 
  23. MODULE_LICENSE("GPL"); 
  24. MODULE_AUTHOR("Laurent Pinchart"); 

重新修改Makefile

  1. usb_f_uvc-y   := entry.o libxxx.a 
  2. obj-y  += usb_f_uvc.o 
  3.  
  4. #obj-$(CONFIG_USB_F_UVC) += usb_f_uvc.o 
  5.  
  6. $(obj)/libxxx.a: $(obj)/libxxx.hex 
  7.     cp $(obj)/libxxx.hex $(obj)/libxxx.a 

這樣重新,編譯內核,就可以用了。以后只需要更新libxxx.a 即可。

總結

本文簡單介紹內核靜態庫,打包遇到的一些坑。通過一個例子,介紹內核靜態庫的封裝,以及遇到的問題。

同時也加深了對編譯和鏈接的理解。有關應用層靜態庫和內核態的庫在使用上是一樣的,不過在制作時有些許麻煩。

  • 頭文件的引用包含
  • 編譯參數選項
  • 是否成功鏈接

 

有關驅動入口的部分,不能做到庫里面,避免踩雷。折騰其他,結果發現是鏈接時出了問題。

 

責任編輯:武曉燕 來源: 漫談嵌入式
相關推薦

2010-01-22 11:01:04

linux內核模塊

2014-08-28 15:08:35

Linux內核

2021-11-14 07:29:55

Linux 內核靜態追蹤Linux 系統

2011-08-10 15:36:26

iPhone靜態庫控件

2022-02-08 15:15:26

OpenHarmonlinux鴻蒙

2017-01-12 19:15:03

Linux內核調試自構proc

2010-01-07 17:36:38

Linux靜態庫

2023-04-10 09:44:22

內核鼠標調試鴻蒙

2016-08-23 09:17:08

LinuxD狀態TASK_RUNNIN

2010-03-04 10:17:57

Linux動態庫

2016-09-19 10:54:36

C語言靜態連接語言

2016-10-28 09:18:47

Linux內核代碼

2021-02-20 06:08:07

LinuxWindows內核

2021-11-02 09:55:57

Linux內核內存

2022-07-12 13:23:59

靜態鏈接庫可執行文件C 目標文件

2012-07-31 16:06:28

Linux內核編譯

2019-04-12 08:10:33

iOS靜態分析Xcode

2011-06-29 17:00:26

QT 靜態編譯 Debug

2010-03-02 09:17:32

Linux local

2010-03-08 15:28:31

Linux內核安全
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 亚洲成人在线免费 | 精品国产精品三级精品av网址 | 国产一级片免费看 | 国产精品久久一区二区三区 | 日韩欧美精品 | 天天草视频 | 国产精品视频网站 | 午夜精品久久久久久久久久久久久 | 国产精品一区二 | 在线免费观看视频你懂的 | 婷婷精品| 国产精品免费一区二区三区四区 | 日韩精品一区二区三区高清免费 | 激情综合五月 | 国产黑丝在线 | 亚洲福利在线视频 | 成人免费在线观看视频 | 91影院在线观看 | 欧美性吧 | 日韩免 | 最近中文字幕在线视频1 | 亚洲男女视频在线观看 | 亚洲免费一区二区 | 日本成人午夜影院 | 国产视频精品在线观看 | 日韩欧美国产精品一区二区 | 亚洲一区 中文字幕 | 久久久久亚洲 | 一色一黄视频 | 日本午夜免费福利视频 | av中文字幕在线观看 | 爱操av | 华丽的挑战在线观看 | 成人深夜福利 | 色婷婷av一区二区三区软件 | 亚洲网站在线观看 | 欧美久久一区二区三区 | www.青娱乐| 欧美日韩专区 | 精品在线一区 | 久久久久国产一区二区三区 |