20145324 20145325 《信息安全系统设计基础》实验四 外设驱动程序设计

本次实验介绍了如何设计外设驱动程序,包括ARM环境配置、驱动程序的基本设计方法及其实现过程。实验通过一个简单的字符设备驱动程序示例,演示了驱动程序的主要功能,如读写操作、打开关闭设备等。

20145324 20145325 《信息安全系统设计基础》实验四 外设驱动程序设计

实验封面

课程:信息安全系统设计基础 班级: 1453

姓名:王嘉澜 张梓靖 学号:20145324 20145325

成绩: 指导教师:娄嘉鹏 实验日期:2016.11.17

实验密级: 预习程度: 实验时间:10:10-12:30

仪器组次: 必修/选修:必修 实验序号:4

实验名称: 外设驱动程序设计

实验目的与要求:
1.正确连接ARM ,设置好环境
2.理解驱动程序的一般设计方法

实验仪器名称数量型号
pc机1XP
arm机1UP-TECH
虚拟机1REDHAT

实验过程

  1. 按实验一的环境配置方法将环境配置好
    885266-20161119192028779-520352167.png

  2. 通过文件共享,将要测试的代码共享到虚拟机里
  3. 修改 makefile 为
    885266-20161119192102232-186649711.png

  4. 在/usr/src 下建立一个 linux 连接
    885266-20161119192155013-1394709364.png

  5. 建立设备节点
  6. 运行程序进行测试
    885266-20161119192236810-2101256381.png

实验代码分析

  1. 完整代码(demo.c)
#ifndef __KERNEL__
# define __KERNEL__
#endif
#ifndef MODULE
# define MODULE
#endif
#include <linux/config.h>
#include <linux/module.h>
#include <linux/devfs_fs_kernel.h>
#include <linux/init.h>
#include <linux/kernel.h> /* printk() */
#include <linux/slab.h> /* kmalloc() */
#include <linux/fs.h> /* everything... */
#include <linux/errno.h> /* error codes */
#include <linux/types.h> /* size_t */
#include <linux/proc_fs.h>
#include <linux/fcntl.h> /* O_ACCMODE */
#include <linux/poll.h> /* COPY_TO_USER */
#include <asm/system.h> /* cli(), *_flags */
#define DEVICE_NAME  "demo"
#define demo_MAJOR 254
#define demo_MINOR 0
static int MAX_BUF_LEN=1024;
static char drv_buf[1024];
static int WRI_LENGTH=0;
/*********************************************************************/
/*逆序排列缓冲区数据*/
static void do_write()
{
int i;
int len = WRI_LENGTH;
char tmp;
for(i = 0; i < (len>>1); i++,len--){
tmp = drv_buf[len-1];
drv_buf[len-1] = drv_buf[i];
drv_buf[i] = tmp;
}
}
/*********************************************************************/
static ssize_t demo_write(struct file *filp,const char *buffer, size_t count) //完成驱动的写接口
{
if(count > MAX_BUF_LEN)count = MAX_BUF_LEN;
copy_from_user(drv_buf , buffer, count);
WRI_LENGTH = count;
printk("user write data to driver\n");
do_write();
return count;
}
/*********************************************************************/
static ssize_t demo_read(struct file *filp, char *buffer, size_t count, loff_t *ppos)  //完成驱动的读接口
{
if(count > MAX_BUF_LEN)
count=MAX_BUF_LEN;
copy_to_user(buffer, drv_buf,count);
printk("user read data from driver\n");
return count;
}
/*********************************************************************/
static int demo_ioctl(struct inode *inode, struct file *file,  //演示ioctl调用接口的实现过程
unsigned int cmd, unsigned long arg)
{
switch(cmd){
case 1:printk("runing command 1 \n");break;
case 2:printk("runing command 2 \n");break;
default:
printk("error cmd number\n");break;
}
return 0;
}
/*********************************************************************/
static int demo_open(struct inode *inode, struct file *file) //打开设备
{
sprintf(drv_buf,"device open sucess!\n");
printk("device open sucess!\n");
return 0;
}
/*********************************************************************/
static int demo_release(struct inode *inode, struct file *filp) //关闭设备
{
MOD_DEC_USE_COUNT;
printk("device release\n");
return 0;
}
/*********************************************************************/
static struct file_operations demo_fops = {  //设置file_operations结构
owner:  THIS_MODULE,
write:demo_write, 
read: demo_read, 
ioctl: demo_ioctl,
open: demo_open,
release:  demo_release,
};
/*********************************************************************/
#ifdef CONFIG_DEVFS_FS
static devfs_handle_t devfs_demo_dir, devfs_demoraw;
#endif
/*********************************************************************/
static int __init demo_init(void)  //初始化
{
#ifdef CONFIG_DEVFS_FS
devfs_demo_dir = devfs_mk_dir(NULL, "demo", NULL);
devfs_demoraw = devfs_register(devfs_demo_dir, "0", DEVFS_FL_DEFAULT,
demo_MAJOR, demo_MINOR, S_IFCHR | S_IRUSR | S_IWUSR,
&demo_fops, NULL);
#else
int result;
SET_MODULE_OWNER(&demo_fops);
result = register_chrdev(demo_MAJOR, "scullc", &demo_fops);
if (result < 0) return result;
// if (demo_MAJOR == 0) demo_MAJOR = result; /* dynamic */
#endif
printk(DEVICE_NAME " initialized\n");
return 0;
}
/*********************************************************************/
static void __exit demo_exit(void)
{
unregister_chrdev(demo_MAJOR, "demo");
//kfree(demo_devices);
printk(DEVICE_NAME " unloaded\n");
}
/*********************************************************************/
module_init(demo_init);
module_exit(demo_exit);
  1. 除了 demo_read 、demo_write 、do_write 函数,添加一个小任务函数 do_read,该函数作用是将从设备中读取到的数据进行数据逆序排列,再通过读取函数读取转换后的数据:
  • 添加的do_read()函数:
