18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later 28c2ecf20Sopenharmony_ci /* I2C access for DA9055 PMICs. 38c2ecf20Sopenharmony_ci * 48c2ecf20Sopenharmony_ci * Copyright(c) 2012 Dialog Semiconductor Ltd. 58c2ecf20Sopenharmony_ci * 68c2ecf20Sopenharmony_ci * Author: David Dajun Chen <dchen@diasemi.com> 78c2ecf20Sopenharmony_ci */ 88c2ecf20Sopenharmony_ci 98c2ecf20Sopenharmony_ci#include <linux/module.h> 108c2ecf20Sopenharmony_ci#include <linux/device.h> 118c2ecf20Sopenharmony_ci#include <linux/i2c.h> 128c2ecf20Sopenharmony_ci#include <linux/err.h> 138c2ecf20Sopenharmony_ci#include <linux/of.h> 148c2ecf20Sopenharmony_ci#include <linux/of_device.h> 158c2ecf20Sopenharmony_ci 168c2ecf20Sopenharmony_ci#include <linux/mfd/da9055/core.h> 178c2ecf20Sopenharmony_ci 188c2ecf20Sopenharmony_cistatic int da9055_i2c_probe(struct i2c_client *i2c, 198c2ecf20Sopenharmony_ci const struct i2c_device_id *id) 208c2ecf20Sopenharmony_ci{ 218c2ecf20Sopenharmony_ci struct da9055 *da9055; 228c2ecf20Sopenharmony_ci int ret; 238c2ecf20Sopenharmony_ci 248c2ecf20Sopenharmony_ci da9055 = devm_kzalloc(&i2c->dev, sizeof(struct da9055), GFP_KERNEL); 258c2ecf20Sopenharmony_ci if (!da9055) 268c2ecf20Sopenharmony_ci return -ENOMEM; 278c2ecf20Sopenharmony_ci 288c2ecf20Sopenharmony_ci da9055->regmap = devm_regmap_init_i2c(i2c, &da9055_regmap_config); 298c2ecf20Sopenharmony_ci if (IS_ERR(da9055->regmap)) { 308c2ecf20Sopenharmony_ci ret = PTR_ERR(da9055->regmap); 318c2ecf20Sopenharmony_ci dev_err(&i2c->dev, "Failed to allocate register map: %d\n", 328c2ecf20Sopenharmony_ci ret); 338c2ecf20Sopenharmony_ci return ret; 348c2ecf20Sopenharmony_ci } 358c2ecf20Sopenharmony_ci 368c2ecf20Sopenharmony_ci da9055->dev = &i2c->dev; 378c2ecf20Sopenharmony_ci da9055->chip_irq = i2c->irq; 388c2ecf20Sopenharmony_ci 398c2ecf20Sopenharmony_ci i2c_set_clientdata(i2c, da9055); 408c2ecf20Sopenharmony_ci 418c2ecf20Sopenharmony_ci return da9055_device_init(da9055); 428c2ecf20Sopenharmony_ci} 438c2ecf20Sopenharmony_ci 448c2ecf20Sopenharmony_cistatic int da9055_i2c_remove(struct i2c_client *i2c) 458c2ecf20Sopenharmony_ci{ 468c2ecf20Sopenharmony_ci struct da9055 *da9055 = i2c_get_clientdata(i2c); 478c2ecf20Sopenharmony_ci 488c2ecf20Sopenharmony_ci da9055_device_exit(da9055); 498c2ecf20Sopenharmony_ci 508c2ecf20Sopenharmony_ci return 0; 518c2ecf20Sopenharmony_ci} 528c2ecf20Sopenharmony_ci 538c2ecf20Sopenharmony_ci/* 548c2ecf20Sopenharmony_ci * DO NOT change the device Ids. The naming is intentionally specific as both 558c2ecf20Sopenharmony_ci * the PMIC and CODEC parts of this chip are instantiated separately as I2C 568c2ecf20Sopenharmony_ci * devices (both have configurable I2C addresses, and are to all intents and 578c2ecf20Sopenharmony_ci * purposes separate). As a result there are specific DA9055 ids for PMIC 588c2ecf20Sopenharmony_ci * and CODEC, which must be different to operate together. 598c2ecf20Sopenharmony_ci */ 608c2ecf20Sopenharmony_cistatic const struct i2c_device_id da9055_i2c_id[] = { 618c2ecf20Sopenharmony_ci {"da9055-pmic", 0}, 628c2ecf20Sopenharmony_ci { } 638c2ecf20Sopenharmony_ci}; 648c2ecf20Sopenharmony_ciMODULE_DEVICE_TABLE(i2c, da9055_i2c_id); 658c2ecf20Sopenharmony_ci 668c2ecf20Sopenharmony_cistatic const struct of_device_id da9055_of_match[] = { 678c2ecf20Sopenharmony_ci { .compatible = "dlg,da9055-pmic", }, 688c2ecf20Sopenharmony_ci { } 698c2ecf20Sopenharmony_ci}; 708c2ecf20Sopenharmony_ci 718c2ecf20Sopenharmony_cistatic struct i2c_driver da9055_i2c_driver = { 728c2ecf20Sopenharmony_ci .probe = da9055_i2c_probe, 738c2ecf20Sopenharmony_ci .remove = da9055_i2c_remove, 748c2ecf20Sopenharmony_ci .id_table = da9055_i2c_id, 758c2ecf20Sopenharmony_ci .driver = { 768c2ecf20Sopenharmony_ci .name = "da9055-pmic", 778c2ecf20Sopenharmony_ci .of_match_table = of_match_ptr(da9055_of_match), 788c2ecf20Sopenharmony_ci }, 798c2ecf20Sopenharmony_ci}; 808c2ecf20Sopenharmony_ci 818c2ecf20Sopenharmony_cistatic int __init da9055_i2c_init(void) 828c2ecf20Sopenharmony_ci{ 838c2ecf20Sopenharmony_ci int ret; 848c2ecf20Sopenharmony_ci 858c2ecf20Sopenharmony_ci ret = i2c_add_driver(&da9055_i2c_driver); 868c2ecf20Sopenharmony_ci if (ret != 0) { 878c2ecf20Sopenharmony_ci pr_err("DA9055 I2C registration failed %d\n", ret); 888c2ecf20Sopenharmony_ci return ret; 898c2ecf20Sopenharmony_ci } 908c2ecf20Sopenharmony_ci 918c2ecf20Sopenharmony_ci return 0; 928c2ecf20Sopenharmony_ci} 938c2ecf20Sopenharmony_cisubsys_initcall(da9055_i2c_init); 948c2ecf20Sopenharmony_ci 958c2ecf20Sopenharmony_cistatic void __exit da9055_i2c_exit(void) 968c2ecf20Sopenharmony_ci{ 978c2ecf20Sopenharmony_ci i2c_del_driver(&da9055_i2c_driver); 988c2ecf20Sopenharmony_ci} 998c2ecf20Sopenharmony_cimodule_exit(da9055_i2c_exit); 1008c2ecf20Sopenharmony_ci 1018c2ecf20Sopenharmony_ciMODULE_AUTHOR("David Dajun Chen <dchen@diasemi.com>"); 1028c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("I2C driver for Dialog DA9055 PMIC"); 1038c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL"); 104