blob: c0b883c14f41c15ff5c81084b367b97a13317af0 [file] [log] [blame]
Haojian Zhuangd50f8f32010-01-08 12:29:23 +01001/*
2 * I2C driver for Maxim MAX8925
3 *
4 * Copyright (C) 2009 Marvell International Ltd.
5 * Haojian Zhuang <haojian.zhuang@marvell.com>
6 *
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.
10 */
11#include <linux/kernel.h>
12#include <linux/module.h>
13#include <linux/platform_device.h>
14#include <linux/i2c.h>
15#include <linux/mfd/max8925.h>
16
Haojian Zhuangb13c0df2010-01-25 06:26:34 -050017#define RTC_I2C_ADDR 0x68
18#define ADC_I2C_ADDR 0x47
19
Haojian Zhuangd50f8f32010-01-08 12:29:23 +010020static inline int max8925_read_device(struct i2c_client *i2c,
21 int reg, int bytes, void *dest)
22{
Haojian Zhuangd50f8f32010-01-08 12:29:23 +010023 int ret;
24
Haojian Zhuangb13c0df2010-01-25 06:26:34 -050025 if (bytes > 1)
26 ret = i2c_smbus_read_i2c_block_data(i2c, reg, bytes, dest);
27 else {
28 ret = i2c_smbus_read_byte_data(i2c, reg);
29 if (ret < 0)
30 return ret;
31 *(unsigned char *)dest = (unsigned char)ret;
32 }
33 return ret;
Haojian Zhuangd50f8f32010-01-08 12:29:23 +010034}
35
36static inline int max8925_write_device(struct i2c_client *i2c,
37 int reg, int bytes, void *src)
38{
39 unsigned char buf[bytes + 1];
40 int ret;
41
42 buf[0] = (unsigned char)reg;
43 memcpy(&buf[1], src, bytes);
44
45 ret = i2c_master_send(i2c, buf, bytes + 1);
46 if (ret < 0)
47 return ret;
48 return 0;
49}
50
51int max8925_reg_read(struct i2c_client *i2c, int reg)
52{
53 struct max8925_chip *chip = i2c_get_clientdata(i2c);
Haojian Zhuangb13c0df2010-01-25 06:26:34 -050054 unsigned char data = 0;
Haojian Zhuangd50f8f32010-01-08 12:29:23 +010055 int ret;
56
57 mutex_lock(&chip->io_lock);
58 ret = max8925_read_device(i2c, reg, 1, &data);
59 mutex_unlock(&chip->io_lock);
60
61 if (ret < 0)
62 return ret;
63 else
64 return (int)data;
65}
66EXPORT_SYMBOL(max8925_reg_read);
67
68int max8925_reg_write(struct i2c_client *i2c, int reg,
69 unsigned char data)
70{
71 struct max8925_chip *chip = i2c_get_clientdata(i2c);
72 int ret;
73
74 mutex_lock(&chip->io_lock);
75 ret = max8925_write_device(i2c, reg, 1, &data);
76 mutex_unlock(&chip->io_lock);
77
78 return ret;
79}
80EXPORT_SYMBOL(max8925_reg_write);
81
82int max8925_bulk_read(struct i2c_client *i2c, int reg,
83 int count, unsigned char *buf)
84{
85 struct max8925_chip *chip = i2c_get_clientdata(i2c);
86 int ret;
87
88 mutex_lock(&chip->io_lock);
89 ret = max8925_read_device(i2c, reg, count, buf);
90 mutex_unlock(&chip->io_lock);
91
92 return ret;
93}
94EXPORT_SYMBOL(max8925_bulk_read);
95
96int max8925_bulk_write(struct i2c_client *i2c, int reg,
97 int count, unsigned char *buf)
98{
99 struct max8925_chip *chip = i2c_get_clientdata(i2c);
100 int ret;
101
102 mutex_lock(&chip->io_lock);
103 ret = max8925_write_device(i2c, reg, count, buf);
104 mutex_unlock(&chip->io_lock);
105
106 return ret;
107}
108EXPORT_SYMBOL(max8925_bulk_write);
109
110int max8925_set_bits(struct i2c_client *i2c, int reg,
111 unsigned char mask, unsigned char data)
112{
113 struct max8925_chip *chip = i2c_get_clientdata(i2c);
114 unsigned char value;
115 int ret;
116
117 mutex_lock(&chip->io_lock);
118 ret = max8925_read_device(i2c, reg, 1, &value);
119 if (ret < 0)
120 goto out;
121 value &= ~mask;
122 value |= data;
123 ret = max8925_write_device(i2c, reg, 1, &value);
124out:
125 mutex_unlock(&chip->io_lock);
126 return ret;
127}
128EXPORT_SYMBOL(max8925_set_bits);
129
130
131static const struct i2c_device_id max8925_id_table[] = {
132 { "max8925", 0 },
Haojian Zhuangb13c0df2010-01-25 06:26:34 -0500133 { },
Haojian Zhuangd50f8f32010-01-08 12:29:23 +0100134};
135MODULE_DEVICE_TABLE(i2c, max8925_id_table);
136
137static int __devinit max8925_probe(struct i2c_client *client,
138 const struct i2c_device_id *id)
139{
140 struct max8925_platform_data *pdata = client->dev.platform_data;
Haojian Zhuangb13c0df2010-01-25 06:26:34 -0500141 static struct max8925_chip *chip;
Haojian Zhuangd50f8f32010-01-08 12:29:23 +0100142
143 if (!pdata) {
144 pr_info("%s: platform data is missing\n", __func__);
145 return -EINVAL;
146 }
Haojian Zhuangd50f8f32010-01-08 12:29:23 +0100147
148 chip = kzalloc(sizeof(struct max8925_chip), GFP_KERNEL);
149 if (chip == NULL)
150 return -ENOMEM;
151 chip->i2c = client;
Haojian Zhuangd50f8f32010-01-08 12:29:23 +0100152 chip->dev = &client->dev;
Haojian Zhuangb13c0df2010-01-25 06:26:34 -0500153 i2c_set_clientdata(client, chip);
Haojian Zhuangd50f8f32010-01-08 12:29:23 +0100154 dev_set_drvdata(chip->dev, chip);
Haojian Zhuangb13c0df2010-01-25 06:26:34 -0500155 mutex_init(&chip->io_lock);
156
157 chip->rtc = i2c_new_dummy(chip->i2c->adapter, RTC_I2C_ADDR);
158 i2c_set_clientdata(chip->rtc, chip);
159
160 chip->adc = i2c_new_dummy(chip->i2c->adapter, ADC_I2C_ADDR);
161 i2c_set_clientdata(chip->adc, chip);
162
Haojian Zhuangd50f8f32010-01-08 12:29:23 +0100163 max8925_device_init(chip, pdata);
164
165 return 0;
166}
167
168static int __devexit max8925_remove(struct i2c_client *client)
169{
170 struct max8925_chip *chip = i2c_get_clientdata(client);
171
172 max8925_device_exit(chip);
Haojian Zhuangb13c0df2010-01-25 06:26:34 -0500173 i2c_unregister_device(chip->adc);
174 i2c_unregister_device(chip->rtc);
175 i2c_set_clientdata(chip->adc, NULL);
176 i2c_set_clientdata(chip->rtc, NULL);
177 i2c_set_clientdata(chip->i2c, NULL);
Haojian Zhuangd50f8f32010-01-08 12:29:23 +0100178 kfree(chip);
179 return 0;
180}
181
182static struct i2c_driver max8925_driver = {
183 .driver = {
184 .name = "max8925",
185 .owner = THIS_MODULE,
186 },
187 .probe = max8925_probe,
188 .remove = __devexit_p(max8925_remove),
189 .id_table = max8925_id_table,
190};
191
192static int __init max8925_i2c_init(void)
193{
194 int ret;
195
196 ret = i2c_add_driver(&max8925_driver);
197 if (ret != 0)
198 pr_err("Failed to register MAX8925 I2C driver: %d\n", ret);
199 return ret;
200}
201subsys_initcall(max8925_i2c_init);
202
203static void __exit max8925_i2c_exit(void)
204{
205 i2c_del_driver(&max8925_driver);
206}
207module_exit(max8925_i2c_exit);
208
209MODULE_DESCRIPTION("I2C Driver for Maxim 8925");
210MODULE_AUTHOR("Haojian Zhuang <haojian.zhuang@marvell.com>");
211MODULE_LICENSE("GPL");