Commit c0bb0608 authored by Dmitry Torokhov's avatar Dmitry Torokhov Committed by Benson Leung

platform/chrome: chromeos_laptop - discard data for unneeded boards

Mark board data as __intconst/__initdata and make a copy of appropriate
entry once we identified the board we are running on. The rest of the data
will be discarded once the kernel finished booting (or module finished
loading).
Signed-off-by: default avatarDmitry Torokhov <dmitry.torokhov@gmail.com>
Signed-off-by: default avatarBenson Leung <bleung@chromium.org>
parent f00c1d19
...@@ -54,13 +54,16 @@ struct i2c_peripheral { ...@@ -54,13 +54,16 @@ struct i2c_peripheral {
struct i2c_client *client; struct i2c_client *client;
}; };
#define MAX_I2C_PERIPHERALS 4
struct chromeos_laptop { struct chromeos_laptop {
struct i2c_peripheral i2c_peripherals[MAX_I2C_PERIPHERALS]; /*
* Note that we can't mark this pointer as const because
* i2c_new_probed_device() changes passed in I2C board info, so.
*/
struct i2c_peripheral *i2c_peripherals;
unsigned int num_i2c_peripherals;
}; };
static struct chromeos_laptop *cros_laptop; static const struct chromeos_laptop *cros_laptop;
static struct i2c_client * static struct i2c_client *
chromes_laptop_instantiate_i2c_device(struct i2c_adapter *adapter, chromes_laptop_instantiate_i2c_device(struct i2c_adapter *adapter,
...@@ -121,13 +124,9 @@ static void chromeos_laptop_check_adapter(struct i2c_adapter *adapter) ...@@ -121,13 +124,9 @@ static void chromeos_laptop_check_adapter(struct i2c_adapter *adapter)
struct i2c_peripheral *i2c_dev; struct i2c_peripheral *i2c_dev;
int i; int i;
for (i = 0; i < MAX_I2C_PERIPHERALS; i++) { for (i = 0; i < cros_laptop->num_i2c_peripherals; i++) {
i2c_dev = &cros_laptop->i2c_peripherals[i]; i2c_dev = &cros_laptop->i2c_peripherals[i];
/* No more peripherals */
if (!i2c_dev->board_info.addr)
break;
/* Skip devices already created */ /* Skip devices already created */
if (i2c_dev->client) if (i2c_dev->client)
continue; continue;
...@@ -154,7 +153,7 @@ static void chromeos_laptop_detach_i2c_client(struct i2c_client *client) ...@@ -154,7 +153,7 @@ static void chromeos_laptop_detach_i2c_client(struct i2c_client *client)
struct i2c_peripheral *i2c_dev; struct i2c_peripheral *i2c_dev;
int i; int i;
for (i = 0; i < MAX_I2C_PERIPHERALS; i++) { for (i = 0; i < cros_laptop->num_i2c_peripherals; i++) {
i2c_dev = &cros_laptop->i2c_peripherals[i]; i2c_dev = &cros_laptop->i2c_peripherals[i];
if (i2c_dev->client == client) if (i2c_dev->client == client)
...@@ -186,8 +185,13 @@ static struct notifier_block chromeos_laptop_i2c_notifier = { ...@@ -186,8 +185,13 @@ static struct notifier_block chromeos_laptop_i2c_notifier = {
.notifier_call = chromeos_laptop_i2c_notifier_call, .notifier_call = chromeos_laptop_i2c_notifier_call,
}; };
static struct chromeos_laptop samsung_series_5_550 = { #define DECLARE_CROS_LAPTOP(_name) \
.i2c_peripherals = { static const struct chromeos_laptop _name __initconst = { \
.i2c_peripherals = _name##_peripherals, \
.num_i2c_peripherals = ARRAY_SIZE(_name##_peripherals), \
}
static struct i2c_peripheral samsung_series_5_550_peripherals[] __initdata = {
/* Touchpad. */ /* Touchpad. */
{ {
.board_info = { .board_info = {
...@@ -205,11 +209,10 @@ static struct chromeos_laptop samsung_series_5_550 = { ...@@ -205,11 +209,10 @@ static struct chromeos_laptop samsung_series_5_550 = {
.dmi_name = "lightsensor", .dmi_name = "lightsensor",
.type = I2C_ADAPTER_SMBUS, .type = I2C_ADAPTER_SMBUS,
}, },
},
}; };
DECLARE_CROS_LAPTOP(samsung_series_5_550);
static struct chromeos_laptop samsung_series_5 = { static struct i2c_peripheral samsung_series_5_peripherals[] __initdata = {
.i2c_peripherals = {
/* Light Sensor. */ /* Light Sensor. */
{ {
.board_info = { .board_info = {
...@@ -217,10 +220,10 @@ static struct chromeos_laptop samsung_series_5 = { ...@@ -217,10 +220,10 @@ static struct chromeos_laptop samsung_series_5 = {
}, },
.type = I2C_ADAPTER_SMBUS, .type = I2C_ADAPTER_SMBUS,
}, },
},
}; };
DECLARE_CROS_LAPTOP(samsung_series_5);
static int chromebook_pixel_tp_keys[] = { static const int chromebook_pixel_tp_keys[] __initconst = {
KEY_RESERVED, KEY_RESERVED,
KEY_RESERVED, KEY_RESERVED,
KEY_RESERVED, KEY_RESERVED,
...@@ -229,13 +232,13 @@ static int chromebook_pixel_tp_keys[] = { ...@@ -229,13 +232,13 @@ static int chromebook_pixel_tp_keys[] = {
BTN_LEFT BTN_LEFT
}; };
static const struct property_entry chromebook_pixel_trackpad_props[] = { static const struct property_entry
chromebook_pixel_trackpad_props[] __initconst = {
PROPERTY_ENTRY_U32_ARRAY("linux,gpio-keymap", chromebook_pixel_tp_keys), PROPERTY_ENTRY_U32_ARRAY("linux,gpio-keymap", chromebook_pixel_tp_keys),
{ } { }
}; };
static struct chromeos_laptop chromebook_pixel = { static struct i2c_peripheral chromebook_pixel_peripherals[] __initdata = {
.i2c_peripherals = {
/* Touch Screen. */ /* Touch Screen. */
{ {
.board_info = { .board_info = {
...@@ -270,11 +273,10 @@ static struct chromeos_laptop chromebook_pixel = { ...@@ -270,11 +273,10 @@ static struct chromeos_laptop chromebook_pixel = {
.dmi_name = "lightsensor", .dmi_name = "lightsensor",
.type = I2C_ADAPTER_PANEL, .type = I2C_ADAPTER_PANEL,
}, },
},
}; };
DECLARE_CROS_LAPTOP(chromebook_pixel);
static struct chromeos_laptop hp_chromebook_14 = { static struct i2c_peripheral hp_chromebook_14_peripherals[] __initdata = {
.i2c_peripherals = {
/* Touchpad. */ /* Touchpad. */
{ {
.board_info = { .board_info = {
...@@ -284,11 +286,10 @@ static struct chromeos_laptop hp_chromebook_14 = { ...@@ -284,11 +286,10 @@ static struct chromeos_laptop hp_chromebook_14 = {
.dmi_name = "trackpad", .dmi_name = "trackpad",
.type = I2C_ADAPTER_DESIGNWARE, .type = I2C_ADAPTER_DESIGNWARE,
}, },
},
}; };
DECLARE_CROS_LAPTOP(hp_chromebook_14);
static struct chromeos_laptop dell_chromebook_11 = { static struct i2c_peripheral dell_chromebook_11_peripherals[] __initdata = {
.i2c_peripherals = {
/* Touchpad. */ /* Touchpad. */
{ {
.board_info = { .board_info = {
...@@ -307,11 +308,10 @@ static struct chromeos_laptop dell_chromebook_11 = { ...@@ -307,11 +308,10 @@ static struct chromeos_laptop dell_chromebook_11 = {
.dmi_name = "trackpad", .dmi_name = "trackpad",
.type = I2C_ADAPTER_DESIGNWARE, .type = I2C_ADAPTER_DESIGNWARE,
}, },
},
}; };
DECLARE_CROS_LAPTOP(dell_chromebook_11);
static struct chromeos_laptop toshiba_cb35 = { static struct i2c_peripheral toshiba_cb35_peripherals[] __initdata = {
.i2c_peripherals = {
/* Touchpad. */ /* Touchpad. */
{ {
.board_info = { .board_info = {
...@@ -321,11 +321,10 @@ static struct chromeos_laptop toshiba_cb35 = { ...@@ -321,11 +321,10 @@ static struct chromeos_laptop toshiba_cb35 = {
.dmi_name = "trackpad", .dmi_name = "trackpad",
.type = I2C_ADAPTER_DESIGNWARE, .type = I2C_ADAPTER_DESIGNWARE,
}, },
},
}; };
DECLARE_CROS_LAPTOP(toshiba_cb35);
static struct chromeos_laptop acer_c7_chromebook = { static struct i2c_peripheral acer_c7_chromebook_peripherals[] __initdata = {
.i2c_peripherals = {
/* Touchpad. */ /* Touchpad. */
{ {
.board_info = { .board_info = {
...@@ -335,11 +334,10 @@ static struct chromeos_laptop acer_c7_chromebook = { ...@@ -335,11 +334,10 @@ static struct chromeos_laptop acer_c7_chromebook = {
.dmi_name = "trackpad", .dmi_name = "trackpad",
.type = I2C_ADAPTER_SMBUS, .type = I2C_ADAPTER_SMBUS,
}, },
},
}; };
DECLARE_CROS_LAPTOP(acer_c7_chromebook);
static struct chromeos_laptop acer_ac700 = { static struct i2c_peripheral acer_ac700_peripherals[] __initdata = {
.i2c_peripherals = {
/* Light Sensor. */ /* Light Sensor. */
{ {
.board_info = { .board_info = {
...@@ -347,11 +345,10 @@ static struct chromeos_laptop acer_ac700 = { ...@@ -347,11 +345,10 @@ static struct chromeos_laptop acer_ac700 = {
}, },
.type = I2C_ADAPTER_SMBUS, .type = I2C_ADAPTER_SMBUS,
}, },
},
}; };
DECLARE_CROS_LAPTOP(acer_ac700);
static struct chromeos_laptop acer_c720 = { static struct i2c_peripheral acer_c720_peripherals[] __initdata = {
.i2c_peripherals = {
/* Touchscreen. */ /* Touchscreen. */
{ {
.board_info = { .board_info = {
...@@ -394,11 +391,11 @@ static struct chromeos_laptop acer_c720 = { ...@@ -394,11 +391,11 @@ static struct chromeos_laptop acer_c720 = {
.type = I2C_ADAPTER_DESIGNWARE, .type = I2C_ADAPTER_DESIGNWARE,
.pci_devid = PCI_DEVID(0, PCI_DEVFN(0x15, 0x2)), .pci_devid = PCI_DEVID(0, PCI_DEVFN(0x15, 0x2)),
}, },
},
}; };
DECLARE_CROS_LAPTOP(acer_c720);
static struct chromeos_laptop hp_pavilion_14_chromebook = { static struct i2c_peripheral
.i2c_peripherals = { hp_pavilion_14_chromebook_peripherals[] __initdata = {
/* Touchpad. */ /* Touchpad. */
{ {
.board_info = { .board_info = {
...@@ -408,11 +405,10 @@ static struct chromeos_laptop hp_pavilion_14_chromebook = { ...@@ -408,11 +405,10 @@ static struct chromeos_laptop hp_pavilion_14_chromebook = {
.dmi_name = "trackpad", .dmi_name = "trackpad",
.type = I2C_ADAPTER_SMBUS, .type = I2C_ADAPTER_SMBUS,
}, },
},
}; };
DECLARE_CROS_LAPTOP(hp_pavilion_14_chromebook);
static struct chromeos_laptop cr48 = { static struct i2c_peripheral cr48_peripherals[] __initdata = {
.i2c_peripherals = {
/* Light Sensor. */ /* Light Sensor. */
{ {
.board_info = { .board_info = {
...@@ -420,8 +416,8 @@ static struct chromeos_laptop cr48 = { ...@@ -420,8 +416,8 @@ static struct chromeos_laptop cr48 = {
}, },
.type = I2C_ADAPTER_SMBUS, .type = I2C_ADAPTER_SMBUS,
}, },
},
}; };
DECLARE_CROS_LAPTOP(cr48);
static const struct dmi_system_id chromeos_laptop_dmi_table[] __initconst = { static const struct dmi_system_id chromeos_laptop_dmi_table[] __initconst = {
{ {
...@@ -541,24 +537,14 @@ static int __init chromeos_laptop_get_irq_from_dmi(const char *dmi_name) ...@@ -541,24 +537,14 @@ static int __init chromeos_laptop_get_irq_from_dmi(const char *dmi_name)
return dev_data->instance; return dev_data->instance;
} }
static struct chromeos_laptop * __init static int __init chromeos_laptop_setup_irq(struct i2c_peripheral *i2c_dev)
chromeos_laptop_prepare(const struct dmi_system_id *id)
{ {
struct i2c_peripheral *i2c_dev;
int irq; int irq;
int i;
cros_laptop = (void *)id->driver_data;
for (i = 0; i < MAX_I2C_PERIPHERALS; i++) {
i2c_dev = &cros_laptop->i2c_peripherals[i];
if (!i2c_dev->dmi_name)
continue;
if (i2c_dev->dmi_name) {
irq = chromeos_laptop_get_irq_from_dmi(i2c_dev->dmi_name); irq = chromeos_laptop_get_irq_from_dmi(i2c_dev->dmi_name);
if (irq < 0) if (irq < 0)
return ERR_PTR(irq); return irq;
i2c_dev->irq_resource = (struct resource) i2c_dev->irq_resource = (struct resource)
DEFINE_RES_NAMED(irq, 1, NULL, DEFINE_RES_NAMED(irq, 1, NULL,
...@@ -567,9 +553,87 @@ chromeos_laptop_prepare(const struct dmi_system_id *id) ...@@ -567,9 +553,87 @@ chromeos_laptop_prepare(const struct dmi_system_id *id)
i2c_dev->board_info.num_resources = 1; i2c_dev->board_info.num_resources = 1;
} }
return 0;
}
static struct chromeos_laptop * __init
chromeos_laptop_prepare(const struct chromeos_laptop *src)
{
struct chromeos_laptop *cros_laptop;
struct i2c_peripheral *i2c_dev;
struct i2c_board_info *info;
int error;
int i;
cros_laptop = kzalloc(sizeof(*cros_laptop), GFP_KERNEL);
if (!cros_laptop)
return ERR_PTR(-ENOMEM);
cros_laptop->i2c_peripherals = kmemdup(src->i2c_peripherals,
src->num_i2c_peripherals *
sizeof(*src->i2c_peripherals),
GFP_KERNEL);
if (!cros_laptop->i2c_peripherals) {
error = -ENOMEM;
goto err_free_cros_laptop;
}
cros_laptop->num_i2c_peripherals = src->num_i2c_peripherals;
for (i = 0; i < cros_laptop->num_i2c_peripherals; i++) {
i2c_dev = &cros_laptop->i2c_peripherals[i];
info = &i2c_dev->board_info;
error = chromeos_laptop_setup_irq(i2c_dev);
if (error)
goto err_destroy_cros_peripherals;
/* We need to deep-copy properties */
if (info->properties) {
info->properties =
property_entries_dup(info->properties);
if (IS_ERR(info->properties)) {
error = PTR_ERR(info->properties);
goto err_destroy_cros_peripherals;
}
}
}
return cros_laptop; return cros_laptop;
err_destroy_cros_peripherals:
while (--i >= 0) {
i2c_dev = &cros_laptop->i2c_peripherals[i];
info = &i2c_dev->board_info;
if (info->properties)
property_entries_free(info->properties);
}
kfree(cros_laptop->i2c_peripherals);
err_free_cros_laptop:
kfree(cros_laptop);
return ERR_PTR(error);
} }
static void chromeos_laptop_destroy(const struct chromeos_laptop *cros_laptop)
{
struct i2c_peripheral *i2c_dev;
struct i2c_board_info *info;
int i;
for (i = 0; i < cros_laptop->num_i2c_peripherals; i++) {
i2c_dev = &cros_laptop->i2c_peripherals[i];
info = &i2c_dev->board_info;
if (i2c_dev->client)
i2c_unregister_device(i2c_dev->client);
if (info->properties)
property_entries_free(info->properties);
}
kfree(cros_laptop->i2c_peripherals);
kfree(cros_laptop);
}
static int __init chromeos_laptop_init(void) static int __init chromeos_laptop_init(void)
{ {
...@@ -584,7 +648,7 @@ static int __init chromeos_laptop_init(void) ...@@ -584,7 +648,7 @@ static int __init chromeos_laptop_init(void)
pr_debug("DMI Matched %s\n", dmi_id->ident); pr_debug("DMI Matched %s\n", dmi_id->ident);
cros_laptop = chromeos_laptop_prepare(dmi_id->driver_data); cros_laptop = chromeos_laptop_prepare((void *)dmi_id->driver_data);
if (IS_ERR(cros_laptop)) if (IS_ERR(cros_laptop))
return PTR_ERR(cros_laptop); return PTR_ERR(cros_laptop);
...@@ -592,6 +656,7 @@ static int __init chromeos_laptop_init(void) ...@@ -592,6 +656,7 @@ static int __init chromeos_laptop_init(void)
&chromeos_laptop_i2c_notifier); &chromeos_laptop_i2c_notifier);
if (error) { if (error) {
pr_err("failed to register i2c bus notifier: %d\n", error); pr_err("failed to register i2c bus notifier: %d\n", error);
chromeos_laptop_destroy(cros_laptop);
return error; return error;
} }
...@@ -606,21 +671,8 @@ static int __init chromeos_laptop_init(void) ...@@ -606,21 +671,8 @@ static int __init chromeos_laptop_init(void)
static void __exit chromeos_laptop_exit(void) static void __exit chromeos_laptop_exit(void)
{ {
struct i2c_peripheral *i2c_dev;
int i;
bus_unregister_notifier(&i2c_bus_type, &chromeos_laptop_i2c_notifier); bus_unregister_notifier(&i2c_bus_type, &chromeos_laptop_i2c_notifier);
chromeos_laptop_destroy(cros_laptop);
for (i = 0; i < MAX_I2C_PERIPHERALS; i++) {
i2c_dev = &cros_laptop->i2c_peripherals[i];
/* No more peripherals */
if (!i2c_dev->board_info.type)
break;
if (i2c_dev->client)
i2c_unregister_device(i2c_dev->client);
}
} }
module_init(chromeos_laptop_init); module_init(chromeos_laptop_init);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment