一. 逻辑架构文章来源:http://www.itnose.net/detail/6036596.html更多文章:http://www.itnose.net/type/110.html
虽然感觉上面向过程的程序设计好像更加适合提高驱动的效率,但如果愿意牺牲一点效率采用一些面向对象的设计方法对于驱动新手是件比较舒服的事。
在这次的RTC编程里,为了使结构更加清晰,将整个驱动结构分成了三层,Hardware, Software, rtc.
Hardware层负责寄存器操作,屏蔽具体的寄存器地址信息。
/** hardware operation*/ struct tagHardware; typedef bool (*REG_OPS_R)( struct tagHardware *hthis, uint8 *val); typedef bool (*REG_OPS_W)( struct tagHardware *hthis, uint8 val); struct tagHardware { /** resource*/ struct resource *res; struct resource *res_mem; void __iomem *base; /** operation*/ //control register REG_OPS_R funcCon_r; REG_OPS_W funcCon_w; REG_OPS_R funcTicnt_r; REG_OPS_W funcTicnt_w; REG_OPS_R funcAlm_r; REG_OPS_W funcAlm_w; //time register REG_OPS_R funcSec_r; REG_OPS_W funcSec_w; REG_OPS_R funcMin_r; REG_OPS_W funcMin_w; REG_OPS_R funcHour_r; REG_OPS_W funcHour_w; REG_OPS_R funcDay_r; REG_OPS_W funcDay_w; REG_OPS_R funcDate_r; REG_OPS_W funcDate_w; REG_OPS_R funcMon_r; REG_OPS_W funcMon_w; REG_OPS_R funcYear_r; REG_OPS_W funcYear_w; //alarm register REG_OPS_R funcASec_r; REG_OPS_W funcASec_w; REG_OPS_R funcAMin_r; REG_OPS_W funcAMin_w; REG_OPS_R funcAHour_r; REG_OPS_W funcAHour_w; REG_OPS_R funcADate_r; REG_OPS_W funcADate_w; REG_OPS_R funcAMon_r; REG_OPS_W funcAMon_w; REG_OPS_R funcAYear_r; REG_OPS_W funcAYear_w; };
Software层负责将寄存器操作组合成有实际意义的操作,例如打开RTCEN使能,设置tick时钟频率。
struct tagSoftware; typedef bool (*BIT_OPS)( struct tagSoftware *sthis, bool isTure); typedef bool (*SETFREQ)( struct tagSoftware *sthis, int freq); typedef bool (*READTIM)( struct tagSoftware *sthis, struct rtc_time *buf); typedef bool (*SETTIM)( struct tagSoftware *sthis, const struct rtc_time *buf); typedef bool (*SETALMEN)( struct tagSoftware *sthis, uint8 *mask); /** basic operation*/ struct tagSoftware { /** resource*/ struct tagHardware *Hardware; struct resource *res_tickno; struct resource *res_alarmno; /** public function*/ BIT_OPS funcRtcen; //总开关,在写时需要打开 BIT_OPS funcClksel; //default as 0 BIT_OPS funcCntsel; //default as 0 BIT_OPS funcClkrst; //RTC clock count reset //BIT_OPS funcInqIRQ_Tick; //enable/disable the irq of tick //BIT_OPS funcInqIRQ_Alarm; //enable/disable the irq of alarm BIT_OPS funcInt_en; //tick time interrupt enable SETFREQ funcSetFreq; //0~127 SETALMEN funcGetAlmEn; BIT_OPS funcAlm_en; BIT_OPS funcYear_en; BIT_OPS funcMon_en; BIT_OPS funcDate_en; BIT_OPS funcHour_en; BIT_OPS funcMin_en; BIT_OPS funcSec_en; READTIM funcGetTime; SETTIM funcSetTime; READTIM funcGetAlm; SETTIM funcSetAlm; };
在rtc处再负责将一些太过琐碎soft的操作组合成较容易理解的操作,到这一步就已经屏蔽了硬件的具体地址细节,之后就可以专心的实现硬件的具体逻辑细节(比如怎么设置值,怎么使能)。
struct tagRTC_Driver; typedef bool (*RTC_EN)( struct tagRTC_Driver *rthis, bool isTrue); typedef bool (*RTC_TIME_OPS)( struct tagRTC_Driver *rthis, struct rtc_time *buf); typedef bool (*RTC_SETTICKEN)( struct tagRTC_Driver *rthis, bool isTrue); typedef bool (*RTC_SETTICKFREQ)( struct tagRTC_Driver *rthis, int freq); typedef bool (*RTC_SETALARMEN)( struct tagRTC_Driver *rthis, uint8 mask); typedef bool (*RTC_GETALARMEN)( struct tagRTC_Driver *rthid, uint8 *mask); struct tagRTC_Driver { //user operation /** resources*/ struct cdev cdev; dev_t devNum; struct semaphore sem; struct class *pClas; struct device *pDev; struct fasync_struct *async_queue; struct tagSoftware *soft_ops; /** public function*/ RTC_EN funcEn; RTC_TIME_OPS funcGetTime; RTC_TIME_OPS funcSetTime; RTC_TIME_OPS funcGetAlarm; RTC_TIME_OPS funcSetAlarm; RTC_SETTICKEN funcSetTickEn; RTC_SETTICKFREQ funcSetTickFreq; RTC_SETALARMEN funcSetAlarmEn; RTC_GETALARMEN funcGetAlarmEn; };
上面的RTC层提供的函数为操作硬件提供了较友好的接口,现在就可以专注内核接口的设计。
二. 一些知识要点
2.1 异步通知异步通知的实现原理较简单,分别从user 和 kernel两条线来看。
从kernel线,凡是申请了异步通知的进程会被登记到一个队列中。当触发条件到达时,就根据队列中的记录信息通知相应的进程。登记信息用到的函数为
fasync_helper(int fd,struct file * filp,int on,struct fasync_struct * * fapp)
根据队列信息通知用户层用到的函数为
kill_fasync(struct fasync_struct * * fp,int sig,int band)
从user线,一个驱动的通知信息只能到达设备文件,一个进程如果要获取通知信息需要使驱动知道自己,这可以通过 fcntl()实现,以及开通自己在这个文件下的异步通知功能。一般这样
fcntl( this->fd, F_SETOWN, getpid()); oflags = fcntl( this->fd, F_GETFL); fcntl( this->fd, F_SETFL, oflags|FASYNC);
2.2 设备添加
在设备模型里面,设备--总线--驱动 三个构成了实际的功能。如果驱动依照这个模型来设计,那么还需要实现的是将设备挂载到总线的设备列表中。关于添加方式,在
http://blog.csdn.net/u012301943/article/details/23095215
已经分析过了自己的思路。
三. RTC驱动代码
3.1 driver//when a tick-irq or alarm-irq arrived, a asynchronous signal ,SIGIO, will be send to user. //a message package , record all information about this interrupt,will be ready for user untill next // interrupt arrive. At this moment, you should read this package immediately to ensure it wouldn't // be lost. #include <linux/module.h> #include <linux/kernel.h> #include <linux/init.h> #include <linux/fs.h> #include <linux/cdev.h> #include <asm/uaccess.h> #include <linux/device.h> #include <linux/platform_device.h> #include <asm/io.h> #include <linux/interrupt.h> #include <linux/module.h> #include <linux/fs.h> #include <linux/string.h> #include <linux/init.h> #include <linux/platform_device.h> #include <linux/interrupt.h> #include <linux/bcd.h> #include <linux/clk.h> #include <mach/hardware.h> #include <asm/irq.h> #include "TQ_rtc_drv2.h" typedef unsigned char uint8; /** relative address*/ #define RTC_BASE_ADDR 0x57000040 //for ensure relative address. #define RTC_END_ADDR 0x5700008b #define RELA_ADDR(x) ( x -RTC_BASE_ADDR) #define RTCCON_ADDR RELA_ADDR(0x57000040) #define TICNT_ADDR RELA_ADDR(0x57000044) #define RTCALM_ADDR RELA_ADDR(0x57000050) #define ALMSEC_ADDR RELA_ADDR(0x57000054) #define ALMMIN_ADDR RELA_ADDR(0x57000058) #define ALMHOUR_ADDR RELA_ADDR(0x5700005c) #define ALMDATE_ADDR RELA_ADDR(0x57000060) #define ALMMON_ADDR RELA_ADDR(0x57000064) #define ALMYEAR_ADDR RELA_ADDR(0x57000068) #define BCDSEC_ADDR RELA_ADDR(0x57000070) #define BCDMIN_ADDR RELA_ADDR(0x57000074) #define BCDHOUR_ADDR RELA_ADDR(0x57000078) #define BCDDATE_ADDR RELA_ADDR(0x5700007c) #define BCDDAY_ADDR RELA_ADDR(0x57000080) #define BCDMON_ADDR RELA_ADDR(0x57000084) #define BCDYEAR_ADDR RELA_ADDR(0x57000088) /*****************************************************************/ //state : the entire software was divided into three parts: Hardware, Software, // and rtc( user operation) // //Hardware : basic register operation //Software : some basic operation for user //rtc : complex operation // /*****************************************************************/ /** hardware operation*/ struct tagHardware; typedef bool (*REG_OPS_R)( struct tagHardware *hthis, uint8 *val); typedef bool (*REG_OPS_W)( struct tagHardware *hthis, uint8 val); struct tagHardware { /** resource*/ struct resource *res; struct resource *res_mem; void __iomem *base; /** operation*/ //control register REG_OPS_R funcCon_r; REG_OPS_W funcCon_w; REG_OPS_R funcTicnt_r; REG_OPS_W funcTicnt_w; REG_OPS_R funcAlm_r; REG_OPS_W funcAlm_w; //time register REG_OPS_R funcSec_r; REG_OPS_W funcSec_w; REG_OPS_R funcMin_r; REG_OPS_W funcMin_w; REG_OPS_R funcHour_r; REG_OPS_W funcHour_w; REG_OPS_R funcDay_r; REG_OPS_W funcDay_w; REG_OPS_R funcDate_r; REG_OPS_W funcDate_w; REG_OPS_R funcMon_r; REG_OPS_W funcMon_w; REG_OPS_R funcYear_r; REG_OPS_W funcYear_w; //alarm register REG_OPS_R funcASec_r; REG_OPS_W funcASec_w; REG_OPS_R funcAMin_r; REG_OPS_W funcAMin_w; REG_OPS_R funcAHour_r; REG_OPS_W funcAHour_w; REG_OPS_R funcADate_r; REG_OPS_W funcADate_w; REG_OPS_R funcAMon_r; REG_OPS_W funcAMon_w; REG_OPS_R funcAYear_r; REG_OPS_W funcAYear_w; }; struct tagSoftware; typedef bool (*BIT_OPS)( struct tagSoftware *sthis, bool isTure); typedef bool (*SETFREQ)( struct tagSoftware *sthis, int freq); typedef bool (*READTIM)( struct tagSoftware *sthis, struct rtc_time *buf); typedef bool (*SETTIM)( struct tagSoftware *sthis, const struct rtc_time *buf); typedef bool (*SETALMEN)( struct tagSoftware *sthis, uint8 *mask); /** basic operation*/ struct tagSoftware { /** resource*/ struct tagHardware *Hardware; struct resource *res_tickno; struct resource *res_alarmno; /** public function*/ BIT_OPS funcRtcen; //总开关,在写时需要打开 BIT_OPS funcClksel; //default as 0 BIT_OPS funcCntsel; //default as 0 BIT_OPS funcClkrst; //RTC clock count reset //BIT_OPS funcInqIRQ_Tick; //enable/disable the irq of tick //BIT_OPS funcInqIRQ_Alarm; //enable/disable the irq of alarm BIT_OPS funcInt_en; //tick time interrupt enable SETFREQ funcSetFreq; //0~127 SETALMEN funcGetAlmEn; BIT_OPS funcAlm_en; BIT_OPS funcYear_en; BIT_OPS funcMon_en; BIT_OPS funcDate_en; BIT_OPS funcHour_en; BIT_OPS funcMin_en; BIT_OPS funcSec_en; READTIM funcGetTime; SETTIM funcSetTime; READTIM funcGetAlm; SETTIM funcSetAlm; }; static bool hard_rRTCCON( struct tagHardware *hthis, uint8 *val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; *val &=0; *val = readb( hthis->base + RTCCON_ADDR); return true; } static bool hard_wRTCCON( struct tagHardware *hthis, uint8 val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; writeb( val, hthis->base + RTCCON_ADDR); return true; } static bool hard_rTICNT( struct tagHardware *hthis, uint8 *val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; *val &= 0; *val = readb( hthis->base + TICNT_ADDR); return true; } static bool hard_wTICNT( struct tagHardware *hthis, uint8 val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; writeb( val, hthis->base + TICNT_ADDR); return true; } static bool hard_rRTCALM( struct tagHardware *hthis, uint8 *val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; *val &= 0; *val = readb( hthis->base + RTCALM_ADDR); return true; } static bool hard_wRTCALM( struct tagHardware *hthis, uint8 val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; writeb( val, hthis->base + RTCALM_ADDR); return true; } static bool hard_rBCDSEC( struct tagHardware *hthis, uint8 *val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; *val &= 0; *val = readb( hthis->base + BCDSEC_ADDR); return true; } static bool hard_wBCDSEC( struct tagHardware *hthis, uint8 val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; writeb( val, hthis->base + BCDSEC_ADDR); return true; } static bool hard_rBCDMIN( struct tagHardware *hthis, uint8 *val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; *val &= 0; *val = readb( hthis->base + BCDMIN_ADDR); return true; } static bool hard_wBCDMIN( struct tagHardware *hthis, uint8 val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; writeb( val, hthis->base + BCDMIN_ADDR); return true; } static bool hard_rBCDHOUR( struct tagHardware *hthis, uint8 *val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; *val &= 0; *val = readb( hthis->base + BCDHOUR_ADDR); return true; } static bool hard_wBCDHOUR( struct tagHardware *hthis, uint8 val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; writeb( val, hthis->base + BCDHOUR_ADDR); return true; } static bool hard_rBCDDAY( struct tagHardware *hthis, uint8 *val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; *val &= 0; *val = readb( hthis->base + BCDDAY_ADDR); return true; } static bool hard_wBCDDAY( struct tagHardware *hthis, uint8 val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; writeb( val, hthis->base + BCDDAY_ADDR); return true; } static bool hard_rBCDDATE( struct tagHardware *hthis, uint8 *val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; *val &= 0; *val = readb( hthis->base + BCDDATE_ADDR); return true; } static bool hard_wBCDDATE( struct tagHardware *hthis, uint8 val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; writeb( val, hthis->base + BCDDATE_ADDR); return true; } static bool hard_rBCDMON( struct tagHardware *hthis, uint8 *val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; *val &= 0; *val = readb( hthis->base + BCDMON_ADDR); return true; } static bool hard_wBCDMON( struct tagHardware *hthis, uint8 val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; writeb( val, hthis->base + BCDMON_ADDR); return true; } static bool hard_rBCDYEAR( struct tagHardware *hthis, uint8 *val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; *val &= 0; *val = readb( hthis->base + BCDYEAR_ADDR); return true; } static bool hard_wBCDYEAR( struct tagHardware *hthis, uint8 val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; writeb( val, hthis->base + BCDYEAR_ADDR); return true; } static bool hard_rALMSEC( struct tagHardware *hthis, uint8 *val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; *val &= 0; *val = readb( hthis->base + ALMSEC_ADDR); return true; } static bool hard_wALMSEC( struct tagHardware *hthis, uint8 val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; writeb( val, hthis->base + ALMSEC_ADDR); return true; } static bool hard_rALMMIN( struct tagHardware *hthis, uint8 *val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; *val &= 0; *val = readb( hthis->base + ALMMIN_ADDR); return true; } static bool hard_wALMMIN( struct tagHardware *hthis, uint8 val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; writeb( val, hthis->base + ALMMIN_ADDR); return true; } static bool hard_rALMHOUR( struct tagHardware *hthis, uint8 *val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; *val &= 0; *val = readb( hthis->base + ALMHOUR_ADDR); return true; } static bool hard_wALMHOUR( struct tagHardware *hthis, uint8 val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; writeb( val, hthis->base + ALMHOUR_ADDR); return true; } static bool hard_rALMDATE( struct tagHardware *hthis, uint8 *val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; *val &= 0; *val = readb( hthis->base + ALMDATE_ADDR); return true; } static bool hard_wALMDATE( struct tagHardware *hthis, uint8 val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; writeb( val, hthis->base + ALMDATE_ADDR); return true; } static bool hard_rALMMON( struct tagHardware *hthis, uint8 *val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; *val &= 0; *val = readb( hthis->base + ALMMON_ADDR); return true; } static bool hard_wALMMON( struct tagHardware *hthis, uint8 val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; writeb( val, hthis->base + ALMMON_ADDR); return true; } static bool hard_rALMYEAR( struct tagHardware *hthis, uint8 *val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; *val &= 0; *val = readb( hthis->base + ALMYEAR_ADDR); return true; } static bool hard_wALMYEAR( struct tagHardware *hthis, uint8 val) { if( NULL==hthis) return false; if( NULL==hthis->base) return false; writeb( val, hthis->base + ALMYEAR_ADDR); return true; } static struct tagHardware hardwar = { /** resource*/ .res = NULL, .res_mem = NULL, .base = NULL, /** public function*/ //control register .funcCon_r = hard_rRTCCON, .funcCon_w = hard_wRTCCON, .funcTicnt_r = hard_rTICNT, .funcTicnt_w = hard_wTICNT, .funcAlm_r = hard_rRTCALM, .funcAlm_w = hard_wRTCALM, //time register .funcSec_r = hard_rBCDSEC, .funcSec_w = hard_wBCDSEC, .funcMin_r = hard_rBCDMIN, .funcMin_w = hard_wBCDMIN, .funcHour_r = hard_rBCDHOUR, .funcHour_w = hard_wBCDHOUR, .funcDay_r = hard_rBCDDAY, .funcDay_w = hard_wBCDDAY, .funcDate_r = hard_rBCDDATE, .funcDate_w = hard_wBCDDATE, .funcMon_r = hard_rBCDMON, .funcMon_w = hard_wBCDMON, .funcYear_r = hard_rBCDYEAR, .funcYear_w = hard_wBCDYEAR, //alarm .funcASec_r = hard_rALMSEC, .funcASec_w = hard_wALMSEC, .funcAMin_r = hard_rALMMIN, .funcAMin_w = hard_wALMMIN, .funcAHour_r = hard_rALMHOUR, .funcAHour_w = hard_wALMHOUR, .funcADate_r = hard_rALMDATE, .funcADate_w = hard_wALMDATE, .funcAMon_r = hard_rALMMON, .funcAMon_w = hard_wALMMON, .funcAYear_r = hard_rALMYEAR, .funcAYear_w = hard_wALMYEAR, }; static bool __RTCCON_BIT(struct tagSoftware *sthis, bool isTrue, uint8 bitval) { struct tagHardware *phard; if( NULL==sthis) return false; phard = sthis->Hardware; if( (NULL!=phard) &&(NULL!=phard->funcCon_r) &&(NULL!=phard->funcCon_w) ) { uint8 reg; if( phard->funcCon_r( phard, ®) ) { if( isTrue) reg |= bitval; else reg &= (~bitval); return phard->funcCon_w( phard, reg); } } return false; } static bool soft_RTCEN( struct tagSoftware *sthis, bool isTrue) //总开关,在写时需要打开 { #define BIT_RTCEN (0x1) return __RTCCON_BIT( sthis, isTrue, BIT_RTCEN); } static bool soft_CLKSEL( struct tagSoftware *sthis, bool isTrue) //default as 0 { #define BIT_CLKSEL (0x1<<1) return __RTCCON_BIT( sthis, isTrue, BIT_CLKSEL); } static bool soft_CNTSEL( struct tagSoftware *sthis, bool isTrue) //default as 0 { #define BIT_CNTSEL (0x1<<2) return __RTCCON_BIT( sthis, isTrue, BIT_CNTSEL); } static bool soft_CLKRST( struct tagSoftware *sthis, bool isTrue) //RTC clock count reset { #define BIT_CLKRST (0x1<<3) return __RTCCON_BIT( sthis, isTrue, BIT_CLKRST); } static bool soft_SetTickEn( struct tagSoftware *sthis, bool isTrue) //tick time interrupt enable { #define BIT_TICNT_EN (0x1<<7) struct tagHardware *phard; if( NULL==sthis) return false; phard = sthis->Hardware; if( (NULL!=phard) &&(NULL!=phard->funcTicnt_r) &&(NULL!=phard->funcTicnt_w) ) { uint8 reg; if( phard->funcTicnt_r( phard, ®) ) { if( isTrue) reg |= BIT_TICNT_EN; else reg &= (~BIT_TICNT_EN); return phard->funcTicnt_w( phard, reg); } } return false; } static bool soft_SetTickFreq( struct tagSoftware *sthis, int freq) //0~127 { struct tagHardware *hard ; uint8 val ; if( NULL==sthis) return false; hard = sthis->Hardware; if( (NULL!=hard) &&(NULL!= hard->funcTicnt_r) &&(NULL!= hard->funcTicnt_w)) { uint8 reg; if( hard->funcTicnt_r( hard, ®) ) { reg &= (BIT_TICNT_EN) ; val = 128/freq -1; reg |= val; return hard->funcTicnt_w( hard, reg); } } return false; } static bool __RTCALM_BIT(struct tagSoftware *sthis, bool isTrue, uint8 bitval) { struct tagHardware *phard; if( NULL==sthis) return false; phard = sthis->Hardware; if( (NULL!=phard) &&(NULL!=phard->funcAlm_r) &&(NULL!=phard->funcAlm_w) ) { uint8 reg; if( phard->funcAlm_r( phard, ®) ) { if( isTrue) reg |= bitval; else reg &= (~bitval); //printk("========================3, %x \n", reg); return phard->funcAlm_w( phard, reg); } } return false; } static bool soft_GetAlmEn( struct tagSoftware *sthis, uint8 *mask) { struct tagHardware *phard; if( NULL==sthis) return false; if( NULL==mask) return false; phard = sthis->Hardware; if( (NULL!=phard) &&(NULL!=phard->funcAlm_r)) { return phard->funcAlm_r( phard, mask); } return false; } static bool soft_ALMEN( struct tagSoftware *sthis, bool isTrue) //alarm global enable { #define BIT_ALMEN (0x1<<6) return __RTCALM_BIT( sthis, isTrue, BIT_ALMEN); } static bool soft_YEAREN( struct tagSoftware *sthis, bool isTrue) { #define BIT_YEAREN (0x1<<5) return __RTCALM_BIT( sthis, isTrue, BIT_YEAREN); } static bool soft_MONEN( struct tagSoftware *sthis, bool isTrue) { #define BIT_MONEN (0x1<<4) return __RTCALM_BIT( sthis, isTrue, BIT_MONEN); } static bool soft_DATEEN( struct tagSoftware *sthis, bool isTrue) { #define BIT_DATEEN (0x1<<3) return __RTCALM_BIT( sthis, isTrue, BIT_DATEEN); } static bool soft_HOUREN( struct tagSoftware *sthis, bool isTrue) { #define BIT_HOUREN (0x1<<2) return __RTCALM_BIT( sthis, isTrue, BIT_HOUREN); } static bool soft_MINEN( struct tagSoftware *sthis, bool isTrue) { #define BIT_MINEN (0x1<<1) return __RTCALM_BIT( sthis, isTrue, BIT_MINEN); } static bool soft_SECEN( struct tagSoftware *sthis, bool isTrue) { #define BIT_SECEN (0x1<<0) return __RTCALM_BIT( sthis, isTrue, BIT_SECEN); } static bool soft_GetTIM( struct tagSoftware *sthis, struct rtc_time *buf) { struct tagHardware *hard; if( (NULL==sthis) ||(NULL==buf) ) return false; hard = sthis->Hardware; if( (NULL==hard)) return false; if( NULL!=hard->funcSec_r) hard->funcSec_r( hard, &(buf->sec)); if( NULL!=hard->funcMin_r) hard->funcMin_r( hard, &(buf->min)); if( NULL!=hard->funcHour_r) hard->funcHour_r( hard, &(buf->hour)); if( NULL!=hard->funcDate_r) hard->funcDate_r( hard, &(buf->date)); if( NULL!=hard->funcMon_r) hard->funcMon_r( hard, &(buf->mon)); if( NULL!=hard->funcYear_r) hard->funcYear_r( hard, &(buf->year)); if( NULL!=hard->funcDay_r) hard->funcDay_r( hard, &(buf->day)); return true; } static bool soft_SetTIM( struct tagSoftware *sthis, const struct rtc_time *buf) { uint8 reg_rtccon; struct tagHardware *hard; if( (NULL==sthis) ||(NULL==buf) ) return false; //check RTCEN, it must be on if want write to those registers of RTC hard = sthis->Hardware; if( (NULL==hard) ||(NULL==hard->funcCon_r)) return false; hard->funcCon_r( hard, ®_rtccon); if( (!reg_rtccon)&BIT_RTCEN ) return false; //write date if( NULL!=hard->funcSec_w) hard->funcSec_w( hard, buf->sec); if( NULL!=hard->funcMin_w) hard->funcMin_w( hard, buf->min); if( NULL!=hard->funcHour_w) hard->funcHour_w( hard, buf->hour); if( NULL!=hard->funcDate_w) hard->funcDate_w( hard, buf->date); if( NULL!=hard->funcMon_w) hard->funcMon_w( hard, buf->mon); if( NULL!=hard->funcYear_w) hard->funcYear_w( hard, buf->year); if( NULL!=hard->funcDay_w) hard->funcDay_w( hard, buf->day); return true; } static bool soft_GetALM( struct tagSoftware *sthis, struct rtc_time *buf) { struct tagHardware *hard; if( (NULL==sthis) ||(NULL==buf) ) return false; hard = sthis->Hardware; if( (NULL==hard)) return false; if( NULL!=hard->funcASec_r) hard->funcASec_r( hard, &(buf->sec)); if( NULL!=hard->funcAMin_r) hard->funcAMin_r( hard, &(buf->min)); if( NULL!=hard->funcAHour_r) hard->funcAHour_r( hard, &(buf->hour)); if( NULL!=hard->funcADate_r) hard->funcADate_r( hard, &(buf->date)); if( NULL!=hard->funcAMon_r) hard->funcAMon_r( hard, &(buf->mon)); if( NULL!=hard->funcAYear_r) hard->funcAYear_r( hard, &(buf->year)); return true; } static bool soft_SetALM( struct tagSoftware *sthis,const struct rtc_time *buf) { uint8 reg_rtccon; struct tagHardware *hard; if( (NULL==sthis) ||(NULL==buf) ) return false; //check RTCEN, it must be on if want write to those registers of RTC hard = sthis->Hardware; if( (NULL==hard) ||(NULL==hard->funcCon_r)) return false; hard->funcCon_r( hard, ®_rtccon); if( (!reg_rtccon)&BIT_RTCEN ) return false; //write date if( NULL!=hard->funcASec_w) hard->funcASec_w( hard, buf->sec); if( NULL!=hard->funcAMin_w) hard->funcAMin_w( hard, buf->min); if( NULL!=hard->funcAHour_w) hard->funcAHour_w( hard, buf->hour); if( NULL!=hard->funcADate_w) hard->funcADate_w( hard, buf->date); if( NULL!=hard->funcAMon_w) hard->funcAMon_w( hard, buf->mon); if( NULL!=hard->funcAYear_w) hard->funcAYear_w( hard, buf->year); return true; } static struct tagSoftware soft = { /** resource*/ .Hardware = &hardwar, /** public function*/ .funcRtcen = soft_RTCEN, //总开关,在写时需要打开 .funcClksel = soft_CLKSEL, //default as 0 .funcCntsel = soft_CNTSEL, //default as 0 .funcClkrst = soft_CLKRST, //RTC clock count reset .funcInt_en = soft_SetTickEn, //tick time interrupt enable .funcSetFreq = soft_SetTickFreq, //0~127 .funcGetAlmEn = soft_GetAlmEn, .funcAlm_en = soft_ALMEN, .funcYear_en = soft_YEAREN, .funcMon_en = soft_MONEN, .funcDate_en = soft_DATEEN, .funcHour_en = soft_HOUREN, .funcMin_en = soft_MINEN, .funcSec_en = soft_SECEN, .funcGetTime = soft_GetTIM, .funcSetTime = soft_SetTIM, .funcGetAlm = soft_GetALM, .funcSetAlm = soft_SetALM, }; /**********************device************************************/ /*****************************************************************/ #define RTC_CLAS_NAME "clas_rtc_test" #define RTC_NAME "rtc_test" #define RTC_DRV_NAME "rtc_drv_test" #define MAJOR_NUM 0 //自动申请设备号 #define MINOR_NUM 0 struct tagRTC_Driver; typedef bool (*RTC_EN)( struct tagRTC_Driver *rthis, bool isTrue); typedef bool (*RTC_TIME_OPS)( struct tagRTC_Driver *rthis, struct rtc_time *buf); typedef bool (*RTC_SETTICKEN)( struct tagRTC_Driver *rthis, bool isTrue); typedef bool (*RTC_SETTICKFREQ)( struct tagRTC_Driver *rthis, int freq); typedef bool (*RTC_SETALARMEN)( struct tagRTC_Driver *rthis, uint8 mask); typedef bool (*RTC_GETALARMEN)( struct tagRTC_Driver *rthid, uint8 *mask); struct tagRTC_Driver { //user operation /** resources*/ struct cdev cdev; dev_t devNum; struct semaphore sem; struct class *pClas; struct device *pDev; struct fasync_struct *async_queue; struct tagSoftware *soft_ops; /** public function*/ RTC_EN funcEn; RTC_TIME_OPS funcGetTime; RTC_TIME_OPS funcSetTime; RTC_TIME_OPS funcGetAlarm; RTC_TIME_OPS funcSetAlarm; RTC_SETTICKEN funcSetTickEn; RTC_SETTICKFREQ funcSetTickFreq; RTC_SETALARMEN funcSetAlarmEn; RTC_GETALARMEN funcGetAlarmEn; }; static bool rtc_en( struct tagRTC_Driver *rthis, bool isTrue) { struct tagSoftware *soft; if( NULL==rthis) return false; soft = rthis->soft_ops; if( NULL==soft) return false; if( isTrue ) { if( ( NULL!=soft->funcRtcen) &&(NULL!=soft->funcClksel) &&(NULL!=soft->funcCntsel) &&(NULL!=soft->funcClkrst) ) { soft->funcRtcen( soft, true); soft->funcClksel( soft, false); soft->funcCntsel( soft, false); soft->funcClkrst( soft, false); } } else { if( (NULL!=soft->funcRtcen) &&(NULL!=soft->funcInt_en)) { soft->funcRtcen( soft, false); soft->funcInt_en( soft, false); } } return true; } static bool rtc_GetTime( struct tagRTC_Driver *rthis, struct rtc_time *buf) { struct tagSoftware *soft ; if( (NULL==rthis) ||(NULL==buf) ) return false; soft = rthis->soft_ops; if( (NULL!=soft) &&(NULL!=soft->funcGetTime)) return soft->funcGetTime( soft, buf); return true; } static bool rtc_setTime( struct tagRTC_Driver *rthis, struct rtc_time *buf) { struct tagSoftware *soft; if( (NULL==rthis) ||(NULL==buf) ) return false; soft = rthis->soft_ops; if( (NULL!=soft) &&(NULL!=soft->funcSetTime)) { return soft->funcSetTime( soft, buf); } return true; } static bool rtc_GetAlarm( struct tagRTC_Driver *rthis, struct rtc_time *buf) { struct tagSoftware *soft ; if( (NULL==rthis) ||(NULL==buf) ) return false; soft = rthis->soft_ops; if( (NULL!=soft) &&(NULL!=soft->funcGetAlm)) return soft->funcGetAlm( soft, buf); return true; } static bool rtc_setAlarm( struct tagRTC_Driver *rthis, struct rtc_time *buf) { struct tagSoftware *soft; if( (NULL==rthis) ||(NULL==buf) ) return false; soft = rthis->soft_ops; if( (NULL!=soft) &&(NULL!=soft->funcSetAlm)) { return soft->funcSetAlm( soft, buf); } return true; } #if 0 static bool rtc_setFreq( struct tagRTC_Driver *rthis, int freq, bool isTrue) { struct tagSoftware *soft ; if( NULL==rthis) return false; soft = rthis->soft_ops; if( ( NULL!=soft) &&( NULL!=soft->funcInt_en) &&( soft->funcInt_en( soft, isTrue) ) ) { freq%=128; return soft->funcSetFreq( soft, freq); } return false; } #endif static bool rtc_SetTickEn( struct tagRTC_Driver *rthis, bool isTrue) { struct tagSoftware *soft; if( NULL==rthis) return false; soft = rthis->soft_ops; if( ( NULL!=soft) &&( NULL!=soft->funcInt_en) ) { return soft->funcInt_en( soft, isTrue); } return false; } static bool rtc_SetTickFreq( struct tagRTC_Driver *rthis, int freq) { struct tagSoftware *soft ; if( NULL==rthis) return false; soft = rthis->soft_ops; if( ( NULL!=soft) &&( NULL!=soft->funcSetFreq) ) { freq%=128; return soft->funcSetFreq( soft, freq); } return false; } static bool _rtc_SetBit( struct tagSoftware *soft, BIT_OPS func, int mask, int x) { bool val; if( NULL==func) return false; val = GETBIT(mask,x); return func( soft, val); } // 1 meaning for enable static bool rtc_SetAlarmEn( struct tagRTC_Driver *rthis, uint8 mask) { struct tagSoftware *soft; bool ret = true; if( NULL==rthis) return false; soft = rthis->soft_ops; if( NULL==soft) return false; if(!_rtc_SetBit( soft, soft->funcSec_en, mask, 0)) ret = false; if(!_rtc_SetBit( soft, soft->funcMin_en, mask, 1)) ret = false; if(!_rtc_SetBit( soft, soft->funcHour_en, mask, 2)) ret = false; if(!_rtc_SetBit( soft, soft->funcDate_en, mask, 3)) ret = false; if(!_rtc_SetBit( soft, soft->funcMon_en, mask, 4)) ret = false; if(!_rtc_SetBit( soft, soft->funcYear_en, mask, 5)) ret = false; if(!_rtc_SetBit( soft, soft->funcAlm_en, mask, 6)) ret = false; return ret; } static bool rtc_GetAlarmEn( struct tagRTC_Driver *rthis, uint8 *mask) { struct tagSoftware *soft; if( NULL==rthis) return false; soft = rthis->soft_ops; if( (NULL!=soft) &&( NULL!=soft->funcGetAlmEn)) { return soft->funcGetAlmEn( soft, mask); } return false; } static struct tagRTC_Driver drvDate = { /** resources*/ .soft_ops = &soft, /** public function*/ .funcEn = rtc_en, .funcGetTime = rtc_GetTime , .funcSetTime = rtc_setTime , .funcGetAlarm = rtc_GetAlarm , .funcSetAlarm = rtc_setAlarm , .funcSetTickEn = rtc_SetTickEn , .funcSetTickFreq = rtc_SetTickFreq , .funcSetAlarmEn = rtc_SetAlarmEn , .funcGetAlarmEn = rtc_GetAlarmEn , }; /*********************************************************/ #define EVENT_MAX 7 struct tagEventQueue; typedef bool (*EVE_OPS)( struct tagEventQueue *ethis, struct rtc_event *eve); struct tagEventQueue { /** data*/ int head; int tail; struct rtc_event queue[EVENT_MAX]; /** function*/ EVE_OPS Add; EVE_OPS Get; }; static bool AddEvent( struct tagEventQueue *ethis, struct rtc_event *inEve) { if( NULL==ethis) return false; if( NULL==inEve) return false; ethis->queue[ethis->tail] = *inEve; ethis->tail ++; if( ethis->tail>=EVENT_MAX ) { ethis->tail = 0; if( ethis->head==0) { ethis->head ++; } } return true; } static bool GetEvent( struct tagEventQueue *ethis, struct rtc_event *OutEve) { if( NULL==ethis) return false; if( NULL==OutEve) return false; if( ethis->head==ethis->tail) { memset( OutEve, 0, sizeof(struct rtc_event)); return false; } *OutEve = ethis->queue[ethis->head]; ethis->head ++ ; if( ethis->head>=EVENT_MAX) ethis->head = 0; return true; } static struct tagEventQueue events = { .head = 0, .tail = 0, .Add = AddEvent , .Get = GetEvent , }; static irqreturn_t rtc_alarmirq( int irq, void *id) { struct rtc_event Eve; printk(" congratulation!!you get alarm-irq\n"); Eve.type = rtc_alarm; Eve.data.alarm.irqnum = irq; if( !events.Add( &events, &Eve)) { printk("warn: fail to record event information, %s, %d\n", __FILE__, __LINE__); } //a notification will be send to user process. kill_fasync( &(drvDate.async_queue), SIGIO, POLL_IN); return IRQ_HANDLED; } static irqreturn_t rtc_tickirq( int irq, void *id) { struct rtc_event Eve; printk(" congratulation!!you get tick-irq, \n"); Eve.type = rtc_tick; Eve.data.tick.irqnum = irq; if( !events.Add( &events, &Eve)) { printk("warn: fail to record event information, %s, %d\n", __FILE__, __LINE__); } kill_fasync( &(drvDate.async_queue), SIGIO, POLL_IN); return IRQ_HANDLED; } static int rtc_open ( struct inode *id, struct file *fp) { //printk("rtc_open...................\n"); int ret; int irq_tick, irq_alarm; if( down_trylock( &(drvDate.sem))) { return -EBUSY; } if( (NULL==drvDate.soft_ops->res_tickno) ||(NULL==drvDate.soft_ops->res_alarmno) ) { printk("error: invalid resource, %s, %d \n", __FILE__, __LINE__); return -EIO; } irq_tick = drvDate.soft_ops->res_tickno->start; ret = request_irq( irq_tick, rtc_tickirq, IRQF_DISABLED, "TQ_rtc_drv2 alarm", &drvDate); if (ret) { printk("error : fail to request irq-alarm\n"); goto ERR_TICK; } irq_alarm = drvDate.soft_ops->res_alarmno->start; ret = request_irq( irq_alarm, rtc_alarmirq, IRQF_DISABLED, "TQ_rtc_drv2 tick", &drvDate); if (ret) { printk("error : fail to request irq-tick\n"); goto ERR_ALARM; } /***************tmp*********************************/ /****************************************************/ return 0; free_irq( drvDate.soft_ops->res_alarmno->start, &drvDate); ERR_ALARM: free_irq( drvDate.soft_ops->res_tickno->start, &drvDate); ERR_TICK: up(&(drvDate.sem)); return ret; } static int rtc_release ( struct inode *id, struct file *fp) { //printk("rtc_release...................\n"); int ret = 0; int irq_tick, irq_alarm; up(&(drvDate.sem)); //close tick close if( (NULL!=drvDate.funcSetTickEn) &&( !drvDate.funcSetTickEn( &drvDate, false)) ) { printk(" error : fail to close tick clock, %s, %d\n", __FILE__, __LINE__); ret = -1; } //release irq if( NULL!=drvDate.soft_ops ) { if( NULL!=drvDate.soft_ops->res_tickno) { irq_tick = drvDate.soft_ops->res_tickno->start; free_irq( irq_tick, &drvDate); } if( NULL!=drvDate.soft_ops->res_alarmno) { irq_alarm = drvDate.soft_ops->res_alarmno->start; free_irq( irq_alarm, &drvDate); } } return ret; } static ssize_t rtc_read ( struct file *fp, char __user *buf, size_t len, loff_t *off) { struct rtc_time time; unsigned long ret = 0; if( (NULL!=drvDate.funcGetTime) &&( !(drvDate.funcGetTime( &drvDate, &time)))) { return 0; } ret = copy_to_user( buf, &time, len); if( ret<0) { printk("warn: fail to copy data to user, %s, %d\n", __FILE__, __LINE__); return 0; } return len; } static ssize_t rtc_write ( struct file *fp, const char __user *buf, size_t len, loff_t *off) { //void __iomem *base = rtcData.addr.base; struct rtc_time time; //unsigned long ret = 0; //printk("rtc_write...................\n"); memset( &time, 0, sizeof(struct rtc_time)); len = copy_from_user( &time, buf, len); if( (NULL!=drvDate.funcSetTime) &&( drvDate.funcSetTime( &drvDate, &time)) ) { return len; //还有len个字节数据没写入 } return -1; } static int rtc_ioctl(struct inode *in, struct file *fp, unsigned int cmd, unsigned long arg) { int ret = -1; //printk(" ioctl : %x, %lx\n", cmd, arg); switch( cmd) { case CMD_SETTIME: { struct rtc_time time; memset( &time, 0, sizeof(struct rtc_time)); if(copy_from_user( &time, (struct rtc_time *)arg, sizeof(struct rtc_time))) { break; } if( (NULL!=drvDate.funcSetTime) && ( drvDate.funcSetTime( &drvDate, &time)) ) { ret = 0; } break; } case CMD_GETTIME: { struct rtc_time time; if( (NULL!=drvDate.funcGetTime) &&( !(drvDate.funcGetTime( &drvDate, &time)))) { break; } if( (0!=arg) &&(copy_to_user( (struct rtc_time *)arg, &time, sizeof(struct rtc_time))<0) ) { printk("warn: fail to copy data to user, %s, %d\n", __FILE__, __LINE__); break; } ret = 0; break; } case CMD_SETALARM: { if( 0!=arg) { struct rtc_time time; copy_from_user( &time, (struct rtc_time *)arg, sizeof(struct rtc_time)); if( (NULL!=drvDate.funcSetAlarm) &&(drvDate.funcSetAlarm( &drvDate, &time))) { ret = 0; } } break; } case CMD_GETALARM: { if( 0!=arg) { struct rtc_time time; if( (NULL!=drvDate.funcGetAlarm) &&(drvDate.funcGetAlarm( &drvDate, &time))) { copy_to_user( (struct rtc_time *)arg, &time, sizeof(struct rtc_time)); ret = 0; } } break; } case CMD_ENTICK: { if( ( NULL!=drvDate.funcSetTickEn) &&(drvDate.funcSetTickEn( &drvDate, true)) ) { ret = 0; } break; } case CMD_DISABTICK: { if( ( NULL!=drvDate.funcSetTickEn) &&(drvDate.funcSetTickEn( &drvDate, false)) ) { ret = 0; } break; } case CMD_SETFREQ: { int freq = arg%128; if( ( NULL!=drvDate.funcSetTickFreq) &&(drvDate.funcSetTickFreq( &drvDate, freq)) ) { ret = 0; } break; } case CMD_SETALARMEN: { int mask = (int)arg; if( ( NULL!=drvDate.funcSetAlarmEn) &&(drvDate.funcSetAlarmEn( &drvDate, mask)) ) { ret = 0; } break; } case CMD_GETALARMEN: { if( 0!=arg) { int *mask = (int *)arg; if( ( NULL!=drvDate.funcGetAlarmEn) &&(drvDate.funcGetAlarmEn( &drvDate, (uint8 *)mask)) ) { ret = 0; } } break; } case CMD_GETEVENT: { struct rtc_event Eve; if( events.Get( &events, &Eve) ) { ret = 0; } if( 0!=arg) copy_to_user( (struct rtc_event *)arg, &Eve, sizeof( struct rtc_event)); break; } default : printk("warn: invalid command\n"); break; } return ret; } static int rtc_fasync(int fd, struct file *filp, int mode) { //process, which is alloc aychronous notification, will be record on the queue by the help of this function. return fasync_helper( fd, filp, mode, &(drvDate.async_queue)); } static struct file_operations rtc_ops = { .fasync = rtc_fasync , .open = rtc_open, .release = rtc_release, .read = rtc_read, .write = rtc_write, .ioctl = rtc_ioctl, }; /********************************************************************/ static int rtc_probe(struct platform_device *dev) { printk("rtc_probe............\n"); //init cdev_init( &(drvDate.cdev), &rtc_ops); drvDate.devNum = 0; drvDate.cdev.owner = THIS_MODULE; sema_init( &(drvDate.sem), 1); drvDate.pClas = NULL; drvDate.pDev = NULL; drvDate.soft_ops = &soft; //alloc device-number if( MAJOR_NUM!=0) { drvDate.devNum = MKDEV( MAJOR_NUM, MINOR_NUM); if(register_chrdev_region( drvDate.devNum, 1, RTC_DRV_NAME)) goto ERR_REG; } else { if(alloc_chrdev_region( &(drvDate.devNum), MINOR_NUM, 1, RTC_DRV_NAME)) goto ERR_REG; } //add char-device if(cdev_add( &(drvDate.cdev), drvDate.devNum, 1)<0) goto ERR_CDEV; //create class drvDate.pClas = class_create( THIS_MODULE, RTC_CLAS_NAME); if( NULL==drvDate.pClas) goto ERR_CLAS; //create device drvDate.pDev= device_create( drvDate.pClas, NULL, drvDate.devNum, NULL, "rtc_device_test"); if( NULL==drvDate.pDev) goto ERR_DEVICE; /***********init device****************************************/ drvDate.soft_ops->Hardware->res = platform_get_resource( dev, IORESOURCE_MEM, 0); if( NULL==drvDate.soft_ops->Hardware->res) { printk("error: %s, %d\n", __FILE__, __LINE__); goto ERR_DEVICE; } drvDate.soft_ops->res_alarmno = platform_get_resource( dev, IORESOURCE_IRQ, 0); if( NULL==drvDate.soft_ops->res_alarmno) { printk("error: %s, %d\n", __FILE__, __LINE__); goto ERR_DEVICE; } drvDate.soft_ops->res_tickno = platform_get_resource( dev, IORESOURCE_IRQ, 1); if( NULL==drvDate.soft_ops->res_tickno) { printk("error: %s, %d\n", __FILE__, __LINE__); goto ERR_DEVICE; } drvDate.soft_ops->Hardware->res_mem = request_mem_region( drvDate.soft_ops->Hardware->res->start, drvDate.soft_ops->Hardware->res->end - drvDate.soft_ops->Hardware->res->start + 1, dev->name); if( NULL==drvDate.soft_ops->Hardware->res_mem ) { printk("error: %s, %d\n", __FILE__, __LINE__); goto ERR_REQ; } drvDate.soft_ops->Hardware->base = ioremap( drvDate.soft_ops->Hardware->res->start, drvDate.soft_ops->Hardware->res->end - drvDate.soft_ops->Hardware->res->start + 1 ); if( NULL==drvDate.soft_ops->Hardware->base) { printk("error: %s, %d\n", __FILE__, __LINE__); goto ERR_REMAP; } if( (NULL==drvDate.funcEn) ||(!(drvDate.funcEn( &drvDate, true)) ) ) goto ERR_EN; /***********************************************************/ return 0; //返回0表示接受这次探测 ERR_EN: iounmap( drvDate.soft_ops->Hardware->base); ERR_REMAP: release_resource( drvDate.soft_ops->Hardware->res_mem); ERR_REQ: device_destroy( drvDate.pClas, drvDate.devNum); ERR_DEVICE: class_destroy( drvDate.pClas); ERR_CLAS: cdev_del( &(drvDate.cdev)); ERR_CDEV: unregister_chrdev_region( drvDate.devNum, 1); ERR_REG: return -1; } static int rtc_remove( struct platform_device *dev) { printk(" rtc_remove.........................\n"); if( NULL!=drvDate.funcEn) drvDate.funcEn( &drvDate, false); iounmap(drvDate.soft_ops->Hardware->base); release_resource( drvDate.soft_ops->Hardware->res_mem); device_destroy( drvDate.pClas, drvDate.devNum); class_destroy( drvDate.pClas); cdev_del( &(drvDate.cdev)); unregister_chrdev_region( drvDate.devNum, 1); return 0; } static struct platform_driver rtc_drv = { .probe = rtc_probe, .remove = rtc_remove, .driver = { .name = RTC_NAME, .owner = THIS_MODULE, }, }; static int __init rtc_init( void) { printk("vision : %s\n", __TIME__); return platform_driver_register( &rtc_drv); } static void __exit rtc_exit( void) { printk("vision : %s\n", __TIME__); platform_driver_unregister( &rtc_drv); } MODULE_LICENSE("GPL"); module_init( rtc_init); module_exit( rtc_exit);
3.2 device
#include <linux/module.h> #include <linux/kernel.h> #include <linux/init.h> #include <linux/fs.h> #include <linux/cdev.h> #include <asm/uaccess.h> #include <linux/device.h> #include <linux/platform_device.h> #include <asm/irq.h> #define RTC_NAME "rtc_test" #define RTC_BASE_ADDR 0x57000040 #define RTC_END_ADDR 0x5700008b static struct resource rtc_res[] = { [0] = { .start = RTC_BASE_ADDR, .end = RTC_END_ADDR, .flags = IORESOURCE_MEM, }, [1] = { .start = IRQ_RTC, .end = IRQ_RTC, .flags = IORESOURCE_IRQ, }, [2] = { .start = IRQ_TICK, .end = IRQ_TICK, .flags = IORESOURCE_IRQ }, }; static struct platform_device rtc_dev = { .name = RTC_NAME, .id = 0, .num_resources = ARRAY_SIZE(rtc_res), .resource = rtc_res, }; static int __init rtc_init( void) { printk("%s\n", __TIME__); return platform_device_register( &(rtc_dev)); } static void __exit rtc_exit( void) { printk("%s\n", __TIME__); platform_device_unregister( &rtc_dev); } MODULE_LICENSE("GPL"); module_init( rtc_init); module_exit( rtc_exit);