1. 程式人生 > >《韋東山視訊第二期》——i2c驅動

《韋東山視訊第二期》——i2c驅動

#include <linux/kernel.h>
#include <linux/init.h>
#include <linux/module.h>
#include <linux/slab.h>
#include <linux/jiffies.h>
#include <linux/i2c.h>
#include <linux/mutex.h>
#include <linux/fs.h>
#include <asm/uaccess.h>

static unsigned short ignore[]      = { I2C_CLIENT_END };
static unsigned short normal_addr[] = { 0x50, I2C_CLIENT_END }; /* 地址值是7位 */
                                        /* 改為0x60的話, 由於不存在裝置地址為0x60的裝置, 所以at24cxx_detect不被呼叫 */

static unsigned short force_addr[] = {ANY_I2C_BUS, 0x60, I2C_CLIENT_END};
static unsigned short * forces[] = {force_addr, NULL};
										
static struct i2c_client_address_data addr_data = {
	.normal_i2c	= normal_addr,  /* 要發出S訊號和裝置地址並得到ACK訊號,才能確定存在這個裝置 */
	.probe		= ignore,
	.ignore		= ignore,
	//.forces     = forces, /* 強制認為存在這個裝置 */
};

static struct i2c_driver at24cxx_driver;


static int major;
static struct class *cls;
struct i2c_client *at24cxx_client;

static ssize_t at24cxx_read(struct file *file, char __user *buf, size_t size, loff_t * offset)
{
	unsigned char address;
	unsigned char data;
	struct i2c_msg msg[2];
	int ret;
	
	/* address = buf[0] 
	 * data    = buf[1]
	 */
	if (size != 1)
		return -EINVAL;
	
	copy_from_user(&address, buf, 1);

	/* 資料傳輸三要素: 源,目的,長度 */

	/* 讀AT24CXX時,要先把要讀的儲存空間的地址發給它 */
	msg[0].addr  = at24cxx_client->addr;  /* 目的 */
	msg[0].buf   = &address;              /* 源 */
	msg[0].len   = 1;                     /* 地址=1 byte */
	msg[0].flags = 0;                     /* 表示寫 */

	/* 然後啟動讀操作 */
	msg[1].addr  = at24cxx_client->addr;  /* 源 */
	msg[1].buf   = &data;                 /* 目的 */
	msg[1].len   = 1;                     /* 資料=1 byte */
	msg[1].flags = I2C_M_RD;                     /* 表示讀 */


	ret = i2c_transfer(at24cxx_client->adapter, msg, 2);
	if (ret == 2)
	{
		copy_to_user(buf, &data, 1);
		return 1;
	}
	else
		return -EIO;
}

static ssize_t at24cxx_write(struct file *file, const char __user *buf, size_t size, loff_t *offset)
{
	unsigned char val[2];
	struct i2c_msg msg[1];
	int ret;
	
	/* address = buf[0] 
	 * data    = buf[1]
	 */
	if (size != 2)
		return -EINVAL;
	
	copy_from_user(val, buf, 2);

	/* 資料傳輸三要素: 源,目的,長度 */
	msg[0].addr  = at24cxx_client->addr;  /* 目的 */
	msg[0].buf   = val;                   /* 源 */
	msg[0].len   = 2;                     /* 地址+資料=2 byte */
	msg[0].flags = 0;                     /* 表示寫 */

	ret = i2c_transfer(at24cxx_client->adapter, msg, 1);
	if (ret == 1)
		return 2;
	else
		return -EIO;
}


static struct file_operations at24cxx_fops = {
	.owner = THIS_MODULE,
	.read  = at24cxx_read,
	.write = at24cxx_write,
};

static int at24cxx_detect(struct i2c_adapter *adapter, int address, int kind)
{	
	printk("at24cxx_detect\n");

	/* 構構一個i2c_client結構體: 以後收改資料時會用到它 */
	at24cxx_client = kzalloc(sizeof(struct i2c_client), GFP_KERNEL);
	at24cxx_client->addr    = address;
	at24cxx_client->adapter = adapter;
	at24cxx_client->driver  = &at24cxx_driver;
	strcpy(at24cxx_client->name, "at24cxx");
	i2c_attach_client(at24cxx_client);
	
	major = register_chrdev(0, "at24cxx", &at24cxx_fops);

	cls = class_create(THIS_MODULE, "at24cxx");
	class_device_create(cls, NULL, MKDEV(major, 0), NULL, "at24cxx"); /* /dev/at24cxx */
	
	return 0;
}

static int at24cxx_attach(struct i2c_adapter *adapter)
{
	return i2c_probe(adapter, &addr_data, at24cxx_detect);
}

static int at24cxx_detach(struct i2c_client *client)
{
	printk("at24cxx_detach\n");
	class_device_destroy(cls, MKDEV(major, 0));
	class_destroy(cls);
	unregister_chrdev(major, "at24cxx");

	i2c_detach_client(client);
	kfree(i2c_get_clientdata(client));

	return 0;
}


/* 1. 分配一個i2c_driver結構體 */
/* 2. 設定i2c_driver結構體 */
static struct i2c_driver at24cxx_driver = {
	.driver = {
		.name	= "at24cxx",
	},
	.attach_adapter = at24cxx_attach,
	.detach_client  = at24cxx_detach,
};

static int at24cxx_init(void)
{
	i2c_add_driver(&at24cxx_driver);
	return 0;
}

static void at24cxx_exit(void)
{
	i2c_del_driver(&at24cxx_driver);
}

module_init(at24cxx_init);
module_exit(at24cxx_exit);

MODULE_LICENSE("GPL");


相關推薦

