控制三盏灯

驱动代码

#include <linux/init.h>
#include <linux/module.h>
#include <linux/fs.h>
#include <linux/uaccess.h>
#include <linux/io.h>
#include "head.h"

unsigned int major;//保存主设备号
char kbuf[128] = {0};
unsigned int *vir1_moder;
unsigned int *vir1_odr;
unsigned int *vir1_rcc;
unsigned int *vir2_moder;
unsigned int *vir2_odr;
unsigned int *vir2_rcc;
unsigned int *vir3_moder;
unsigned int *vir3_odr;
unsigned int *vir3_rcc;
//封装操作方法
int mycdev_open(struct inode *inode,struct file *file)
{
    printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);
    return 0;
};
ssize_t mycdev_read(struct file *file,char *ubuf,size_t size,loff_t *lof)
{
    int ret;
    ret = copy_to_user(ubuf,kbuf,size);
     if (ret)
    {
        printk("copy_to_user err\n");
        return -EIO;
    }
    return 0;
}
ssize_t mycdev_write(struct file *file,const char *ubuf,size_t size,loff_t *lof)
{
     int ret = 0;
    ret = copy_from_user(kbuf,ubuf,size);
    if (ret)
    {
        printk("copy_from_user err\n");
        return -EIO;
    }
    switch(kbuf[0])
    {
    case '1':
        (*vir1_odr) |= (0x1 << 10);
        break;
    case '2':   
        (*vir1_odr) &= (~(0x1 << 10));
        break;
    case '3':
        (*vir2_odr) |= (0x1 << 10);
        break;
    case '4':   
        (*vir2_odr) &= (~(0x1 << 10));
        break;
    case '5':
        (*vir3_odr) |= (0x1 << 8);
        break;
    case '6':   
        (*vir3_odr) &= (~(0x1 << 8));
        break;
        default:
        printk("输入错误\n");
        break;
     }
    return 0;
}
int mycdev_close(struct inode *inode,struct file *file)
{
    printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);
    return 0;
};
//定义一个操作方法结构体对象并初始化
struct file_operations fops = {
    .open = mycdev_open,
    .read = mycdev_read,
    .write = mycdev_write,
    .release = mycdev_close,
};
static int __init mycdev_init(void)
{
    //字符设备驱动注册
    major = register_chrdev(0,"mychrdev",&fops);
    if(major < 0)
    {
        printk("字符串设备驱动注册失败\n");
    }
    printk("字符设备驱动注册成功 major = %d\n",major);
    //相关寄存器内存映射
    vir1_odr = ioremap(LED1_ODR,4);
    if (vir1_odr == NULL)  
    {   
        printk("物理内存映射失败%d\n",__LINE__);
        return -EFAULT;
    }
    vir1_rcc = ioremap(LED1_RCC,4);
    if (vir1_rcc == NULL)  
    {   
        printk("物理内存映射失败%d\n",__LINE__);
        return -EFAULT;
    }
    vir1_moder = ioremap(LED1_MODER,4);
    if (vir1_moder == NULL)  
    {   
        printk("物理内存映射失败%d\n",__LINE__);
        return -EFAULT;
    }
     vir2_odr = ioremap(LED2_ODR,4);
    if (vir2_odr == NULL)  
    {   
        printk("物理内存映射失败%d\n",__LINE__);
        return -EFAULT;
    }
    vir2_rcc = ioremap(LED2_RCC,4);
    if (vir2_rcc == NULL)  
    {   
        printk("物理内存映射失败%d\n",__LINE__);
        return -EFAULT;
    }
    vir2_moder = ioremap(LED2_MODER,4);
    if (vir2_moder == NULL)  
    {   
        printk("物理内存映射失败%d\n",__LINE__);
        return -EFAULT;
    }
     vir3_odr = ioremap(LED3_ODR,4);
    if (vir3_odr == NULL)  
    {   
        printk("物理内存映射失败%d\n",__LINE__);
        return -EFAULT;
    }
    vir3_rcc = ioremap(LED3_RCC,4);
    if (vir3_rcc == NULL)  
    {   
        printk("物理内存映射失败%d\n",__LINE__);
        return -EFAULT;
    }
    vir3_moder = ioremap(LED3_MODER,4);
    if (vir3_moder == NULL)  
    {   
        printk("物理内存映射失败%d\n",__LINE__);
        return -EFAULT;
    }
    printk("寄存器内存映射成功\n");
    //硬件寄存器的初始化
    //RCC使能
    (*vir1_rcc) |= (0x1 << 4);
    //设置PE10为输出
    (*vir1_moder) &= (~(0x3 << 20));
    (*vir1_moder) |= (0x1 << 20);
    //默认关灯
    (*vir1_odr) &= (~(0x1 << 10));
     //RCC使能
    (*vir2_rcc) |= (0x1 << 5);
    //设置PE10为输出
    (*vir2_moder) &= (~(0x3 << 20));
    (*vir2_moder) |= (0x1 << 20);
    //默认关灯
    (*vir1_odr) &= (~(0x1 << 10));
     //RCC使能
    (*vir1_rcc) |= (0x1 << 4);
    //设置PE10为输出
    (*vir1_moder) &= (~(0x3 << 16));
    (*vir1_moder) |= (0x1 << 16);
    //默认关灯
    (*vir1_odr) &= (~(0x1 << 8));
    
    return 0;
}
static void __exit mycdev_exit(void)
{
    //取消物理内存映射
    iounmap(vir1_moder);
    iounmap(vir1_rcc);
    iounmap(vir1_odr);
    iounmap(vir2_moder);
    iounmap(vir2_rcc);
    iounmap(vir2_odr);
    iounmap(vir3_moder);
    iounmap(vir3_rcc);
    iounmap(vir3_odr);
    //字符设备驱动注销
    unregister_chrdev(major,"mychrdev");

}
module_init(mycdev_init);
module_exit(mycdev_exit);
MODULE_LICENSE("GPL");

