函数fb_find_logo实现在文件kernel/goldfish/drivers/video/logo/logo.c文件中,如下所示:

  1. extern const struct linux_logo logo_linux_mono;  
  2. extern const struct linux_logo logo_linux_vga16;  
  3. extern const struct linux_logo logo_linux_clut224;  
  4. extern const struct linux_logo logo_blackfin_vga16;  
  5. extern const struct linux_logo logo_blackfin_clut224;  
  6. extern const struct linux_logo logo_dec_clut224;  
  7. extern const struct linux_logo logo_mac_clut224;  
  8. extern const struct linux_logo logo_parisc_clut224;  
  9. extern const struct linux_logo logo_sgi_clut224;  
  10. extern const struct linux_logo logo_sun_clut224;  
  11. extern const struct linux_logo logo_superh_mono;  
  12. extern const struct linux_logo logo_superh_vga16;  
  13. extern const struct linux_logo logo_superh_clut224;  
  14. extern const struct linux_logo logo_m32r_clut224;  
  15.   
  16. static int nologo;  
  17. module_param(nologo, bool, 0);  
  18. MODULE_PARM_DESC(nologo, "Disables startup logo");  
  19.   
  20. /* logo's are marked __initdata. Use __init_refok to tell 
  21.  * modpost that it is intended that this function uses data 
  22.  * marked __initdata. 
  23.  */  
  24. const struct linux_logo * __init_refok fb_find_logo(int depth)  
  25. {  
  26.         const struct linux_logo *logo = NULL;  
  27.   
  28.         if (nologo)  
  29.                 return NULL;  
  30.   
  31.         if (depth >= 1) {  
  32. #ifdef CONFIG_LOGO_LINUX_MONO  
  33.                 /* Generic Linux logo */  
  34.                 logo = &logo_linux_mono;  
  35. #endif  
  36. #ifdef CONFIG_LOGO_SUPERH_MONO  
  37.                 /* SuperH Linux logo */  
  38.                 logo = &logo_superh_mono;  
  39. #endif  
  40.         }  
  41.   
  42.         if (depth >= 4) {  
  43. #ifdef CONFIG_LOGO_LINUX_VGA16  
  44.                 /* Generic Linux logo */  
  45.                 logo = &logo_linux_vga16;  
  46. #endif  
  47. #ifdef CONFIG_LOGO_BLACKFIN_VGA16  
  48.                 /* Blackfin processor logo */  
  49.                 logo = &logo_blackfin_vga16;  
  50. #endif  
  51. #ifdef CONFIG_LOGO_SUPERH_VGA16  
  52.                 /* SuperH Linux logo */  
  53.                 logo = &logo_superh_vga16;  
  54. #endif  
  55.         }  
  56.   
  57.         if (depth >= 8) {  
  58. #ifdef CONFIG_LOGO_LINUX_CLUT224  
  59.                 /* Generic Linux logo */  
  60.                 logo = &logo_linux_clut224;  
  61. #endif  
  62. #ifdef CONFIG_LOGO_BLACKFIN_CLUT224  
  63.                 /* Blackfin Linux logo */  
  64.                 logo = &logo_blackfin_clut224;  
  65. #endif  
  66. #ifdef CONFIG_LOGO_DEC_CLUT224  
  67.                 /* DEC Linux logo on MIPS/MIPS64 or ALPHA */  
  68.                 logo = &logo_dec_clut224;  
  69. #endif  
  70. #ifdef CONFIG_LOGO_MAC_CLUT224  
  71.                 /* Macintosh Linux logo on m68k */  
  72.                 if (MACH_IS_MAC)  
  73.                         logo = &logo_mac_clut224;  
  74. #endif  
  75. #ifdef CONFIG_LOGO_PARISC_CLUT224  
  76.                 /* PA-RISC Linux logo */  
  77.                 logo = &logo_parisc_clut224;  
  78. #endif  
  79. #ifdef CONFIG_LOGO_SGI_CLUT224  
  80.                 /* SGI Linux logo on MIPS/MIPS64 and VISWS */  
  81.                 logo = &logo_sgi_clut224;  
  82. #endif  
  83. #ifdef CONFIG_LOGO_SUN_CLUT224  
  84.                 /* Sun Linux logo */  
  85.                 logo = &logo_sun_clut224;  
  86. #endif  
  87. #ifdef CONFIG_LOGO_SUPERH_CLUT224  
  88.                 /* SuperH Linux logo */  
  89.                 logo = &logo_superh_clut224;  
  90. #endif  
  91. #ifdef CONFIG_LOGO_M32R_CLUT224  
  92.                 /* M32R Linux logo */  
  93.                 logo = &logo_m32r_clut224;  
  94. #endif  
  95.         }  
  96.         return logo;  
  97. }  
  98. EXPORT_SYMBOL_GPL(fb_find_logo);  
        文件开始声明的一系列linux_logo结构体变量分别用来保存kernel/goldfish/drivers/video/logo目录下的一系列ppm或者pbm文件的内容的。这些ppm或者pbm文件都是用来描述第一个开机画面的。
 
        全局变量nologo是一个类型为布尔变量的模块参数,它的默认值等于0,表示要显示第一个开机画面。在这种情况下,函数fb_find_logo就会根据参数depth的值以及不同的编译选项来选择第一个开机画面的内容,并且保存在变量logo中返回给调用者。
        这一步执行完成之后,第一个开机画面的内容就保存在模块fbmem的全局变量fb_logo的成员变量logo中了。这时候控制台的初始化过程也结束了,接下来系统就会执行切换控制台的操作。前面提到,当系统执行切换控制台的操作的时候,模块fbcon中的函数fbcon_switch就会被调用。在调用的过程中,就会执行显示第一个开机画面的操作。 
  1. static int fbcon_switch(struct vc_data *vc)  
  2. {  
  3.         struct fb_info *info, *old_info = NULL;  
  4.         struct fbcon_ops *ops;  
  5.         struct display *p = &fb_display[vc->vc_num];  
  6.         struct fb_var_screeninfo var;  
  7.         int i, prev_console, charcnt = 256;  
  8.   
  9.         ......  
  10.   
  11.         if (logo_shown == FBCON_LOGO_DRAW) {  
  12.                 logo_shown = fg_console;  
  13.                 /* This is protected above by initmem_freed */  
  14.                 fb_show_logo(info, ops->rotate);  
  15.                 ......  
  16.                 return 0;  
  17.         }  
  18.         return 1;  
  19. }  
        由于前面在准备第一个开机画面的内容的时候,全局变量logo_show的值被设置为FBCON_LOGO_DRAW,因此,接下来就会调用函数fb_show_logo来显示第一个开机画面。在显示之前,这个函数会将全局变量logo_shown的值设置为fg_console,后者表示系统当前可见的控制台的编号。
 
        函数fb_show_logo实现在文件kernel/goldfish/drivers/video/fbmem.c中,如下所示:
  1. int fb_show_logo(struct fb_info *info, int rotate)  
  2. {  
  3.         int y;  
  4.   
  5.         y = fb_show_logo_line(info, rotate, fb_logo.logo, 0,  
  6.                               num_online_cpus());  
  7.         ......  
  8.   
  9.         return y;  
  10. }  
       这个函数调用另外一个函数fb_show_logo_line来进一步执行渲染第一个开机画面的操作。