static void do_read()
{
int i;
int len = WRI_LENGTH_R;
char tmp;
for(i = 0; i < (len>>1); i++,len--){
tmp = buffer[len-1];
buffer[len-1] = buffer[i];
buffer[i] = tmp;
}
}
  • 相应的对demo_read函数的修改:
static int WRI_LENGTH_R=0;
static ssize_t demo_read(struct file *filp, char *buffer, size_t count, loff_t *ppos)
{
if(count > MAX_BUF_LEN)
count=MAX_BUF_LEN;
copy_to_user(buffer, drv_buf,count);
WRI_LENGTH_R = count;
printk("user read data from driver\n");
do_read();
return count;
}

实验问题及解决

  1. make不成功
    885266-20161119192359232-1446113451.png
  • 发现没有插入驱动模块 demo.o
  1. 运行程序不成功
    885266-20161119192428310-1609990671.png
  • 还没有建立设备节点。建立以后就可以正常运行了

实验体会

makkefile中需要修改的地方的含义:KERNELDIR表示内核源代码路径,INCLUDEDIR表示配置文件,#CROSS_COMPILE表示编译适合编译的主机运行的二进制文件。通过这次实验,我们知道了在运行外设驱动程序时,要插入驱动模块 demo.o,要在/usr/src 下建立一个 linux 连接,如果是使用交叉编译器,不需要建立设备节点等等。

转载于:https://www.cnblogs.com/zzjzzjzzj/p/6081279.html

用C语言做的DOS界面的串口通信程序 void setPort(int size, char check, int stop){ unsigned char b = 0x00; disable(); switch(size){ case 6:b |= 0x01; break; case 7:b |= 0x02;break; case 8:b |= 0x03; break; } switch(check){ case 'N': case 'n':b &= ~0x08;break; case 'J': case 'j':b |= 0x08; break; case 'O': case 'o':b |= 0x18;break; } switch(stop){ case 1: b &= ~0x04;break; case 2: b |= 0x04;break; } outportb(PORT[PORTNUM] + 3, b); /*register interrupt*/ outportb(PORT[PORTNUM] + 1, 0x01);/*接收中断*/ outportb(PORT[PORTNUM] + 4, 0x0b);/*零MODEM,恒有效状态*/ oldhandle = getvect(INTR[PORTNUM]);/*取得硬中断向量表的中断处理程序*/ setvect(INTR[PORTNUM],newhandle); /*开启IRQx*/ outportb(0x21, inportb(0x21) & INTRENTER[PORTNUM]); enable(); } /* baund:波特率 */ void initCom(int baund){ unsigned char b; disable(); outportb(PORT[PORTNUM] + 3, 0x80);/*设DLAB=1*/ outportb(PORT[PORTNUM] + 1, 0X00);/*高八位设0*/ switch(baund){ case 2400:b = 0x30;break; case 3600:b = 0x20;break; case 4800:b = 0x18;break; case 7200:b = 0x10;break; case 9600:b = 0x0c;break; } outportb(PORT[PORTNUM], b); enable(); } void closeCom(){ disable(); setvect(INTR[PORTNUM],oldhandle); enable(); } void setDefineOpion(){ } void readOpion(){ } int AcceptOrRefuse(){ int index=0; int key = 0; int ret; char str[2][7]={"accpet","refuse"}; while(key!=13) { gotoxy(12,5+index); textcolor(15); printf("%s",str[index]); index=1-index; gotoxy(12,5+index); textcolor(13); printf("%s",str[index]); key=bioskey(0); key=key&0xff; } if (index == 0){ ret = 0; strcpy(sendmsg, "_accept"); strcat(sendmsg, "\0"); sendMessage(); } else { ret = 1; strcpy(sendmsg, "_refuse"); strcat(sendmsg, "\0"); sendMessage();
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值