字符设备(6)阻塞IO

/*
只是写了读操作方面,写操作还没写
	阻塞型IO,在等待队列上睡眠(初始化一个wait_queue 加入等待队列头的链表中)
        非阻塞型IO ,判断O_NONBLOCK标志
*/
#include <linux/module.h>
#include <linux/init.h>
#include <linux/kernel.h>
#include <linux/cdev.h>
#include <linux/fs.h>
#include <asm/uaccess.h>
#include <linux/errno.h>
#include <linux/slab.h>

#include <linux/wait.h>
#include <linux/sched.h>


#define P_DEBUG(fmt, args...) printk("<kernel>[%s]:"fmt, __FUNCTION__, ##args)
#define DEVSIZE 512
//将设备属性封装成一个结构体,表示一个字符设备
struct test_dev_t {
	char kbuf[DEVSIZE];
	int curr_size;
	int major;
	int minor;
	dev_t devno;
	struct cdev test_cdev;
	wait_queue_head_t queue;//定义等待队列头
}*dev;
/*
static dev_t devno;
static int major = 0;
static int minor = 0;
struct cdev test_cdev;
*/
//设备驱动的操作方法
static int test_open(struct inode *,struct file *);
static int test_close(struct inode *,struct file *);
static ssize_t test_read(struct file *, char __user *, size_t , loff_t *);
static ssize_t test_write(struct file *, const char __user *, size_t, loff_t *);


struct file_operations test_fops = {
	.open = test_open,
	.release = test_close,
	.read = test_read,
	.write = test_write,
};
//打开设备
static int test_open(struct inode *inode,struct file *filp)
{
	P_DEBUG("dev open!\n");	

	struct test_dev_t *dev;
	dev = container_of(inode->i_cdev,struct test_dev_t,test_cdev);
	filp->private_data = dev;//利用设备文件的私有数据结构,保存设备相关的数据结构

	return 0;
}
//释放设备
static int test_close(struct inode *inode,struct file *filp)
{
	P_DEBUG("dev close!\n");
	return 0;
}
//从设备读取
static ssize_t test_read(struct file *filp, char __user *buf, size_t count, loff_t *offset)
{
	int ret;
	
	struct test_dev_t *dev = filp->private_data; 
	
	if (*offset > DEVSIZE) {
		return count ? -ENXIO : 0;
	}
	
	if (*offset + count > DEVSIZE) {
		count = DEVSIZE - *offset;
	} 

	//非阻塞,默认阻塞
	filp->f_flags |= O_NONBLOCK;
	if (filp->f_flags & O_NONBLOCK)
		return -EAGAIN;		

	P_DEBUG("wait read data....\n");
	//阻塞,条件为假,读进程睡眠
	if (wait_event_interruptible(dev->queue,dev->curr_size > 0)) {
		ret = -ERESTARTSYS;
		goto RET;
	}

	if (copy_to_user(buf, dev->kbuf + *offset, count)) {
		return -EFAULT;
	}
	else {
		P_DEBUG("offset is [%d]\n", *offset);
		ret = count;

		dev->curr_size -= count;
		*offset += count;
		P_DEBUG("buf is [%s]\n", buf);
	}
RET:
	return ret; 
} 

//写入设备
static ssize_t test_write(struct file *filp, const char __user *buf, size_t count, loff_t *offset)
{
	int ret;	

	struct test_dev_t *dev = filp->private_data;
	P_DEBUG("dev->curr_size[%d]\n",dev->curr_size);

	if (*offset > DEVSIZE) {
		return count ? -ENXIO : 0;
	}
	
	if (*offset + count > DEVSIZE) { 
		count = DEVSIZE - *offset;
	} 

	if (copy_from_user(dev->kbuf,buf,count)) {
		return -EFAULT;
	}
	else {
		ret = count;
		dev->curr_size += count;
		*offset += count;
		P_DEBUG("dev->curr_size[%d]\n",dev->curr_size);	
 		P_DEBUG("kbuf is [%s]\n", dev->kbuf); 
	}

	wake_up_interruptible(&dev->queue);//唤醒睡眠的读进程
 	return ret;
 }

                                        
//初始化设备属性
static struct test_dev_t* dev_init(struct test_dev_t *dev)
{
	dev = kmalloc(sizeof(struct test_dev_t),GFP_KERNEL);
	memset(dev->kbuf,0,DEVSIZE);
	dev->curr_size = 0;
	dev->major = 0;
	dev->minor = 0;
	
