18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * SPI access for Dialog DA9052 PMICs. 48c2ecf20Sopenharmony_ci * 58c2ecf20Sopenharmony_ci * Copyright(c) 2011 Dialog Semiconductor Ltd. 68c2ecf20Sopenharmony_ci * 78c2ecf20Sopenharmony_ci * Author: David Dajun Chen <dchen@diasemi.com> 88c2ecf20Sopenharmony_ci */ 98c2ecf20Sopenharmony_ci 108c2ecf20Sopenharmony_ci#include <linux/device.h> 118c2ecf20Sopenharmony_ci#include <linux/module.h> 128c2ecf20Sopenharmony_ci#include <linux/input.h> 138c2ecf20Sopenharmony_ci#include <linux/mfd/core.h> 148c2ecf20Sopenharmony_ci#include <linux/spi/spi.h> 158c2ecf20Sopenharmony_ci#include <linux/err.h> 168c2ecf20Sopenharmony_ci 178c2ecf20Sopenharmony_ci#include <linux/mfd/da9052/da9052.h> 188c2ecf20Sopenharmony_ci 198c2ecf20Sopenharmony_cistatic int da9052_spi_probe(struct spi_device *spi) 208c2ecf20Sopenharmony_ci{ 218c2ecf20Sopenharmony_ci struct regmap_config config; 228c2ecf20Sopenharmony_ci int ret; 238c2ecf20Sopenharmony_ci const struct spi_device_id *id = spi_get_device_id(spi); 248c2ecf20Sopenharmony_ci struct da9052 *da9052; 258c2ecf20Sopenharmony_ci 268c2ecf20Sopenharmony_ci da9052 = devm_kzalloc(&spi->dev, sizeof(struct da9052), GFP_KERNEL); 278c2ecf20Sopenharmony_ci if (!da9052) 288c2ecf20Sopenharmony_ci return -ENOMEM; 298c2ecf20Sopenharmony_ci 308c2ecf20Sopenharmony_ci spi->mode = SPI_MODE_0; 318c2ecf20Sopenharmony_ci spi->bits_per_word = 8; 328c2ecf20Sopenharmony_ci spi_setup(spi); 338c2ecf20Sopenharmony_ci 348c2ecf20Sopenharmony_ci da9052->dev = &spi->dev; 358c2ecf20Sopenharmony_ci da9052->chip_irq = spi->irq; 368c2ecf20Sopenharmony_ci 378c2ecf20Sopenharmony_ci spi_set_drvdata(spi, da9052); 388c2ecf20Sopenharmony_ci 398c2ecf20Sopenharmony_ci config = da9052_regmap_config; 408c2ecf20Sopenharmony_ci config.read_flag_mask = 1; 418c2ecf20Sopenharmony_ci config.reg_bits = 7; 428c2ecf20Sopenharmony_ci config.pad_bits = 1; 438c2ecf20Sopenharmony_ci config.val_bits = 8; 448c2ecf20Sopenharmony_ci config.use_single_read = true; 458c2ecf20Sopenharmony_ci config.use_single_write = true; 468c2ecf20Sopenharmony_ci 478c2ecf20Sopenharmony_ci da9052->regmap = devm_regmap_init_spi(spi, &config); 488c2ecf20Sopenharmony_ci if (IS_ERR(da9052->regmap)) { 498c2ecf20Sopenharmony_ci ret = PTR_ERR(da9052->regmap); 508c2ecf20Sopenharmony_ci dev_err(&spi->dev, "Failed to allocate register map: %d\n", 518c2ecf20Sopenharmony_ci ret); 528c2ecf20Sopenharmony_ci return ret; 538c2ecf20Sopenharmony_ci } 548c2ecf20Sopenharmony_ci 558c2ecf20Sopenharmony_ci return da9052_device_init(da9052, id->driver_data); 568c2ecf20Sopenharmony_ci} 578c2ecf20Sopenharmony_ci 588c2ecf20Sopenharmony_cistatic int da9052_spi_remove(struct spi_device *spi) 598c2ecf20Sopenharmony_ci{ 608c2ecf20Sopenharmony_ci struct da9052 *da9052 = spi_get_drvdata(spi); 618c2ecf20Sopenharmony_ci 628c2ecf20Sopenharmony_ci da9052_device_exit(da9052); 638c2ecf20Sopenharmony_ci return 0; 648c2ecf20Sopenharmony_ci} 658c2ecf20Sopenharmony_ci 668c2ecf20Sopenharmony_cistatic const struct spi_device_id da9052_spi_id[] = { 678c2ecf20Sopenharmony_ci {"da9052", DA9052}, 688c2ecf20Sopenharmony_ci {"da9053-aa", DA9053_AA}, 698c2ecf20Sopenharmony_ci {"da9053-ba", DA9053_BA}, 708c2ecf20Sopenharmony_ci {"da9053-bb", DA9053_BB}, 718c2ecf20Sopenharmony_ci {"da9053-bc", DA9053_BC}, 728c2ecf20Sopenharmony_ci {} 738c2ecf20Sopenharmony_ci}; 748c2ecf20Sopenharmony_ci 758c2ecf20Sopenharmony_cistatic struct spi_driver da9052_spi_driver = { 768c2ecf20Sopenharmony_ci .probe = da9052_spi_probe, 778c2ecf20Sopenharmony_ci .remove = da9052_spi_remove, 788c2ecf20Sopenharmony_ci .id_table = da9052_spi_id, 798c2ecf20Sopenharmony_ci .driver = { 808c2ecf20Sopenharmony_ci .name = "da9052", 818c2ecf20Sopenharmony_ci }, 828c2ecf20Sopenharmony_ci}; 838c2ecf20Sopenharmony_ci 848c2ecf20Sopenharmony_cistatic int __init da9052_spi_init(void) 858c2ecf20Sopenharmony_ci{ 868c2ecf20Sopenharmony_ci int ret; 878c2ecf20Sopenharmony_ci 888c2ecf20Sopenharmony_ci ret = spi_register_driver(&da9052_spi_driver); 898c2ecf20Sopenharmony_ci if (ret != 0) { 908c2ecf20Sopenharmony_ci pr_err("Failed to register DA9052 SPI driver, %d\n", ret); 918c2ecf20Sopenharmony_ci return ret; 928c2ecf20Sopenharmony_ci } 938c2ecf20Sopenharmony_ci 948c2ecf20Sopenharmony_ci return 0; 958c2ecf20Sopenharmony_ci} 968c2ecf20Sopenharmony_cisubsys_initcall(da9052_spi_init); 978c2ecf20Sopenharmony_ci 988c2ecf20Sopenharmony_cistatic void __exit da9052_spi_exit(void) 998c2ecf20Sopenharmony_ci{ 1008c2ecf20Sopenharmony_ci spi_unregister_driver(&da9052_spi_driver); 1018c2ecf20Sopenharmony_ci} 1028c2ecf20Sopenharmony_cimodule_exit(da9052_spi_exit); 1038c2ecf20Sopenharmony_ci 1048c2ecf20Sopenharmony_ciMODULE_AUTHOR("David Dajun Chen <dchen@diasemi.com>"); 1058c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("SPI driver for Dialog DA9052 PMIC"); 1068c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL"); 107