TEA5676 + AT24C08 FM收音机 搜台 存台 mmap 实现读写

 


硬件说明
TEA5767 + AT24c08
要使用耳机收听,不加功放芯片,声音非常小。

这2个芯片都支持 3.3 或 5.0 电源支持
连线比较简单,sda scl 接到 2440 对应的 排针上,找出一路 3.3v GND ,这里我是从 com 口上接出。
如下接线图,tea5767 at24c08 来自于,之前用 51 单片机做的 FM 收音机,因为电平不同,就把 51 单片机取下来了。
1602 因为 2440 开发板,没有这么多可用 GPIO 这次就不接了。

中间的转换板是我自己焊的,是一个 5V 转 3.3V 和 LED 台灯 和 排针连接工具。 用的是 asm1117 3.3 。
制作中出现过一个严重的问题,led 灯全烧了,之前用 笔记本上面的 usb 供电,led 灯虽然说是亮些,但是没有烧过。
这次新买了一个 台达5v 5a 电源,接上线后全烧了。加了一个100欧,限流电阻后正常了。

编译环境
ubunto 12
arm-linux-gcc 4.3.2

使用说明
本项目基于 linux4.3.36 (2.6 内核肯定是不能用,3.1 之后的也许可以,如果不能用请自行修改)
首先要配置内核启用 i2c 的支持
make menuconfig
-> Device Drivers
   -> I2C support
      -> I2C support (I2C [=y])
         -> I2C Hardware Bus support    

-> Device Drivers
    -> I2C support
       -> I2C support (I2C [=y])


基本上有难度的是 at24c08 的 mmap 实现。 不是普通的读写。 这样做,程序写起来比较简单。

打开 fm 应用程序后,打开 tea5767 和 at24c08 对 at24c08 进行 mmap 映射为 unsigned long 来表现,电台的 频率, 单位是 Khz (因为小数不容易存)。
搜到台以后, 把台的频率(单位是Khz)保存到 mmap 中, 退出的时候,在写回 at24c08 。简化了程序。

内核驱动中需要注意的是, mmap 的 内存要求是物理地址连续,所以使用 dma_alloc_writecombine 分配。

请把 项目中的所有文件,下载到本地,修改 Makefile 中的 内核路径,除非你的和我的一样,这样就不用修改了。
执行
在项目目录中 执行 make
生成 4个 ko 文件 和 fm 应用程序。 把这5个文件复制到,开发板上,或 nfs 中。

启动开发板,请一定要注意,内核有 i2c 硬件支援,然后加载4个ko 文件。
insmod tea5767/tea5767.ko
insmod tea5767/tea5767_dev.ko
insmod tea5767/at24cxx.ko
insmod tea5767/at24cxx_dev.ko

执行测试 fm 程序
tea5767/fm

输入 s 按回车 自动搜台
搜完毕后,自动打开第1个台
可以按 n 切换下一台
可以按 p 切换上一台
可以按 q 退出
可以按 h 查看帮助信息
可以按 t 进行测试模式 这里收听一个 北京台 99.6Mhz

主要发下 at24c08 的 mmap 实现

#include <linux/i2c.h>
#include <linux/init.h>
#include <linux/module.h>
#include <linux/of.h>
#include <linux/slab.h>
#include <linux/spinlock.h>
#include <linux/sysfs.h>
#include <asm/uaccess.h>
#include <linux/fs.h>
#include <linux/mm.h>
#include <linux/mman.h>
#include <linux/mutex.h>
#include <linux/slab.h>
#include <linux/dma-mapping.h>
#include <linux/delay.h>

//主设备号
static int major;
//类描述为创建设备使用
static struct class *cls;
//i2c client 发送接收数据使用
static struct i2c_client *at24cxx_client;

//互斥锁
static struct mutex at24cxx_mutex;

//i2c table 和 board 中进行比较 相同调用 probe
static struct i2c_device_id at24cxx_id_table[] = {
    {"at24c08", 0},
    {}
};

//mmap 使用内存
static unsigned char *at24cxx_buf;
//物理地址 使用和 lcd fb 一样的分配内存的方法 dma_alloc_writecombine
static dma_addr_t at24cxx_phys;

#define AT24cxx_BUF_SIZE 64

static int at24cxx_mmap(struct file *file, struct vm_area_struct *vma)
{
    u64 len = vma->vm_end - vma->vm_start;
    len = min(len, AT24cxx_BUF_SIZE);
    //no cache
    vma->vm_page_prot = pgprot_noncached(vma->vm_page_prot);
    return vm_iomap_memory(vma, at24cxx_phys, len);
}

static int at24cxx_open(struct inode *note, struct file *f)
{
    //打开时 填充 mmap 的buf
    int i;
    memset(at24cxx_buf, 0, AT24cxx_BUF_SIZE);
    for(i=0; i<AT24cxx_BUF_SIZE; i++)
    {
        at24cxx_buf[i] = i2c_smbus_read_byte_data(at24cxx_client, i);
    }
    return 0;
}

static int at24cxx_release(struct inode *note, struct file *f)
{
    //写 mmap 中的数据回 at24c08
    int i;
    for(i=0; i<AT24cxx_BUF_SIZE; i++)
    {
        i2c_smbus_write_byte_data(at24cxx_client, i, at24cxx_buf[i]);
        mdelay(100);
    }
    return 0;
}

static struct file_operations at24cxx_ops = {
    .owner   = THIS_MODULE,
    .open    = at24cxx_open,
    .release = at24cxx_release,
    .mmap    = at24cxx_mmap,
};

//当idtab中比较相同以后调用这个
static int at24cxx_probe(struct i2c_client *client,const struct i2c_device_id *dev)
{
    //初始化锁
    mutex_init(&at24cxx_mutex);
    //设给全局变量
    at24cxx_client = client;
    //创建字库设备
    cls = class_create(THIS_MODULE, "at24cxx");
    major = register_chrdev(0, "at24cxx", &at24cxx_ops);
    device_create(cls, NULL, MKDEV(major, 0), NULL, "at24cxx");

    //vmalloc 分配的物理地址连续
    at24cxx_buf = dma_alloc_writecombine(NULL, AT24cxx_BUF_SIZE, &at24cxx_phys, GFP_KERNEL);
    return 0;
}

static int at24cxx_remove(struct i2c_client *client)
{
    device_destroy(cls, MKDEV(major, 0));
    class_destroy(cls);
    unregister_chrdev(major, "at24cxx");
    dma_free_writecombine(NULL, AT24cxx_BUF_SIZE, at24cxx_buf, at24cxx_phys);
    return 0;
}

static struct i2c_driver at24cxx_drv = {
    .driver = {
        .name  = "at24cxx",
        .owner = THIS_MODULE,
    },
    .probe  = at24cxx_probe,
    .remove = at24cxx_remove,
    .id_table = at24cxx_id_table,
};

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

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

module_init(at24cxx_init);
module_exit(at24cxx_exit);
MODULE_LICENSE("GPL");

 
 GIT 项目地址 https://github.com/nejidev/tea5767_at24c08_fm_radio_linux

posted @ 2017-03-16 22:06  宁次  阅读(1163)  评论(0编辑  收藏  举报