一、Regmap SPI
Regmap是一個與設備寄存器通訊的抽象層,它可以用作寄存器I/O,收發I2C/SPI數據,讀取/寫入寄存器等。Regmap的一大優勢是可以支持多種通訊總線,包括I2C、SPI、MMIO等。在SPI總線上使用Regmap時,需要通過spi_new_device函數來註冊spi_device,同時提供寄存器編碼信息和相關回調函數。以下是SPI接口的使用示例:
static int regmap_spi_probe(struct spi_device *spi)
{
int ret = 0;
struct regmap_config config = {
.reg_bits = 8,
.val_bits = 8,
};
struct regmap *regmap;
regmap = devm_regmap_init_spi(spi, &config);
if (IS_ERR(regmap)) {
return PTR_ERR(regmap);
}
/* Do something with regmap */
return ret;
}
static struct spi_driver regmap_spi_driver = {
.driver = {
.name = "regmap-spi",
.owner = THIS_MODULE,
},
.probe = regmap_spi_probe,
};
module_spi_driver(regmap_spi_driver);
二、Regmap子系統詳解
Regmap子系統是Regmap的一個重要組成部分,它維護了一個通過Regmap接口訪問的設備寄存器映射表,將硬件寄存器和內存地址進行匹配。Regmap的客戶端通過調用regmap_read/regmap_write函數來進行讀寫操作。以下是Regmap子系統的使用示例:
static const struct regmap_config my_config = {
.name = "my-config",
.val_bits = 8,
.reg_bits = 8,
};
static int regmap_probe(struct spi_device *spi)
{
struct regmap *regmap;
regmap = devm_regmap_init_spi(spi, &my_config);
if (IS_ERR(regmap))
return PTR_ERR(regmap);
/* Do something with regmap */
return 0;
}
static struct spi_driver regmap_driver = {
.driver = {
.name = "my-regmap",
.owner = THIS_MODULE,
},
.probe = regmap_probe,
};
module_spi_driver(regmap_driver);
三、Regmap_read
Regmap_read函數是Regmap的一個基本函數,用於從硬件寄存器中讀取數據。以下是Regmap_read的代碼示例:
int regmap_read(struct regmap *map, unsigned int reg, unsigned int *val)
其中map是Regmap的實例,reg是寄存器地址,val是存放讀取結果的指針。使用時,可以通過函數返回值來判斷讀取結果是否正確。以下是使用Regmap_read的示例:
static int my_driver_probe(struct platform_device *pdev)
{
int ret;
struct regmap *map;
unsigned int val;
/* Initialize regmap with appropriate config */
ret = regmap_read(map, 0x10, &val);
if (ret) {
dev_err(&pdev->dev, "Failed to read from reg 0x10: %d\n", ret);
return ret;
}
/* Do something with the read value */
return 0;
}
四、Regmap_write
Regmap_write函數是Regmap的另一個基本函數,用於向硬件寄存器中寫入數據。以下是Regmap_write的代碼示例:
int regmap_write(struct regmap *map, unsigned int reg, unsigned int val)
其中map是Regmap的實例,reg是寄存器地址,val是要寫入的數據。使用時,同樣可以通過函數返回值來判斷寫入結果是否正確。以下是使用Regmap_write的示例:
static int my_driver_probe(struct platform_device *pdev)
{
int ret;
struct regmap *map;
/* Initialize regmap with appropriate config */
ret = regmap_write(map, 0x10, 0x1234);
if (ret) {
dev_err(&pdev->dev, "Failed to write to reg 0x10: %d\n", ret);
return ret;
}
/* Do something after write operation */
return 0;
}
五、Regmap_寫16位
Regmap_寫16位函數是Regmap的一個特殊函數,專門用於向寄存器中寫入16位數據。如果使用Regmap_write函數寫入16位數據,將需要兩次寫入操作。以下是Regmap_寫16位的代碼示例:
int regmap_write16(struct regmap *map, unsigned int reg, unsigned int val)
其中map是Regmap的實例,reg是寄存器地址,val是要寫入的數據。同樣可以通過函數返回值來判斷寫入結果是否正確。以下是使用Regmap_寫16位的示例:
static int my_driver_probe(struct platform_device *pdev)
{
int ret;
struct regmap *map;
/* Initialize regmap with appropriate config */
ret = regmap_write16(map, 0x10, 0x1234);
if (ret) {
dev_err(&pdev->dev, "Failed to write to reg 0x10: %d\n", ret);
return ret;
}
/* Do something after write operation */
return 0;
}
六、Regmap使用的多嗎
Regmap是一個非常流行和廣泛使用的Linux內核組件,因為它可以讓驅動程序更加靈活地與具體硬件平台進行通訊。如果硬件平台發生變化,只需更新Regmap的配置和映射表即可,而不需要修改驅動程序本身。Regmap的使用頻率非常高,被廣泛應用於各種硬件設備的驅動程序中,例如I2C、SPI、MMIO、PCI和USB等。
七、Regmap_init_mmio
Regmap_init_mmio函數用於初始化Regmap的實例,並與內存映射IO(MMIO)地址等參數進行關聯。以下是Regmap_init_mmio的代碼示例:
struct regmap *regmap_init_mmio(const char *name,
void __iomem *base,
const struct regmap_config *config)
其中name是Regmap的名字,base是內存映射的地址,config確定了Regmap的配置。使用示例如下:
static int my_driver_probe(struct platform_device *pdev)
{
int ret;
struct regmap *map;
void __iomem *base;
/* Initialize base and map */
map = regmap_init_mmio("my-regmap", base, &my_config);
if (IS_ERR(map)) {
dev_err(&pdev->dev, "Failed to init regmap: %ld\n", PTR_ERR(map));
return PTR_ERR(map);
}
/* Do something after initialization */
return 0;
}
八、Regmap_update_bits
Regmap_update_bits函數用於將指定寄存器的指定位的值更新為指定值。以下是Regmap_update_bits的代碼示例:
int regmap_update_bits(struct regmap *map, unsigned int reg,
unsigned int mask, unsigned int val)
其中map是Regmap的實例,reg是寄存器地址,mask是待更新的位掩碼,val是待更新的值。使用示例如下:
static int my_driver_probe(struct platform_device *pdev)
{
int ret;
struct regmap *map;
/* Initialize regmap with appropriate config */
/* Update bits of register 0x10 from bit 4 to bit 7 to value 0x7 */
ret = regmap_update_bits(map, 0x10, 0xf0, 0x70);
if (ret) {
dev_err(&pdev->dev, "Failed to update bits: %d\n", ret);
return ret;
}
/* Do something after bits update */
return 0;
}
九、Regmap_read_devmem2
Regmap_read_devmem2函數用於從指定內存地址讀取數據。以下是Regmap_read_devmem2的代碼示例:
int regmap_read_devmem2(unsigned int base_addr, unsigned int reg, unsigned int *val)
其中base_addr是內存地址,reg是寄存器地址,val是指向讀取值的指針。使用示例如下:
static int my_driver_probe(struct platform_device *pdev)
{
int ret;
unsigned int val;
/* Read from a specific memory address */
ret = regmap_read_devmem2(0x80000000, 0x10, &val);
if (ret) {
dev_err(&pdev->dev, "Failed to read from memory: %d\n", ret);
return ret;
}
/* Do something with the read value */
return 0;
}
十、Regmap_add_irq_chip
Regmap_add_irq_chip函數用於將一個硬件中斷代碼(ISR)與一個設備相關聯。以下是Regmap_add_irq_chip的代碼示例:
int regmap_add_irq_chip(struct device *dev, unsigned int irq,
struct regmap *regmap,
irq_flow_handler_t handler,
void *data)
其中dev是設備結構體指針,irq是中斷號,regmap是Regmap的實例,handler是中斷處理函數,data是中斷處理函數的私有數據。使用示例如下:
static int my_driver_probe(struct platform_device *pdev)
{
int ret;
struct regmap *map;
/* Initialize regmap with appropriate config */
/* Add an IRQ chip for this device */
ret = regmap_add_irq_chip(&pdev->dev, irq_num, map, my_isr, my_data);
if (ret) {
dev_err(&pdev->dev, "Failed to add IRQ chip: %d\n", ret);
return ret;
}
/* Do something after IRQ chip added */
return 0;
}
原創文章,作者:小藍,如若轉載,請註明出處:https://www.506064.com/zh-hk/n/297962.html