東山視訊第二》——i2c驅動

#include <linux/kernel.h> #include <linux/init.h> #include <linux/module.h> #include <linux/slab.h> #include <

東山視訊第二之心得體會

   最近一直在收看韋東山老師的嵌入式視訊,收穫頗豐。第二期主要是關於一些裝置驅動的編寫示例,從最基礎的按鍵驅動、LED驅動到複雜的LCD、觸控式螢幕、網絡卡、以及各種匯流排驅動。今天主要對之前學習的LCD驅動做一個學習心得,將學習中的一些問題和收穫和大家一起分享。    

東山視訊第1 學習計劃

  學習計劃按照《萬勇210學習日記》中的問題,在後面補充各部門的學習筆記,在文中加入跳轉,以便形成一個學習體系。下面,先看下萬勇同學的學習歷程: 問題1:當Tiny210開發板到手後,最想幹的事情無疑就是把二期講的驅動在Tiny210上面通通給實現了, 那該怎麼操作呢?

第四篇:東山教程需安裝的驅動和軟件

二進制 image test ask x文件 exe info 自制 編輯 目錄 一、eop燒寫器驅動安裝 二、oflash燒寫軟件安裝 三、串口工具(MobaXterm_Portable_v10.4.exe) 四、uboo通過USB燒寫程序需安裝的驅動和軟件 五、hex

東山視訊教程視訊的章節和要點

第一部分 ARM體系結構與裸機實驗 韋東山Linux視訊第1期 第01課 環境搭建及工具、概念介紹 第1部分的細化,以後就在Linux上工作了,環境必須先建好 第02課&nbs

東山視訊中的uboot引數的設定

這篇文章只做備忘之用, 以前的引數(主要): mtddevname=bootloader mtdids=nand0=nandflash0 mtdparts=mtdparts=nandflash0:25

東山ARM第一總結

文章目錄 01 - 裸機開發特點 1.1 - 51 / Arduino / STM等微控制器 1.2 - ARM微控制器 02 - 總結 一切都站在裸機開發者的角度而言

東山ARM第一作業(十)IIC-SPI-感測器

文章目錄 01 - 作業所在路徑 02 - 作業描述 03 - 作業解答   第19~22課因為需要額外買模組,但是某些原因小白沒有買,所以只能看視訊,不能做實驗,現在也不想買了,於是也沒有做作業,要是做了的歡迎聯絡小白,直接補上而且

東山ARM第一作業(九)觸控式螢幕

文章目錄 01 - 作業所在路徑 02 - 作業描述 03 - 作業解答 01 - 作業所在路徑   ARM裸機1期加強版\原始碼文件圖片\文件圖片\第018課_ADC和觸控式螢幕 02 - 作業描述 百度搜索下載

東山ARM第一作業(八)LCD

文章目錄 01 - 作業所在路徑 02 - 作業描述 2.1 - 作業1 2.2 - 作業2 03 - 作業解答 3.1 - 作業1解答 3.2 - 作業2解答

東山ARM第一作業(七)NandFlash

文章目錄 01 - 作業所在路徑 02 - 作業描述 2.1 - 作業1 2.2 - 作業2 2.3 - 作業3 2.4 - 作業4 03 - 作業解答

東山ARM第一作業(六)NorFlash

文章目錄 01 - 作業所在路徑 02 - 作業描述 2.1 - 作業1 2.2 - 作業2 03 - 作業解答 3.1 - 作業1解答 3.2 - 作業2解答

東山ARM第一作業(五)異常和中斷

文章目錄 01 - 作業所在路徑 02 - 作業描述 2.1 - 作業1 2.2 - 作業2 2.3 - 作業3 2.4 - 作業4 2.5 - 作業5 03 - 作業解答

東山ARM第一作業(四)程式碼重定位

文章目錄 01 - 作業所在路徑 02 - 作業描述 2.1 - 作業1 2.2 - 作業2 03 - 作業解答 3.1 - 作業1解答 3.2 - 作業2解答

東山ARM第一作業(三)記憶體控制器和SDRAM

文章目錄 01 - 作業所在路徑 02 - 作業描述 03 - 作業解答 01 - 作業所在路徑   ARM裸機1期加強版\原始碼文件圖片\文件圖片\第012課_記憶體控制器與SDRAM 02 - 作業描述   把第11課

東山ARM第一作業(二)UART串列埠

文章目錄 01 - 作業所在路徑 02 - 作業描述 2.1 - 作業1 2.2 - 作業2 03 - 作業解答 3.1 - 作業1解答 3.2 - 作業2解答

東山ARM第一作業(一)ARM時鐘體系

文章目錄 01 - 作業所在路徑 02 - 作業描述 03 - 作業解答 01 - 作業所在路徑   ARM裸機1期加強版\原始碼文件圖片\文件圖片\第010課_掌握ARM晶片時鐘體系 02 - 作業描述   根據S3C4

東山視訊

◆先把開發板當作微控制器來用,這可以讓你熟悉LINUX環境程式設計、熟悉ARM板的硬體操作就是寫各種裸板程式◆接著學習UBOOT,你可以認為UBOOT就是各種裸板程式的集合。如果覺得UBOOT太複雜,可以看畢業班視訊裡從零寫Bootloader,你把它理解了就可以◆接著移植LINUX,這稍做一下就可以了,不用

東山視訊實驗之Input子系統分析之一

原理闡述:主要有驅動層,input核心層,handler處理層,應用層                 當有按鍵觸發時, 從前往後一直傳送到應用層 先看一張韋老師提供的一張圖,下面會進行更詳細的介紹

s3c_lcd 驅動(轉東山老師視訊例程)

#include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #inclu