从printk到framebuff

源地址:http://blog.sina.com.cn/s/blog_633f46290100k4yw.html

在linux内核的使用framebuff的vga显示中,有两种方式(可能不止,但笔者能力所限,仅知道两种)向屏幕输出,一种是通过printk向>屏幕输出,一种是通过向tty文件使用文件操作书写。

1:从printk到vc_cons
printk  --------------------------------------kernel/printk.c
printk处理完参数后调用vprintk
vprintk --------------------------------------kernel/printk.c
vprintk调用emit_log_char 将字符填充进log_buff中,并调用release_console_sem()将字符打印出
release_console_sem --------------------------kernel/printk.c
函数的本意是释放信号量,但同时调用了call_console_drivers(_con_start, _log_end)将log_buff中的字符打印出。
call_console_drivers ------------------------kernel/printk.c
调用_call_console_drivers打印字符
_call_console_drivers-------------------------kernel/printk.c
调用__call_console_drivers打印字符
__call_console_drivers------------------------kernel/printk.c
__call_console_drivers中语句如下:

 436     for (con = console_drivers; con; con = con->next) {

 437         if ((con->flags & CON_ENABLED) && con->write &&

 438                 (cpu_online(smp_processor_id()) ||

 439                 (con->flags & CON_ANYTIME)))

 440             con->write(con, &LOG_BUF(start), end - start);

 441     }

 

可见其以此调用的是注册的console中的write函数。
不妨以driver/char/vt.c中注册的vt_console_driver,其对应的write函数是vt_console_print
vt_console_print --------------------------driver/char/vt.c
vt_console_print 最终调用的是vc_cons[fg_console].d->vc_sw->con_putcs(vc, start, cnt, vc->vc_y, vc->vc_x)显示字符。

因此由printk可以追溯到vc_data结构体数组vc_cons中某个的成员vc_sw->con_putcs函数。


2:从tty到vc_cons
在写入tty设备文件时调用的操作是tty_fops(drivers/char/tty_io.c)
 

 810 static const struct file_operations tty_fops = {

 811     .llseek     = no_llseek,

 812     .read       = tty_read,

 813     .write      = tty_write,

 814     .poll       = tty_poll,

 815     .unlocked_ioctl = tty_ioctl,

 816     .compat_ioctl   = tty_compat_ioctl,

 817     .open       = tty_open,

 818     .release    = tty_release,

 819     .fasync     = tty_fasync,

 820 };

 

在tty_open中选择tty_driver (具体请见偶以前书写的 console,tty,keyboard关系的分析)

2234     retval = init_dev(driver, index, &tty);

的语句,使得tty_driver于tty相关联。

通过一个具体的tty_driver的注册例子来表示
在drivers/char/vt.c的vty_init函数中

2965     console_driver->owner = THIS_MODULE;

2966     console_driver->name = "tty";

2967     console_driver->name_base = 1;

2968     console_driver->major = TTY_MAJOR;

2969     console_driver->minor_start = 1;

2970     console_driver->type = TTY_DRIVER_TYPE_CONSOLE;

2971     console_driver->init_termios = tty_std_termios;

2972     if (default_utf8)

2973         console_driver->init_termios.c_iflag |= IUTF8;

2974     console_driver->flags = TTY_DRIVER_REAL_RAW | TTY_DRIVER_RESET_TERMIOS;

2975     tty_set_operations(console_driver, &con_ops);

2976     if (tty_register_driver(console_driver))

2977         panic("Couldn't register console driver\n");

 

其中tty_set_operations(console_driver, &con_ops);是设置tty_driver使用的一系列操作,con_ops是tty_operations类型,tty_register_driver(console_driver)即是注册tty_driver

其中注册的con_ops如下所示

2939 static const struct tty_operations con_ops = {

2940     .open = con_open,

2941     .close = con_close,

2942     .write = con_write,

2943     .write_room = con_write_room,

2944     .put_char = con_put_char,

2945     .flush_chars = con_flush_chars,

2946     .chars_in_buffer = con_chars_in_buffer,

2947     .ioctl = vt_ioctl,

2948     .stop = con_stop,

2949     .start = con_start,

2950     .throttle = con_throttle,

2951     .unthrottle = con_unthrottle,

2952     .resize = vt_resize,

2953 };

 

 

因此对tty设备文件写操作使用的是con_write
con_write -----------------------------------------------driver/char/vt.c
con_write调用do_con_write
do_con_write --------------------------------------------driver/char/vt.c
do_con_write中定义的有宏

2096 #define FLUSH if (draw_x >= 0) { \

2097     vc->vc_sw->con_putcs(vc, (u16 *)draw_from, (u16 *)draw_to - (u16 *)draw_from, vc->vc_y, draw_x); \

2098     draw_x = -1; \

2099     }

2100 #endif

 

因此其调用的也是vc_data结构体数组vc_cons中某个的成员vc_sw->con_putcs函数。