	return dev;
}

static int __init test_init(void)
{
	int ret;
		
	dev = dev_init(dev);
	//申请字符设备号
	if (dev->major) {
		dev->devno = MKDEV(dev->major,dev->minor);
		ret = register_chrdev_region(dev->devno,1,"test_dev");
	}
	else {//动态分配
		ret = alloc_chrdev_region(&dev->devno,dev->minor,1,"test_dev");
		dev->major = MAJOR(dev->devno);
		dev->minor = MINOR(dev->devno);
	}	
	
	if (ret < 0) {
		P_DEBUG("register chrdev devno err!\n");
		goto err_regDevno;
	}
	
	printk("major[%d] minor[%d]\n",dev->major,dev->minor);

	
	cdev_init(&dev->test_cdev,&test_fops);
	dev->test_cdev.owner = THIS_MODULE;

	init_waitqueue_head(&dev->queue);//初始化等待队列头

	ret = cdev_add(&dev->test_cdev,dev->devno,1);

	if (ret < 0) {
		P_DEBUG("add cdev error!\n");
		goto err_addCdev;
	}
	
	return 0;
err_addCdev:
	unregister_chrdev_region(dev->devno,1);	
err_regDevno:
	return ret;
}

static int __exit test_exit(void)
{
	cdev_del(&dev->test_cdev);
	unregister_chrdev_region(dev->devno,1);

	return 0;
}

module_init(test_init);
module_exit(test_exit);

MODULE_LICENSE("GPL");
MODULE_AUTHOR("LL");
MODULE_VERSION("1.4");


测试

1.读阻塞,等待写数据的到来

#include <stdio.h>
#include <fcntl.h>

#include <sys/types.h>
#include <errno.h>

int main(void)
{
	int fd ,ret;
	char buf[10];
	fd = open("/dev/test",O_RDWR);
	if (fd <0) {
		perror("open");	
		return -1;
	}

	ret = read(fd,buf,10);
	if (ret == -1) { 
		perror("read");
		printf("errno = %d\n",errno); //read 返回的错误errno
	} 
	else {
		printf("buf is [%s]\n\n",buf);
	}

	close(fd);
	return ret;
}

2. 写数据,唤醒等待的读

#include <stdio.h>
#include <fcntl.h>

#include <sys/types.h>

int main(void)
{
	int fd;
	char *buf = "1234567";
	fd = open("/dev/test",O_RDWR);
	if (fd < 0) {
		perror("open");	
		return -1;
	}

	write(fd,buf,8);

	close(fd);
	return 0;
}


基于SSM框架的智能家政保洁预约系统,是一个旨在提高家政保洁服务预约效率和管理水平的平台。该系统通过集成现代信息技术,为家政公司、家政服务人员和消费者提供了一个便捷的在线预约和管理系统。 系统的主要功能包括: 1. **用户管理**:允许消费者注册、登录,并管理他们的个人资料和预约历史。 2. **家政人员管理**:家政服务人员可以注册并更新自己的个人信息、服务类别和服务时间。 3. **服务预约**:消费者可以浏览不同的家政服务选项,选择合适的服务人员,并在线预约服务。 4. **订单管理**:系统支持订单的创建、跟踪和管理,包括订单的确认、完成和评价。 5. **评价系统**:消费者可以在家政服务完成后对服务进行评价,帮助提高服务质量和透明度。 6. **后台管理**:管理员可以管理用户、家政人员信息、服务类别、预约订单以及处理用户反馈。 系统采用Java语言开发,使用MySQL数据库进行数据存储,通过B/S架构实现用户与服务的在线交互。系统设计考虑了不同用户角色的需求,包括管理员、家政服务人员和普通用户,每个角色都有相应的权限和功能。此外,系统还采用了软件组件化、精化体系结构、分离逻辑和数据等方法,以便于未来的系统升级和维护。 智能家政保洁预约系统通过提供一个集中的平台,不仅方便了消费者的预约和管理,也为家政服务人员提供了一个展示和推广自己服务的机会。同时,系统的后台管理功能为家政公司提供了强大的数据支持和决策辅助,有助于提高服务质量和管理效率。该系统的设计与实现,标志着家政保洁服务向现代化和网络化的转型,为管理决策和控制提供保障,是行业发展中的重要里程碑。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值