应用层代码

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
int main(int argc,const char * argv[])
{
    char buf[128] = {0};
    int fd = open("/dev/mychrdev",O_RDWR);
    if(fd < 0)
    {
        printf("设备文件打开失败\n");
        exit(-1);
    }  
    while(1)
    {
        printf("请输入对LED的控制命令1(LED1开灯)2(LED1关灯)3(LED2开灯)4(LED2关灯)5(LED3开灯)6(LED3关灯)>");
        fgets(buf,sizeof(buf),stdin);//在终端输入数据到buf
        buf[strlen(buf)-1] = '\0';
        write(fd,buf,sizeof(buf));
        memset(buf,0,sizeof(buf));
    }
    close(fd);
    return 0;
}

好的,我来为您设计一个符合要求的PLC梯形图程序。这个程序将控制盏灯的顺序启动和停止。以下是程序的设计思路: 1. 使用定时器T1来控制第1盏灯启动后延时5秒启动第2盏灯。 2. 使用定时器T2来控制第2盏灯启动后延时10秒启动第3盏灯。 3. 使用定时器T3来控制3盏灯同时运行60秒后全部关闭。 4. 使用辅助继电器M1、M2和M3来分别控制3盏灯的开关状态。 5. 使用启动按钮SB1来启动整个程序。 具体的梯形图程序如下: ``` | SB1 |----| |-----------------------( M0 )--------------------| | M0 |----| |-----------------------( 灯1 )----------------| | | | M0 |----| |---[ TON T1, 5s ]---| | | T1 |----| |-----------------------( M1 )--------------------| | | | M1 |----| |---[ TON T2, 10s ]---| | | T2 |----| |-----------------------( M2 )--------------------| | | | M2 |--| |-----------------------( 灯2 )----------------| | | | M3 |----| |-----------------------( 灯3 )----------------| ``` 程序说明: 1. 当按下启动按钮SB1时,M0线圈闭合,启动第1盏灯。 2. 同时,M0也启动了定时器T1。T1计时5秒后,M1线圈闭合,启动第2盏灯。 3. M1启动定时器T2。T2计时10秒后,M2线圈闭合,启动第3盏灯。 4. M2启动定时器T3。T3计时60秒后,M3线圈闭合,关闭所有灯。 5. M3同时也控制了第2和第3盏灯的开关。 6. 当T3计时结束,M3闭合时,所有灯都会关闭,程序回到初始状态。 这个程序满足了您的所有要求,包括顺序启动、延时控制以及总运行时间控制。您可以根据实际需要调整各个定时器的时间参数。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值