注册 platform 设备实验

Linux 系统可以支持不同架构处理器和不同的开发板,例如迅为的 RK3568、2K1000 等开发
板都可以运行 Linux 系统。使用平台总线模型编写驱动程序可以让驱动程序有更好的重用性和
跨平台的能力。比如在 RK3568 开发板上编写的 LED 驱动程序,只需要很小的改动就可以放在
2K1000 开发板上运行。
平台总线模型也叫 platform 总线模型,平台总线是 Linux 虚拟出来的一条总线,并没有实际的 物理总线。平台总线模型把驱动分成了 device 和 driver 两部分,即 platform 设备与 platform 驱
动两部分。device 部分是硬件相关代码,因为不同的开发板或处理器描述硬件资源的代码不通
用,所以这部分代码被归类到 device 部分。driver 部分是通用的驱动逻辑代码,如在驱动代码
中操作 LED 灯亮灭,不同的开发板操作逻辑是一样的,所以把驱动代码单独拿出来归类到 driver
部分可以避免重复造轮子的工作。这样在不同开发板上移植相同驱动的时候,只需要单独编写
device 部分的硬件资源描述代码即可,driver 部分几乎无需改动,从而大幅提升驱动的兼容性
和可移植性。
platform_device.c 代码

cpp 复制代码
#include <linux/module.h>
#include <linux/platform_device.h>
#include <linux/ioport.h>

#define MEM_START_ADDR 0xFDD60000
#define MEM_END_ADDR 0xFDD60004
#define IRQ_NUMBER 101

static struct resource my_resources[] = {
    {   .start = MEM_START_ADDR, // 内存资源起始地址
        .end = MEM_END_ADDR, // 内存资源结束地址
        .flags = IORESOURCE_MEM, // 标记为内存资源

    },
    {
        .start = IRQ_NUMBER, // 中断资源号
        .end = IRQ_NUMBER, // 中断资源号
        .flags = IORESOURCE_IRQ, // 标记为中断资源
    },
};

static void my_platform_device_release(struct device *dev)
{
    // 释放资源的回调函数
}

static struct platform_device my_platform_device = {
    .name = "guosite_platform_device", // 设备名称
    .id = -1, // 设备 ID
    .num_resources = ARRAY_SIZE(my_resources), // 资源数量
    .resource = my_resources, // 资源数组
    .dev.release = my_platform_device_release, // 释放资源的回调函数
};

static int __init my_platform_device_init(void)
{
    int ret;
    ret = platform_device_register(&my_platform_device); // 注册平台设备
    if (ret) {
        printk(KERN_ERR "Failed to register platform device\n");
        return ret;
    }
    printk(KERN_INFO "Platform device registered\n");
    return 0;
}

static void __exit my_platform_device_exit(void)
{
    platform_device_unregister(&my_platform_device); // 注销平台设备
    printk(KERN_INFO "Platform device unregistered\n");
}

module_init(my_platform_device_init);
module_exit(my_platform_device_exit);

MODULE_LICENSE("GPL v2");
MODULE_AUTHOR("topeet");

Makefile

cpp 复制代码
# 移除ARM64架构和交叉编译器配置,默认使用x86本地编译环境
# export ARCH=arm64  # 注释掉ARM64架构设置
# export CROSS_COMPILE=aarch64-linux-gnu-  # 注释掉交叉编译器前缀

obj-m += platform_devices.o  # 驱动源文件名称,保持不变

# 修改内核目录为x86系统的本地内核源码/头文件目录
# 方案1:使用系统当前运行内核的头文件(推荐,无需手动下载内核源码)
KDIR := /lib/modules/$(shell uname -r)/build
# 方案2:如果你有本地下载的x86内核源码,替换成对应的路径,例如:
# KDIR := /home/yourname/linux-x86-kernel  # 根据实际路径修改

PWD ?= $(shell pwd)

all:
	make -C $(KDIR) M=$(PWD) modules  # 本地编译x86内核模块

clean:
	make -C $(KDIR) M=$(PWD) clean  # 清理编译产物


cd /sys/bus/platform/devices

ls guosite_platform_device


sudo rmmod platform_devices.ko

相关推荐
LUCIFER7 小时前
[驱动进阶——MIPI摄像头驱动(五)]rk3588+OV13855摄像头驱动加载过程详细解析第四部分——ISP驱动
linux·驱动开发
暮云星影7 小时前
四、linux系统 应用开发:UI开发环境配置概述 (一)
linux·ui·arm
a程序小傲8 小时前
得物Java面试被问:RocketMQ的消息轨迹追踪实现
java·linux·spring·面试·职场和发展·rocketmq·java-rocketmq
Ghost Face...8 小时前
i386 CPU页式存储管理深度解析
java·linux·服务器
LEEE@FPGA8 小时前
zynq 是不是有了设备树,再linux中不需要编写驱动也能控制
linux·运维·单片机
RisunJan8 小时前
Linux命令-less(分页查看器)
linux·运维
梁正雄9 小时前
linux服务-MariaDB 10.6 Galera Cluster+garbd
linux·运维·mariadb
Coder个人博客9 小时前
Linux6.19-ARM64 mm mem_encrypt子模块深入分析
linux·安全·车载系统·系统架构·系统安全·鸿蒙系统·安全架构
hweiyu009 小时前
Linux 命令:fold
linux·运维
搬砖者(视觉算法工程师)9 小时前
Ubuntu 24.04 LTS 系统上树莓派摄像头模块 v2.1(IMX219)的安装配置与故障排查
linux·数据库·ubuntu