综上可知,使用printk和对tty文件写最后均调用到vc_cons中某个的成员vc_sw->con_putcs函数。


3:从vc_cons 到framebuff
vc_cons[]是在con_init中初始化
con_init -----------------------------------------------drivers/char/vt.c

2908     for (currcons = 0; currcons < MIN_NR_CONSOLES; currcons++) {

2909         vc_cons[currcons].d = vc = alloc_bootmem(sizeof(struct vc_data));

2910         INIT_WORK(&vc_cons[currcons].SAK_work, vc_SAK);

2911         visual_init(vc, currcons, 1);

2912         vc->vc_screenbuf = (unsigned short *)alloc_bootmem(vc->vc_screenbuf_size);

2913         vc->vc_kmalloced = 0;

2914         vc_init(vc, vc->vc_rows, vc->vc_cols,

2915             currcons || !vc->vc_sw->con_save_screen);

2916     }

 

其中visual_init 包含对vc_sw的初始化
visual_init -------------------------------------------drivers/char/vt.c
函数中有
 

724     vc->vc_sw = conswitchp;

 

可见vc_sw来自conswitchp全局变量。

bind_con_driver中有对conswitchp的赋值
bind_con_driver----------------------------------------drivers/char/vt.c

2992 static int bind_con_driver(const struct consw *csw, int first, int last,

2993                int deflt)

3029         conswitchp = csw;

 

故conswitchp的值与其调用者有关

bind_con_driver 被vt_bind调用
vt_bind -------------------------------------------------driver/char/vt.c
vt_bind被store_bind调用
store_bind ----------------------------------------------driver/char/vt.c
其中有

3317     struct con_driver *con = dev_get_drvdata(dev);

3320     if (bind)

3321         vt_bind(con);

故conswitchp来自dev_get_drvdata(dev)

而在vtconsole_init_device有

3359     dev_set_drvdata(con->dev, con);

 

因此conswitchp来自vtconsole_init_device的调用者

vtconsole_init_device虽有两处调用 但归根揭底是到register_con_driver调用
register_con_driver -----------------------------------driver/char/vt.c
其中有语句

3420 int register_con_driver(const struct consw *csw, int first, int last)

 

故conswitchp来自register_con_driver调用者

register_con_driver由take_over_console调用
take_over_console-------------------drivers/char/vt.c

在drivers/video/console/fbcon.c中fb_takeover调用了take_over_console
fb_takeover -------------------------------------------drivers/char/vt.c
其中语句:

 581     err = take_over_console(&fb_con, first_fb_vc, last_fb_vc,

 582                 fbcon_is_default);

 

因此conswitchp来自fb_con.

fbcon 定义如下

3320 static const struct consw fb_con = { -----------------drivers/char/vt.c

3321     .owner          = THIS_MODULE,

3322     .con_startup        = fbcon_startup,

3323     .con_init       = fbcon_init,

3324     .con_deinit         = fbcon_deinit,

3325     .con_clear      = fbcon_clear,

3326     .con_putc       = fbcon_putc,

3327     .con_putcs      = fbcon_putcs,

3328     .con_cursor         = fbcon_cursor,

3329     .con_scroll         = fbcon_scroll,

3330     .con_bmove      = fbcon_bmove,

3331     .con_switch         = fbcon_switch,

3332     .con_blank      = fbcon_blank,

3333     .con_font_set       = fbcon_set_font,

3334     .con_font_get       = fbcon_get_font,

3335     .con_font_default   = fbcon_set_def_font,

3336     .con_font_copy      = fbcon_copy_font,

3337     .con_set_palette    = fbcon_set_palette,

3338     .con_scrolldelta    = fbcon_scrolldelta,

3339     .con_set_origin     = fbcon_set_origin,

3340     .con_invert_region  = fbcon_invert_region,

3341     .con_screen_pos     = fbcon_screen_pos,

3342     .con_getxy      = fbcon_getxy,

3343     .con_resize             = fbcon_resize,

3344 };

 

 

因此vc_con[]中变量的成员变量vc_sw->con_putcs 在framebuff中对应的是fbcon_putcs
fbcon_putcs ------------------------------------------drivers/video/console/fbcon.c
代码如下:

1332     struct fb_info *info = registered_fb[con2fb_map[vc->vc_num]];

1333     struct display *p = &fb_display[vc->vc_num];

1334     struct fbcon_ops *ops = info->fbcon_par;

1335

1336     if (!fbcon_is_inactive(vc, info))

1337         ops->putcs(vc, info, s, count, real_y(p, ypos), xpos,

1338                get_color(vc, info, scr_readw(s), 1),

1339                get_color(vc, info, scr_readw(s), 0));

 

 

可见其具体调用的是其注册的register_fb中的putcs

由此完成了从printk到framebuff 从tty到framebuff的调用查询

你可能感兴趣的:(frame)