2 * BQ27xxx battery monitor I2C driver
4 * Copyright (C) 2015 Texas Instruments Incorporated - http://www.ti.com/
5 * Andrew F. Davis <afd@ti.com>
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License version 2 as
9 * published by the Free Software Foundation.
11 * This program is distributed "as is" WITHOUT ANY WARRANTY of any
12 * kind, whether express or implied; without even the implied warranty
13 * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
17 #include <linux/i2c.h>
18 #include <linux/interrupt.h>
19 #include <linux/module.h>
20 #include <asm/unaligned.h>
22 #include <linux/power/bq27xxx_battery.h>
24 static DEFINE_IDR(battery_id
);
25 static DEFINE_MUTEX(battery_mutex
);
27 static irqreturn_t
bq27xxx_battery_irq_handler_thread(int irq
, void *data
)
29 struct bq27xxx_device_info
*di
= data
;
31 bq27xxx_battery_update(di
);
36 static int bq27xxx_battery_i2c_read(struct bq27xxx_device_info
*di
, u8 reg
,
39 struct i2c_client
*client
= to_i2c_client(di
->dev
);
40 struct i2c_msg msg
[2];
41 unsigned char data
[2];
47 msg
[0].addr
= client
->addr
;
50 msg
[0].len
= sizeof(reg
);
51 msg
[1].addr
= client
->addr
;
52 msg
[1].flags
= I2C_M_RD
;
59 ret
= i2c_transfer(client
->adapter
, msg
, ARRAY_SIZE(msg
));
64 ret
= get_unaligned_le16(data
);
71 static int bq27xxx_battery_i2c_probe(struct i2c_client
*client
,
72 const struct i2c_device_id
*id
)
74 struct bq27xxx_device_info
*di
;
79 /* Get new ID for the new battery device */
80 mutex_lock(&battery_mutex
);
81 num
= idr_alloc(&battery_id
, client
, 0, 0, GFP_KERNEL
);
82 mutex_unlock(&battery_mutex
);
86 name
= devm_kasprintf(&client
->dev
, GFP_KERNEL
, "%s-%d", id
->name
, num
);
90 di
= devm_kzalloc(&client
->dev
, sizeof(*di
), GFP_KERNEL
);
95 di
->dev
= &client
->dev
;
96 di
->chip
= id
->driver_data
;
98 di
->bus
.read
= bq27xxx_battery_i2c_read
;
100 ret
= bq27xxx_battery_setup(di
);
104 /* Schedule a polling after about 1 min */
105 schedule_delayed_work(&di
->work
, 60 * HZ
);
107 i2c_set_clientdata(client
, di
);
110 ret
= devm_request_threaded_irq(&client
->dev
, client
->irq
,
111 NULL
, bq27xxx_battery_irq_handler_thread
,
115 dev_err(&client
->dev
,
116 "Unable to register IRQ %d error %d\n",
128 mutex_lock(&battery_mutex
);
129 idr_remove(&battery_id
, num
);
130 mutex_unlock(&battery_mutex
);
135 static int bq27xxx_battery_i2c_remove(struct i2c_client
*client
)
137 struct bq27xxx_device_info
*di
= i2c_get_clientdata(client
);
139 bq27xxx_battery_teardown(di
);
141 mutex_lock(&battery_mutex
);
142 idr_remove(&battery_id
, di
->id
);
143 mutex_unlock(&battery_mutex
);
148 static const struct i2c_device_id bq27xxx_i2c_id_table
[] = {
149 { "bq27200", BQ27000
},
150 { "bq27210", BQ27010
},
151 { "bq27500", BQ27500
},
152 { "bq27510", BQ27500
},
153 { "bq27520", BQ27500
},
154 { "bq27530", BQ27530
},
155 { "bq27531", BQ27530
},
156 { "bq27541", BQ27541
},
157 { "bq27542", BQ27541
},
158 { "bq27546", BQ27541
},
159 { "bq27742", BQ27541
},
160 { "bq27545", BQ27545
},
161 { "bq27421", BQ27421
},
162 { "bq27425", BQ27421
},
163 { "bq27441", BQ27421
},
164 { "bq27621", BQ27421
},
167 MODULE_DEVICE_TABLE(i2c
, bq27xxx_i2c_id_table
);
170 static const struct of_device_id bq27xxx_battery_i2c_of_match_table
[] = {
171 { .compatible
= "ti,bq27200" },
172 { .compatible
= "ti,bq27210" },
173 { .compatible
= "ti,bq27500" },
174 { .compatible
= "ti,bq27510" },
175 { .compatible
= "ti,bq27520" },
176 { .compatible
= "ti,bq27530" },
177 { .compatible
= "ti,bq27531" },
178 { .compatible
= "ti,bq27541" },
179 { .compatible
= "ti,bq27542" },
180 { .compatible
= "ti,bq27546" },
181 { .compatible
= "ti,bq27742" },
182 { .compatible
= "ti,bq27545" },
183 { .compatible
= "ti,bq27421" },
184 { .compatible
= "ti,bq27425" },
185 { .compatible
= "ti,bq27441" },
186 { .compatible
= "ti,bq27621" },
189 MODULE_DEVICE_TABLE(of
, bq27xxx_battery_i2c_of_match_table
);
192 static struct i2c_driver bq27xxx_battery_i2c_driver
= {
194 .name
= "bq27xxx-battery",
195 .of_match_table
= of_match_ptr(bq27xxx_battery_i2c_of_match_table
),
197 .probe
= bq27xxx_battery_i2c_probe
,
198 .remove
= bq27xxx_battery_i2c_remove
,
199 .id_table
= bq27xxx_i2c_id_table
,
201 module_i2c_driver(bq27xxx_battery_i2c_driver
);
203 MODULE_AUTHOR("Andrew F. Davis <afd@ti.com>");
204 MODULE_DESCRIPTION("BQ27xxx battery monitor i2c driver");
205 MODULE_LICENSE("GPL");