微波EDA网,见证研发工程师的成长!
首页 > 硬件设计 > 嵌入式设计 > ARM-Linux s3c2440 之UART分析(三)

ARM-Linux s3c2440 之UART分析(三)

时间:11-19 来源:互联网 点击:
回顾一下上文,s3c2440串口底层驱动围绕三个数据结构展开:

UART特定的驱动程序结构定义:struct uart_driver s3c24xx_uart_drv;

UART端口结构定义: struct uart_port s3c24xx_serial_ops;

UART相关操作函数结构定义: struct uart_ops s3c24xx_serial_ops;

实现了这三个数据结构体,基本完成了驱动操作函数的实现,紧接着需要对串口设备及设备驱动进行初始化,首先是模块初始化module_init(s3c24xx_serial_modinit):

  1. staticint__inits3c24xx_serial_modinit(void)
  2. {
  3. intret;
  4. ret=uart_register_driver(&s3c24xx_uart_drv);//注册uart驱动,在serial_core.c中实现
  5. if(ret<0){
  6. printk(KERN_ERR"failedtoregisterUARTdriver\n");
  7. return-1;
  8. }
  9. return0;
  10. }

uart_register_driver在串口核心及TTY层间进行相关注册:

  1. intuart_register_driver(structuart_driver*drv)
  2. {
  3. structtty_driver*normal=NULL;
  4. drv->tty_driver=normal;
  5. normal->owner=drv->owner;
  6. normal->driver_name=drv->driver_name;
  7. normal->name=drv->dev_name;
  8. normal->major=drv->major;
  9. normal->minor_start=drv->minor;
  10. normal->type=TTY_DRIVER_TYPE_SERIAL;
  11. normal->subtype=SERIAL_TYPE_NORMAL;
  12. normal->init_termios=tty_std_termios;
  13. normal->init_termios.c_cflag=B9600|CS8|CREAD|HUPCL|CLOCAL;
  14. normal->init_termios.c_ispeed=normal->init_termios.c_ospeed=9600;
  15. normal->flags=TTY_DRIVER_REAL_RAW|TTY_DRIVER_DYNAMIC_DEV;
  16. normal->driver_state=drv;
  17. tty_set_operations(normal,&uart_ops);
  18. ….
  19. retval=tty_register_driver(normal);//tty驱动注册
  20. out:
  21. if(retval<0){
  22. put_tty_driver(normal);
  23. kfree(drv->state);
  24. }
  25. returnretval;
  26. }

接着进行总线驱动platform_driver_register:

  1. ints3c24xx_serial_init(structplatform_driver*drv,
  2. structs3c24xx_uart_info*info)
  3. {
  4. #ifdefCONFIG_PM
  5. drv->suspend=s3c24xx_serial_suspend;
  6. drv->resume=s3c24xx_serial_resume;
  7. #endif
  8. returnplatform_driver_register(drv);//
  9. }

其中s3c24xx_serial_init为__init段,模块加载时初始化:

  1. staticint__inits3c2440_serial_init(void)
  2. {
  3. returns3c24xx_serial_init(&s3c2440_serial_drv,&s3c2440_uart_inf);//
  4. }

platfrom_driver_register()中调用driver_register()

  1. intplatform_driver_register(structplatform_driver*drv)
  2. {
  3. drv->driver.bus=&platform_bus_type;//
  4. if(drv->probe)
  5. drv->driver.probe=platform_drv_probe;
  6. if(drv->remove)
  7. drv->driver.remove=platform_drv_remove;
  8. if(drv->shutdown)
  9. drv->driver.shutdown=platform_drv_shutdown;
  10. if(drv->suspend)
  11. drv->driver.suspend=platform_drv_suspend;
  12. if(drv->resume)
  13. drv->driver.resume=platform_drv_resume;
  14. returndriver_register(&drv->driver);//
  15. }

因为串口设备在系统是一种platform_device所以是一种总线驱动类型,总线设备驱动注册platform_driver_register()之后,串口设备即可和相应的驱动关联起来了,这样就完成了串口设备与串口驱动的注册过程。

下一篇讲进一步结合源码分析。

Copyright © 2017-2020 微波EDA网 版权所有

网站地图

Top