blob: 08ce7a105e768d9e2f56533a083989fc97e14e0a [file] [log] [blame]
Dmitry Torokhov203e0ba2018-03-20 15:31:28 -07001// SPDX-License-Identifier: GPL-2.0+
2// Driver to instantiate Chromebook i2c/smbus devices.
3//
4// Copyright (C) 2012 Google, Inc.
5// Author: Benson Leung <bleung@chromium.org>
Benson Leungd1381f42012-10-25 14:21:21 -07006
Dmitry Torokhov4f27f672018-03-20 15:31:30 -07007#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
8
Benson Leungd1381f42012-10-25 14:21:21 -07009#include <linux/dmi.h>
10#include <linux/i2c.h>
Nick Dyer7f3884f72015-08-04 16:36:29 -070011#include <linux/platform_data/atmel_mxt_ts.h>
Benson Leung2ef39202013-03-07 19:43:34 -080012#include <linux/input.h>
13#include <linux/interrupt.h>
Benson Leungd1381f42012-10-25 14:21:21 -070014#include <linux/module.h>
Benson Leung9ad36922013-10-20 20:58:25 -070015#include <linux/platform_device.h>
Benson Leungd1381f42012-10-25 14:21:21 -070016
Benson Leung8e1ad4c2013-02-21 12:14:59 -080017#define ATMEL_TP_I2C_ADDR 0x4b
18#define ATMEL_TP_I2C_BL_ADDR 0x25
Yufeng Shen33a84f82013-02-21 12:15:00 -080019#define ATMEL_TS_I2C_ADDR 0x4a
20#define ATMEL_TS_I2C_BL_ADDR 0x26
Benson Leungd1381f42012-10-25 14:21:21 -070021#define CYAPA_TP_I2C_ADDR 0x67
Benson Leung9bd9a902016-05-02 08:57:16 +080022#define ELAN_TP_I2C_ADDR 0x15
Benson Leungd1381f42012-10-25 14:21:21 -070023#define ISL_ALS_I2C_ADDR 0x44
Benson Leungaabf3f42013-02-01 14:34:45 -080024#define TAOS_ALS_I2C_ADDR 0x29
Benson Leungd1381f42012-10-25 14:21:21 -070025
Benson Leung55024862014-07-15 17:43:11 -070026#define MAX_I2C_DEVICE_DEFERRALS 5
27
Benson Leungd1381f42012-10-25 14:21:21 -070028static struct i2c_client *als;
29static struct i2c_client *tp;
Yufeng Shen33a84f82013-02-21 12:15:00 -080030static struct i2c_client *ts;
Benson Leungd1381f42012-10-25 14:21:21 -070031
Olof Johansson6d3c1af2013-11-25 13:10:25 -080032static const char *i2c_adapter_names[] = {
Benson Leungd1381f42012-10-25 14:21:21 -070033 "SMBus I801 adapter",
Benson Leung741bf0c2013-02-21 12:14:55 -080034 "i915 gmbus vga",
35 "i915 gmbus panel",
Jarkko Nikulaebaf31c2015-11-03 13:09:00 +020036 "Synopsys DesignWare I2C adapter",
37 "Synopsys DesignWare I2C adapter",
Benson Leungd1381f42012-10-25 14:21:21 -070038};
39
40/* Keep this enum consistent with i2c_adapter_names */
41enum i2c_adapter_type {
42 I2C_ADAPTER_SMBUS = 0,
Benson Leung741bf0c2013-02-21 12:14:55 -080043 I2C_ADAPTER_VGADDC,
44 I2C_ADAPTER_PANEL,
Mika Westerbergda3b0ab2014-06-17 14:02:00 -070045 I2C_ADAPTER_DESIGNWARE_0,
46 I2C_ADAPTER_DESIGNWARE_1,
Benson Leungd1381f42012-10-25 14:21:21 -070047};
48
Benson Leung55024862014-07-15 17:43:11 -070049enum i2c_peripheral_state {
50 UNPROBED = 0,
51 PROBED,
52 TIMEDOUT,
Benson Leungd1381f42012-10-25 14:21:21 -070053};
54
Aaron Durbinec199dd2013-10-20 20:58:24 -070055struct i2c_peripheral {
Benson Leung9ad36922013-10-20 20:58:25 -070056 int (*add)(enum i2c_adapter_type type);
Aaron Durbinec199dd2013-10-20 20:58:24 -070057 enum i2c_adapter_type type;
Benson Leung55024862014-07-15 17:43:11 -070058 enum i2c_peripheral_state state;
59 int tries;
Aaron Durbinec199dd2013-10-20 20:58:24 -070060};
61
Benson Leung9bd9a902016-05-02 08:57:16 +080062#define MAX_I2C_PERIPHERALS 4
Aaron Durbinec199dd2013-10-20 20:58:24 -070063
64struct chromeos_laptop {
65 struct i2c_peripheral i2c_peripherals[MAX_I2C_PERIPHERALS];
66};
67
Benson Leung9ad36922013-10-20 20:58:25 -070068static struct chromeos_laptop *cros_laptop;
69
70static struct i2c_board_info cyapa_device = {
Benson Leungd1381f42012-10-25 14:21:21 -070071 I2C_BOARD_INFO("cyapa", CYAPA_TP_I2C_ADDR),
72 .flags = I2C_CLIENT_WAKE,
73};
74
Benson Leung9bd9a902016-05-02 08:57:16 +080075static struct i2c_board_info elantech_device = {
76 I2C_BOARD_INFO("elan_i2c", ELAN_TP_I2C_ADDR),
77 .flags = I2C_CLIENT_WAKE,
78};
79
Benson Leung9ad36922013-10-20 20:58:25 -070080static struct i2c_board_info isl_als_device = {
Benson Leungd1381f42012-10-25 14:21:21 -070081 I2C_BOARD_INFO("isl29018", ISL_ALS_I2C_ADDR),
82};
83
Benson Leung9ad36922013-10-20 20:58:25 -070084static struct i2c_board_info tsl2583_als_device = {
Benson Leung8016bcb2013-02-01 14:34:46 -080085 I2C_BOARD_INFO("tsl2583", TAOS_ALS_I2C_ADDR),
86};
87
Benson Leung9ad36922013-10-20 20:58:25 -070088static struct i2c_board_info tsl2563_als_device = {
Benson Leungaabf3f42013-02-01 14:34:45 -080089 I2C_BOARD_INFO("tsl2563", TAOS_ALS_I2C_ADDR),
90};
91
Nick Dyerfb5e4c3e2014-05-18 23:00:15 -070092static int mxt_t19_keys[] = {
93 KEY_RESERVED,
94 KEY_RESERVED,
95 KEY_RESERVED,
96 KEY_RESERVED,
97 KEY_RESERVED,
98 BTN_LEFT
99};
100
Benson Leung2ef39202013-03-07 19:43:34 -0800101static struct mxt_platform_data atmel_224s_tp_platform_data = {
Benson Leung2ef39202013-03-07 19:43:34 -0800102 .irqflags = IRQF_TRIGGER_FALLING,
Nick Dyerfb5e4c3e2014-05-18 23:00:15 -0700103 .t19_num_keys = ARRAY_SIZE(mxt_t19_keys),
104 .t19_keymap = mxt_t19_keys,
Benson Leung2ef39202013-03-07 19:43:34 -0800105};
106
Benson Leung9ad36922013-10-20 20:58:25 -0700107static struct i2c_board_info atmel_224s_tp_device = {
Benson Leung8e1ad4c2013-02-21 12:14:59 -0800108 I2C_BOARD_INFO("atmel_mxt_tp", ATMEL_TP_I2C_ADDR),
Benson Leung2ef39202013-03-07 19:43:34 -0800109 .platform_data = &atmel_224s_tp_platform_data,
Benson Leung8e1ad4c2013-02-21 12:14:59 -0800110 .flags = I2C_CLIENT_WAKE,
111};
112
Benson Leung2ef39202013-03-07 19:43:34 -0800113static struct mxt_platform_data atmel_1664s_platform_data = {
Benson Leung2ef39202013-03-07 19:43:34 -0800114 .irqflags = IRQF_TRIGGER_FALLING,
Benson Leung2ef39202013-03-07 19:43:34 -0800115};
116
Benson Leung9ad36922013-10-20 20:58:25 -0700117static struct i2c_board_info atmel_1664s_device = {
Yufeng Shen33a84f82013-02-21 12:15:00 -0800118 I2C_BOARD_INFO("atmel_mxt_ts", ATMEL_TS_I2C_ADDR),
Benson Leung2ef39202013-03-07 19:43:34 -0800119 .platform_data = &atmel_1664s_platform_data,
Yufeng Shen33a84f82013-02-21 12:15:00 -0800120 .flags = I2C_CLIENT_WAKE,
121};
122
Benson Leung9ad36922013-10-20 20:58:25 -0700123static struct i2c_client *__add_probed_i2c_device(
Benson Leungd1381f42012-10-25 14:21:21 -0700124 const char *name,
125 int bus,
126 struct i2c_board_info *info,
Dmitry Torokhov96cba9b2015-04-14 13:50:09 -0700127 const unsigned short *alt_addr_list)
Benson Leungd1381f42012-10-25 14:21:21 -0700128{
129 const struct dmi_device *dmi_dev;
130 const struct dmi_dev_onboard *dev_data;
131 struct i2c_adapter *adapter;
Dmitry Torokhov96cba9b2015-04-14 13:50:09 -0700132 struct i2c_client *client = NULL;
133 const unsigned short addr_list[] = { info->addr, I2C_CLIENT_END };
Benson Leungd1381f42012-10-25 14:21:21 -0700134
135 if (bus < 0)
136 return NULL;
137 /*
138 * If a name is specified, look for irq platform information stashed
139 * in DMI_DEV_TYPE_DEV_ONBOARD by the Chrome OS custom system firmware.
140 */
141 if (name) {
142 dmi_dev = dmi_find_device(DMI_DEV_TYPE_DEV_ONBOARD, name, NULL);
143 if (!dmi_dev) {
Dmitry Torokhov4f27f672018-03-20 15:31:30 -0700144 pr_err("failed to dmi find device %s\n", name);
Benson Leungd1381f42012-10-25 14:21:21 -0700145 return NULL;
146 }
147 dev_data = (struct dmi_dev_onboard *)dmi_dev->device_data;
148 if (!dev_data) {
Dmitry Torokhov4f27f672018-03-20 15:31:30 -0700149 pr_err("failed to get data from dmi for %s\n", name);
Benson Leungd1381f42012-10-25 14:21:21 -0700150 return NULL;
151 }
152 info->irq = dev_data->instance;
153 }
154
155 adapter = i2c_get_adapter(bus);
156 if (!adapter) {
Dmitry Torokhov4f27f672018-03-20 15:31:30 -0700157 pr_err("failed to get i2c adapter %d\n", bus);
Benson Leungd1381f42012-10-25 14:21:21 -0700158 return NULL;
159 }
160
Dmitry Torokhov96cba9b2015-04-14 13:50:09 -0700161 /*
162 * Add the i2c device. If we can't detect it at the primary
163 * address we scan secondary addresses. In any case the client
164 * structure gets assigned primary address.
165 */
166 client = i2c_new_probed_device(adapter, info, addr_list, NULL);
167 if (!client && alt_addr_list) {
168 struct i2c_board_info dummy_info = {
169 I2C_BOARD_INFO("dummy", info->addr),
170 };
171 struct i2c_client *dummy;
172
173 dummy = i2c_new_probed_device(adapter, &dummy_info,
174 alt_addr_list, NULL);
175 if (dummy) {
Dmitry Torokhov4f27f672018-03-20 15:31:30 -0700176 pr_debug("%d-%02x is probed at %02x\n",
177 bus, info->addr, dummy->addr);
Dmitry Torokhov96cba9b2015-04-14 13:50:09 -0700178 i2c_unregister_device(dummy);
179 client = i2c_new_device(adapter, info);
180 }
181 }
182
Benson Leungd1381f42012-10-25 14:21:21 -0700183 if (!client)
Dmitry Torokhov4f27f672018-03-20 15:31:30 -0700184 pr_notice("failed to register device %d-%02x\n",
185 bus, info->addr);
Benson Leungd1381f42012-10-25 14:21:21 -0700186 else
Dmitry Torokhov4f27f672018-03-20 15:31:30 -0700187 pr_debug("added i2c device %d-%02x\n", bus, info->addr);
Benson Leungd1381f42012-10-25 14:21:21 -0700188
189 i2c_put_adapter(adapter);
190 return client;
191}
192
Mika Westerbergda3b0ab2014-06-17 14:02:00 -0700193struct i2c_lookup {
194 const char *name;
195 int instance;
196 int n;
197};
198
Benson Leung9ad36922013-10-20 20:58:25 -0700199static int __find_i2c_adap(struct device *dev, void *data)
Benson Leungd1381f42012-10-25 14:21:21 -0700200{
Mika Westerbergda3b0ab2014-06-17 14:02:00 -0700201 struct i2c_lookup *lookup = data;
Benson Leungd1381f42012-10-25 14:21:21 -0700202 static const char *prefix = "i2c-";
203 struct i2c_adapter *adapter;
Robin Schroer49c68a22014-05-29 20:45:07 +0200204
Benson Leungd1381f42012-10-25 14:21:21 -0700205 if (strncmp(dev_name(dev), prefix, strlen(prefix)) != 0)
206 return 0;
207 adapter = to_i2c_adapter(dev);
Mika Westerbergda3b0ab2014-06-17 14:02:00 -0700208 if (strncmp(adapter->name, lookup->name, strlen(lookup->name)) == 0 &&
209 lookup->n++ == lookup->instance)
210 return 1;
211 return 0;
Benson Leungd1381f42012-10-25 14:21:21 -0700212}
213
Benson Leung9ad36922013-10-20 20:58:25 -0700214static int find_i2c_adapter_num(enum i2c_adapter_type type)
Benson Leungd1381f42012-10-25 14:21:21 -0700215{
216 struct device *dev = NULL;
217 struct i2c_adapter *adapter;
Mika Westerbergda3b0ab2014-06-17 14:02:00 -0700218 struct i2c_lookup lookup;
219
220 memset(&lookup, 0, sizeof(lookup));
221 lookup.name = i2c_adapter_names[type];
222 lookup.instance = (type == I2C_ADAPTER_DESIGNWARE_1) ? 1 : 0;
223
Benson Leungd1381f42012-10-25 14:21:21 -0700224 /* find the adapter by name */
Mika Westerbergda3b0ab2014-06-17 14:02:00 -0700225 dev = bus_find_device(&i2c_bus_type, NULL, &lookup, __find_i2c_adap);
Benson Leungd1381f42012-10-25 14:21:21 -0700226 if (!dev) {
Benson Leung9ad36922013-10-20 20:58:25 -0700227 /* Adapters may appear later. Deferred probing will retry */
Dmitry Torokhov4f27f672018-03-20 15:31:30 -0700228 pr_notice("i2c adapter %s not found on system.\n",
Mika Westerbergda3b0ab2014-06-17 14:02:00 -0700229 lookup.name);
Benson Leungd1381f42012-10-25 14:21:21 -0700230 return -ENODEV;
231 }
232 adapter = to_i2c_adapter(dev);
233 return adapter->nr;
234}
235
236/*
Benson Leungbcaf0892013-02-21 12:14:58 -0800237 * Takes a list of addresses in addrs as such :
238 * { addr1, ... , addrn, I2C_CLIENT_END };
239 * add_probed_i2c_device will use i2c_new_probed_device
240 * and probe for devices at all of the addresses listed.
241 * Returns NULL if no devices found.
242 * See Documentation/i2c/instantiating-devices for more information.
243 */
Benson Leung9ad36922013-10-20 20:58:25 -0700244static struct i2c_client *add_probed_i2c_device(
Benson Leungbcaf0892013-02-21 12:14:58 -0800245 const char *name,
246 enum i2c_adapter_type type,
247 struct i2c_board_info *info,
248 const unsigned short *addrs)
249{
250 return __add_probed_i2c_device(name,
251 find_i2c_adapter_num(type),
252 info,
253 addrs);
254}
255
256/*
Benson Leungd1381f42012-10-25 14:21:21 -0700257 * Probes for a device at a single address, the one provided by
258 * info->addr.
259 * Returns NULL if no device found.
260 */
Benson Leung9ad36922013-10-20 20:58:25 -0700261static struct i2c_client *add_i2c_device(const char *name,
Benson Leunge7b28842013-02-21 12:14:56 -0800262 enum i2c_adapter_type type,
263 struct i2c_board_info *info)
Benson Leungd1381f42012-10-25 14:21:21 -0700264{
Benson Leungd1381f42012-10-25 14:21:21 -0700265 return __add_probed_i2c_device(name,
Benson Leunge7b28842013-02-21 12:14:56 -0800266 find_i2c_adapter_num(type),
Benson Leungd1381f42012-10-25 14:21:21 -0700267 info,
Dmitry Torokhov96cba9b2015-04-14 13:50:09 -0700268 NULL);
Benson Leungd1381f42012-10-25 14:21:21 -0700269}
270
Benson Leung9ad36922013-10-20 20:58:25 -0700271static int setup_cyapa_tp(enum i2c_adapter_type type)
Benson Leunge7b28842013-02-21 12:14:56 -0800272{
Benson Leung9ad36922013-10-20 20:58:25 -0700273 if (tp)
274 return 0;
275
Aaron Durbinec199dd2013-10-20 20:58:24 -0700276 /* add cyapa touchpad */
277 tp = add_i2c_device("trackpad", type, &cyapa_device);
Benson Leung9ad36922013-10-20 20:58:25 -0700278 return (!tp) ? -EAGAIN : 0;
Benson Leungd1381f42012-10-25 14:21:21 -0700279}
280
Benson Leung9ad36922013-10-20 20:58:25 -0700281static int setup_atmel_224s_tp(enum i2c_adapter_type type)
Benson Leung8e1ad4c2013-02-21 12:14:59 -0800282{
283 const unsigned short addr_list[] = { ATMEL_TP_I2C_BL_ADDR,
Benson Leung8e1ad4c2013-02-21 12:14:59 -0800284 I2C_CLIENT_END };
Benson Leung9ad36922013-10-20 20:58:25 -0700285 if (tp)
286 return 0;
Benson Leung8e1ad4c2013-02-21 12:14:59 -0800287
Aaron Durbinec199dd2013-10-20 20:58:24 -0700288 /* add atmel mxt touchpad */
289 tp = add_probed_i2c_device("trackpad", type,
Benson Leung8e1ad4c2013-02-21 12:14:59 -0800290 &atmel_224s_tp_device, addr_list);
Benson Leung9ad36922013-10-20 20:58:25 -0700291 return (!tp) ? -EAGAIN : 0;
Benson Leung8e1ad4c2013-02-21 12:14:59 -0800292}
293
Benson Leung9bd9a902016-05-02 08:57:16 +0800294static int setup_elantech_tp(enum i2c_adapter_type type)
295{
296 if (tp)
297 return 0;
298
299 /* add elantech touchpad */
300 tp = add_i2c_device("trackpad", type, &elantech_device);
301 return (!tp) ? -EAGAIN : 0;
302}
303
Benson Leung9ad36922013-10-20 20:58:25 -0700304static int setup_atmel_1664s_ts(enum i2c_adapter_type type)
Yufeng Shen33a84f82013-02-21 12:15:00 -0800305{
306 const unsigned short addr_list[] = { ATMEL_TS_I2C_BL_ADDR,
Yufeng Shen33a84f82013-02-21 12:15:00 -0800307 I2C_CLIENT_END };
Benson Leung9ad36922013-10-20 20:58:25 -0700308 if (ts)
309 return 0;
Yufeng Shen33a84f82013-02-21 12:15:00 -0800310
Aaron Durbinec199dd2013-10-20 20:58:24 -0700311 /* add atmel mxt touch device */
312 ts = add_probed_i2c_device("touchscreen", type,
Yufeng Shen33a84f82013-02-21 12:15:00 -0800313 &atmel_1664s_device, addr_list);
Benson Leung9ad36922013-10-20 20:58:25 -0700314 return (!ts) ? -EAGAIN : 0;
Yufeng Shen33a84f82013-02-21 12:15:00 -0800315}
316
Benson Leung9ad36922013-10-20 20:58:25 -0700317static int setup_isl29018_als(enum i2c_adapter_type type)
Benson Leungd1381f42012-10-25 14:21:21 -0700318{
Benson Leung9ad36922013-10-20 20:58:25 -0700319 if (als)
320 return 0;
321
Benson Leungd1381f42012-10-25 14:21:21 -0700322 /* add isl29018 light sensor */
Aaron Durbinec199dd2013-10-20 20:58:24 -0700323 als = add_i2c_device("lightsensor", type, &isl_als_device);
Benson Leung9ad36922013-10-20 20:58:25 -0700324 return (!als) ? -EAGAIN : 0;
Benson Leungd1381f42012-10-25 14:21:21 -0700325}
326
Benson Leung9ad36922013-10-20 20:58:25 -0700327static int setup_tsl2583_als(enum i2c_adapter_type type)
Benson Leungcc5c3982013-02-21 12:14:57 -0800328{
Benson Leung9ad36922013-10-20 20:58:25 -0700329 if (als)
330 return 0;
331
Aaron Durbinec199dd2013-10-20 20:58:24 -0700332 /* add tsl2583 light sensor */
333 als = add_i2c_device(NULL, type, &tsl2583_als_device);
Benson Leung9ad36922013-10-20 20:58:25 -0700334 return (!als) ? -EAGAIN : 0;
Benson Leungcc5c3982013-02-21 12:14:57 -0800335}
336
Benson Leung9ad36922013-10-20 20:58:25 -0700337static int setup_tsl2563_als(enum i2c_adapter_type type)
Benson Leung8016bcb2013-02-01 14:34:46 -0800338{
Benson Leung9ad36922013-10-20 20:58:25 -0700339 if (als)
340 return 0;
341
Aaron Durbinec199dd2013-10-20 20:58:24 -0700342 /* add tsl2563 light sensor */
343 als = add_i2c_device(NULL, type, &tsl2563_als_device);
Benson Leung9ad36922013-10-20 20:58:25 -0700344 return (!als) ? -EAGAIN : 0;
Benson Leung8016bcb2013-02-01 14:34:46 -0800345}
346
Benson Leung9ad36922013-10-20 20:58:25 -0700347static int __init chromeos_laptop_dmi_matched(const struct dmi_system_id *id)
348{
349 cros_laptop = (void *)id->driver_data;
Dmitry Torokhov4f27f672018-03-20 15:31:30 -0700350 pr_debug("DMI Matched %s\n", id->ident);
Benson Leung9ad36922013-10-20 20:58:25 -0700351
352 /* Indicate to dmi_scan that processing is done. */
353 return 1;
354}
355
356static int chromeos_laptop_probe(struct platform_device *pdev)
Benson Leungaabf3f42013-02-01 14:34:45 -0800357{
Aaron Durbinec199dd2013-10-20 20:58:24 -0700358 int i;
Benson Leung9ad36922013-10-20 20:58:25 -0700359 int ret = 0;
Aaron Durbinec199dd2013-10-20 20:58:24 -0700360
361 for (i = 0; i < MAX_I2C_PERIPHERALS; i++) {
362 struct i2c_peripheral *i2c_dev;
363
364 i2c_dev = &cros_laptop->i2c_peripherals[i];
365
366 /* No more peripherals. */
367 if (i2c_dev->add == NULL)
368 break;
369
Benson Leung55024862014-07-15 17:43:11 -0700370 if (i2c_dev->state == TIMEDOUT || i2c_dev->state == PROBED)
371 continue;
372
373 /*
374 * Check that the i2c adapter is present.
375 * -EPROBE_DEFER if missing as the adapter may appear much
376 * later.
377 */
378 if (find_i2c_adapter_num(i2c_dev->type) == -ENODEV) {
Benson Leung9ad36922013-10-20 20:58:25 -0700379 ret = -EPROBE_DEFER;
Benson Leung55024862014-07-15 17:43:11 -0700380 continue;
381 }
382
383 /* Add the device. */
384 if (i2c_dev->add(i2c_dev->type) == -EAGAIN) {
385 /*
386 * Set -EPROBE_DEFER a limited num of times
387 * if device is not successfully added.
388 */
389 if (++i2c_dev->tries < MAX_I2C_DEVICE_DEFERRALS) {
390 ret = -EPROBE_DEFER;
391 } else {
392 /* Ran out of tries. */
Dmitry Torokhov4f27f672018-03-20 15:31:30 -0700393 pr_notice("ran out of tries for device.\n");
Benson Leung55024862014-07-15 17:43:11 -0700394 i2c_dev->state = TIMEDOUT;
395 }
396 } else {
397 i2c_dev->state = PROBED;
398 }
Aaron Durbinec199dd2013-10-20 20:58:24 -0700399 }
400
Benson Leung9ad36922013-10-20 20:58:25 -0700401 return ret;
Benson Leungaabf3f42013-02-01 14:34:45 -0800402}
403
Dmitry Torokhovfc88bbd2018-03-06 10:59:15 -0800404static struct chromeos_laptop samsung_series_5_550 = {
Aaron Durbinec199dd2013-10-20 20:58:24 -0700405 .i2c_peripherals = {
406 /* Touchpad. */
407 { .add = setup_cyapa_tp, I2C_ADAPTER_SMBUS },
408 /* Light Sensor. */
409 { .add = setup_isl29018_als, I2C_ADAPTER_SMBUS },
410 },
411};
412
Dmitry Torokhovfc88bbd2018-03-06 10:59:15 -0800413static struct chromeos_laptop samsung_series_5 = {
Aaron Durbinec199dd2013-10-20 20:58:24 -0700414 .i2c_peripherals = {
415 /* Light Sensor. */
416 { .add = setup_tsl2583_als, I2C_ADAPTER_SMBUS },
417 },
418};
419
Dmitry Torokhovfc88bbd2018-03-06 10:59:15 -0800420static struct chromeos_laptop chromebook_pixel = {
Aaron Durbinec199dd2013-10-20 20:58:24 -0700421 .i2c_peripherals = {
422 /* Touch Screen. */
423 { .add = setup_atmel_1664s_ts, I2C_ADAPTER_PANEL },
424 /* Touchpad. */
425 { .add = setup_atmel_224s_tp, I2C_ADAPTER_VGADDC },
426 /* Light Sensor. */
427 { .add = setup_isl29018_als, I2C_ADAPTER_PANEL },
428 },
429};
430
Dmitry Torokhovfc88bbd2018-03-06 10:59:15 -0800431static struct chromeos_laptop hp_chromebook_14 = {
Benson Leung5ea95672014-06-17 14:02:01 -0700432 .i2c_peripherals = {
433 /* Touchpad. */
434 { .add = setup_cyapa_tp, I2C_ADAPTER_DESIGNWARE_0 },
435 },
436};
437
Dmitry Torokhovfc88bbd2018-03-06 10:59:15 -0800438static struct chromeos_laptop dell_chromebook_11 = {
Mohammed Habibulla0e1e5e52014-06-17 14:02:02 -0700439 .i2c_peripherals = {
440 /* Touchpad. */
441 { .add = setup_cyapa_tp, I2C_ADAPTER_DESIGNWARE_0 },
Charlie Mooney9e96aa72016-05-02 08:57:17 +0800442 /* Elan Touchpad option. */
443 { .add = setup_elantech_tp, I2C_ADAPTER_DESIGNWARE_0 },
Mohammed Habibulla0e1e5e52014-06-17 14:02:02 -0700444 },
445};
446
Dmitry Torokhovfc88bbd2018-03-06 10:59:15 -0800447static struct chromeos_laptop toshiba_cb35 = {
Gene Chen963cb6f2014-06-17 14:02:03 -0700448 .i2c_peripherals = {
449 /* Touchpad. */
450 { .add = setup_cyapa_tp, I2C_ADAPTER_DESIGNWARE_0 },
451 },
452};
453
Dmitry Torokhovfc88bbd2018-03-06 10:59:15 -0800454static struct chromeos_laptop acer_c7_chromebook = {
Aaron Durbinec199dd2013-10-20 20:58:24 -0700455 .i2c_peripherals = {
456 /* Touchpad. */
457 { .add = setup_cyapa_tp, I2C_ADAPTER_SMBUS },
458 },
459};
460
Dmitry Torokhovfc88bbd2018-03-06 10:59:15 -0800461static struct chromeos_laptop acer_ac700 = {
Aaron Durbinec199dd2013-10-20 20:58:24 -0700462 .i2c_peripherals = {
463 /* Light Sensor. */
464 { .add = setup_tsl2563_als, I2C_ADAPTER_SMBUS },
465 },
466};
467
Dmitry Torokhovfc88bbd2018-03-06 10:59:15 -0800468static struct chromeos_laptop acer_c720 = {
Mika Westerbergda3b0ab2014-06-17 14:02:00 -0700469 .i2c_peripherals = {
Michael Mullinb90b3c42014-07-15 20:00:54 -0400470 /* Touchscreen. */
471 { .add = setup_atmel_1664s_ts, I2C_ADAPTER_DESIGNWARE_1 },
Mika Westerbergda3b0ab2014-06-17 14:02:00 -0700472 /* Touchpad. */
473 { .add = setup_cyapa_tp, I2C_ADAPTER_DESIGNWARE_0 },
Benson Leung9bd9a902016-05-02 08:57:16 +0800474 /* Elan Touchpad option. */
475 { .add = setup_elantech_tp, I2C_ADAPTER_DESIGNWARE_0 },
Mika Westerbergda3b0ab2014-06-17 14:02:00 -0700476 /* Light Sensor. */
477 { .add = setup_isl29018_als, I2C_ADAPTER_DESIGNWARE_1 },
478 },
479};
480
Dmitry Torokhovfc88bbd2018-03-06 10:59:15 -0800481static struct chromeos_laptop hp_pavilion_14_chromebook = {
Aaron Durbinec199dd2013-10-20 20:58:24 -0700482 .i2c_peripherals = {
483 /* Touchpad. */
484 { .add = setup_cyapa_tp, I2C_ADAPTER_SMBUS },
485 },
486};
487
Dmitry Torokhovfc88bbd2018-03-06 10:59:15 -0800488static struct chromeos_laptop cr48 = {
Aaron Durbinec199dd2013-10-20 20:58:24 -0700489 .i2c_peripherals = {
490 /* Light Sensor. */
491 { .add = setup_tsl2563_als, I2C_ADAPTER_SMBUS },
492 },
493};
494
495#define _CBDD(board_) \
Benson Leung9ad36922013-10-20 20:58:25 -0700496 .callback = chromeos_laptop_dmi_matched, \
Aaron Durbinec199dd2013-10-20 20:58:24 -0700497 .driver_data = (void *)&board_
498
Christoph Hellwig6faadbb2017-09-14 11:59:30 +0200499static const struct dmi_system_id chromeos_laptop_dmi_table[] __initconst = {
Benson Leungd1381f42012-10-25 14:21:21 -0700500 {
Aaron Durbinec199dd2013-10-20 20:58:24 -0700501 .ident = "Samsung Series 5 550",
Benson Leungd1381f42012-10-25 14:21:21 -0700502 .matches = {
503 DMI_MATCH(DMI_SYS_VENDOR, "SAMSUNG"),
504 DMI_MATCH(DMI_PRODUCT_NAME, "Lumpy"),
505 },
Aaron Durbinec199dd2013-10-20 20:58:24 -0700506 _CBDD(samsung_series_5_550),
Benson Leungd1381f42012-10-25 14:21:21 -0700507 },
508 {
Aaron Durbinec199dd2013-10-20 20:58:24 -0700509 .ident = "Samsung Series 5",
Benson Leung8016bcb2013-02-01 14:34:46 -0800510 .matches = {
511 DMI_MATCH(DMI_PRODUCT_NAME, "Alex"),
512 },
Aaron Durbinec199dd2013-10-20 20:58:24 -0700513 _CBDD(samsung_series_5),
Benson Leung8016bcb2013-02-01 14:34:46 -0800514 },
515 {
Aaron Durbinec199dd2013-10-20 20:58:24 -0700516 .ident = "Chromebook Pixel",
Benson Leungaabf3f42013-02-01 14:34:45 -0800517 .matches = {
Aaron Durbinec199dd2013-10-20 20:58:24 -0700518 DMI_MATCH(DMI_SYS_VENDOR, "GOOGLE"),
519 DMI_MATCH(DMI_PRODUCT_NAME, "Link"),
Benson Leungaabf3f42013-02-01 14:34:45 -0800520 },
Aaron Durbinec199dd2013-10-20 20:58:24 -0700521 _CBDD(chromebook_pixel),
Benson Leungaabf3f42013-02-01 14:34:45 -0800522 },
523 {
Mohammed Habibulla0e1e5e52014-06-17 14:02:02 -0700524 .ident = "Wolf",
525 .matches = {
526 DMI_MATCH(DMI_BIOS_VENDOR, "coreboot"),
527 DMI_MATCH(DMI_PRODUCT_NAME, "Wolf"),
528 },
529 _CBDD(dell_chromebook_11),
530 },
531 {
Benson Leung5ea95672014-06-17 14:02:01 -0700532 .ident = "HP Chromebook 14",
533 .matches = {
534 DMI_MATCH(DMI_BIOS_VENDOR, "coreboot"),
535 DMI_MATCH(DMI_PRODUCT_NAME, "Falco"),
536 },
537 _CBDD(hp_chromebook_14),
538 },
539 {
Gene Chen963cb6f2014-06-17 14:02:03 -0700540 .ident = "Toshiba CB35",
541 .matches = {
542 DMI_MATCH(DMI_BIOS_VENDOR, "coreboot"),
543 DMI_MATCH(DMI_PRODUCT_NAME, "Leon"),
544 },
545 _CBDD(toshiba_cb35),
546 },
547 {
Aaron Durbinec199dd2013-10-20 20:58:24 -0700548 .ident = "Acer C7 Chromebook",
549 .matches = {
550 DMI_MATCH(DMI_PRODUCT_NAME, "Parrot"),
551 },
552 _CBDD(acer_c7_chromebook),
553 },
554 {
555 .ident = "Acer AC700",
Benson Leungaabf3f42013-02-01 14:34:45 -0800556 .matches = {
557 DMI_MATCH(DMI_PRODUCT_NAME, "ZGB"),
558 },
Aaron Durbinec199dd2013-10-20 20:58:24 -0700559 _CBDD(acer_ac700),
560 },
561 {
Mika Westerbergda3b0ab2014-06-17 14:02:00 -0700562 .ident = "Acer C720",
563 .matches = {
564 DMI_MATCH(DMI_PRODUCT_NAME, "Peppy"),
565 },
566 _CBDD(acer_c720),
567 },
568 {
Aaron Durbinec199dd2013-10-20 20:58:24 -0700569 .ident = "HP Pavilion 14 Chromebook",
570 .matches = {
571 DMI_MATCH(DMI_PRODUCT_NAME, "Butterfly"),
572 },
573 _CBDD(hp_pavilion_14_chromebook),
574 },
575 {
576 .ident = "Cr-48",
577 .matches = {
578 DMI_MATCH(DMI_PRODUCT_NAME, "Mario"),
579 },
580 _CBDD(cr48),
Benson Leungaabf3f42013-02-01 14:34:45 -0800581 },
Benson Leungd1381f42012-10-25 14:21:21 -0700582 { }
583};
584MODULE_DEVICE_TABLE(dmi, chromeos_laptop_dmi_table);
585
Benson Leung9ad36922013-10-20 20:58:25 -0700586static struct platform_device *cros_platform_device;
587
588static struct platform_driver cros_platform_driver = {
589 .driver = {
590 .name = "chromeos_laptop",
Benson Leung9ad36922013-10-20 20:58:25 -0700591 },
592 .probe = chromeos_laptop_probe,
593};
594
Benson Leungd1381f42012-10-25 14:21:21 -0700595static int __init chromeos_laptop_init(void)
596{
Benson Leung9ad36922013-10-20 20:58:25 -0700597 int ret;
Robin Schroer49c68a22014-05-29 20:45:07 +0200598
Benson Leungd1381f42012-10-25 14:21:21 -0700599 if (!dmi_check_system(chromeos_laptop_dmi_table)) {
Dmitry Torokhov4f27f672018-03-20 15:31:30 -0700600 pr_debug("unsupported system\n");
Benson Leungd1381f42012-10-25 14:21:21 -0700601 return -ENODEV;
602 }
Benson Leung9ad36922013-10-20 20:58:25 -0700603
604 ret = platform_driver_register(&cros_platform_driver);
605 if (ret)
606 return ret;
607
608 cros_platform_device = platform_device_alloc("chromeos_laptop", -1);
609 if (!cros_platform_device) {
610 ret = -ENOMEM;
611 goto fail_platform_device1;
612 }
613
614 ret = platform_device_add(cros_platform_device);
615 if (ret)
616 goto fail_platform_device2;
617
Benson Leungd1381f42012-10-25 14:21:21 -0700618 return 0;
Benson Leung9ad36922013-10-20 20:58:25 -0700619
620fail_platform_device2:
621 platform_device_put(cros_platform_device);
622fail_platform_device1:
623 platform_driver_unregister(&cros_platform_driver);
624 return ret;
Benson Leungd1381f42012-10-25 14:21:21 -0700625}
626
627static void __exit chromeos_laptop_exit(void)
628{
629 if (als)
630 i2c_unregister_device(als);
631 if (tp)
632 i2c_unregister_device(tp);
Yufeng Shen33a84f82013-02-21 12:15:00 -0800633 if (ts)
634 i2c_unregister_device(ts);
Wei Yongjun2b8454a2013-11-27 11:34:58 +0800635
636 platform_device_unregister(cros_platform_device);
637 platform_driver_unregister(&cros_platform_driver);
Benson Leungd1381f42012-10-25 14:21:21 -0700638}
639
640module_init(chromeos_laptop_init);
641module_exit(chromeos_laptop_exit);
642
643MODULE_DESCRIPTION("Chrome OS Laptop driver");
644MODULE_AUTHOR("Benson Leung <bleung@chromium.org>");
645MODULE_LICENSE("GPL");