162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * Export SMBIOS/DMI info via sysfs to userspace
462306a36Sopenharmony_ci *
562306a36Sopenharmony_ci * Copyright 2007, Lennart Poettering
662306a36Sopenharmony_ci */
762306a36Sopenharmony_ci
862306a36Sopenharmony_ci#include <linux/module.h>
962306a36Sopenharmony_ci#include <linux/kernel.h>
1062306a36Sopenharmony_ci#include <linux/init.h>
1162306a36Sopenharmony_ci#include <linux/dmi.h>
1262306a36Sopenharmony_ci#include <linux/device.h>
1362306a36Sopenharmony_ci#include <linux/slab.h>
1462306a36Sopenharmony_ci
1562306a36Sopenharmony_cistruct dmi_device_attribute{
1662306a36Sopenharmony_ci	struct device_attribute dev_attr;
1762306a36Sopenharmony_ci	int field;
1862306a36Sopenharmony_ci};
1962306a36Sopenharmony_ci#define to_dmi_dev_attr(_dev_attr) \
2062306a36Sopenharmony_ci	container_of(_dev_attr, struct dmi_device_attribute, dev_attr)
2162306a36Sopenharmony_ci
2262306a36Sopenharmony_cistatic ssize_t sys_dmi_field_show(struct device *dev,
2362306a36Sopenharmony_ci				  struct device_attribute *attr,
2462306a36Sopenharmony_ci				  char *page)
2562306a36Sopenharmony_ci{
2662306a36Sopenharmony_ci	int field = to_dmi_dev_attr(attr)->field;
2762306a36Sopenharmony_ci	ssize_t len;
2862306a36Sopenharmony_ci	len = scnprintf(page, PAGE_SIZE, "%s\n", dmi_get_system_info(field));
2962306a36Sopenharmony_ci	page[len-1] = '\n';
3062306a36Sopenharmony_ci	return len;
3162306a36Sopenharmony_ci}
3262306a36Sopenharmony_ci
3362306a36Sopenharmony_ci#define DMI_ATTR(_name, _mode, _show, _field)			\
3462306a36Sopenharmony_ci	{ .dev_attr = __ATTR(_name, _mode, _show, NULL),	\
3562306a36Sopenharmony_ci	  .field = _field }
3662306a36Sopenharmony_ci
3762306a36Sopenharmony_ci#define DEFINE_DMI_ATTR_WITH_SHOW(_name, _mode, _field)		\
3862306a36Sopenharmony_cistatic struct dmi_device_attribute sys_dmi_##_name##_attr =	\
3962306a36Sopenharmony_ci	DMI_ATTR(_name, _mode, sys_dmi_field_show, _field);
4062306a36Sopenharmony_ci
4162306a36Sopenharmony_ciDEFINE_DMI_ATTR_WITH_SHOW(bios_vendor,		0444, DMI_BIOS_VENDOR);
4262306a36Sopenharmony_ciDEFINE_DMI_ATTR_WITH_SHOW(bios_version,		0444, DMI_BIOS_VERSION);
4362306a36Sopenharmony_ciDEFINE_DMI_ATTR_WITH_SHOW(bios_date,		0444, DMI_BIOS_DATE);
4462306a36Sopenharmony_ciDEFINE_DMI_ATTR_WITH_SHOW(sys_vendor,		0444, DMI_SYS_VENDOR);
4562306a36Sopenharmony_ciDEFINE_DMI_ATTR_WITH_SHOW(bios_release,		0444, DMI_BIOS_RELEASE);
4662306a36Sopenharmony_ciDEFINE_DMI_ATTR_WITH_SHOW(ec_firmware_release,	0444, DMI_EC_FIRMWARE_RELEASE);
4762306a36Sopenharmony_ciDEFINE_DMI_ATTR_WITH_SHOW(product_name,		0444, DMI_PRODUCT_NAME);
4862306a36Sopenharmony_ciDEFINE_DMI_ATTR_WITH_SHOW(product_version,	0444, DMI_PRODUCT_VERSION);
4962306a36Sopenharmony_ciDEFINE_DMI_ATTR_WITH_SHOW(product_serial,	0400, DMI_PRODUCT_SERIAL);
5062306a36Sopenharmony_ciDEFINE_DMI_ATTR_WITH_SHOW(product_uuid,		0400, DMI_PRODUCT_UUID);
5162306a36Sopenharmony_ciDEFINE_DMI_ATTR_WITH_SHOW(product_sku,		0444, DMI_PRODUCT_SKU);
5262306a36Sopenharmony_ciDEFINE_DMI_ATTR_WITH_SHOW(product_family,	0444, DMI_PRODUCT_FAMILY);
5362306a36Sopenharmony_ciDEFINE_DMI_ATTR_WITH_SHOW(board_vendor,		0444, DMI_BOARD_VENDOR);
5462306a36Sopenharmony_ciDEFINE_DMI_ATTR_WITH_SHOW(board_name,		0444, DMI_BOARD_NAME);
5562306a36Sopenharmony_ciDEFINE_DMI_ATTR_WITH_SHOW(board_version,	0444, DMI_BOARD_VERSION);
5662306a36Sopenharmony_ciDEFINE_DMI_ATTR_WITH_SHOW(board_serial,		0400, DMI_BOARD_SERIAL);
5762306a36Sopenharmony_ciDEFINE_DMI_ATTR_WITH_SHOW(board_asset_tag,	0444, DMI_BOARD_ASSET_TAG);
5862306a36Sopenharmony_ciDEFINE_DMI_ATTR_WITH_SHOW(chassis_vendor,	0444, DMI_CHASSIS_VENDOR);
5962306a36Sopenharmony_ciDEFINE_DMI_ATTR_WITH_SHOW(chassis_type,		0444, DMI_CHASSIS_TYPE);
6062306a36Sopenharmony_ciDEFINE_DMI_ATTR_WITH_SHOW(chassis_version,	0444, DMI_CHASSIS_VERSION);
6162306a36Sopenharmony_ciDEFINE_DMI_ATTR_WITH_SHOW(chassis_serial,	0400, DMI_CHASSIS_SERIAL);
6262306a36Sopenharmony_ciDEFINE_DMI_ATTR_WITH_SHOW(chassis_asset_tag,	0444, DMI_CHASSIS_ASSET_TAG);
6362306a36Sopenharmony_ci
6462306a36Sopenharmony_cistatic void ascii_filter(char *d, const char *s)
6562306a36Sopenharmony_ci{
6662306a36Sopenharmony_ci	/* Filter out characters we don't want to see in the modalias string */
6762306a36Sopenharmony_ci	for (; *s; s++)
6862306a36Sopenharmony_ci		if (*s > ' ' && *s < 127 && *s != ':')
6962306a36Sopenharmony_ci			*(d++) = *s;
7062306a36Sopenharmony_ci
7162306a36Sopenharmony_ci	*d = 0;
7262306a36Sopenharmony_ci}
7362306a36Sopenharmony_ci
7462306a36Sopenharmony_cistatic ssize_t get_modalias(char *buffer, size_t buffer_size)
7562306a36Sopenharmony_ci{
7662306a36Sopenharmony_ci	/*
7762306a36Sopenharmony_ci	 * Note new fields need to be added at the end to keep compatibility
7862306a36Sopenharmony_ci	 * with udev's hwdb which does matches on "`cat dmi/id/modalias`*".
7962306a36Sopenharmony_ci	 */
8062306a36Sopenharmony_ci	static const struct mafield {
8162306a36Sopenharmony_ci		const char *prefix;
8262306a36Sopenharmony_ci		int field;
8362306a36Sopenharmony_ci	} fields[] = {
8462306a36Sopenharmony_ci		{ "bvn", DMI_BIOS_VENDOR },
8562306a36Sopenharmony_ci		{ "bvr", DMI_BIOS_VERSION },
8662306a36Sopenharmony_ci		{ "bd",  DMI_BIOS_DATE },
8762306a36Sopenharmony_ci		{ "br",  DMI_BIOS_RELEASE },
8862306a36Sopenharmony_ci		{ "efr", DMI_EC_FIRMWARE_RELEASE },
8962306a36Sopenharmony_ci		{ "svn", DMI_SYS_VENDOR },
9062306a36Sopenharmony_ci		{ "pn",  DMI_PRODUCT_NAME },
9162306a36Sopenharmony_ci		{ "pvr", DMI_PRODUCT_VERSION },
9262306a36Sopenharmony_ci		{ "rvn", DMI_BOARD_VENDOR },
9362306a36Sopenharmony_ci		{ "rn",  DMI_BOARD_NAME },
9462306a36Sopenharmony_ci		{ "rvr", DMI_BOARD_VERSION },
9562306a36Sopenharmony_ci		{ "cvn", DMI_CHASSIS_VENDOR },
9662306a36Sopenharmony_ci		{ "ct",  DMI_CHASSIS_TYPE },
9762306a36Sopenharmony_ci		{ "cvr", DMI_CHASSIS_VERSION },
9862306a36Sopenharmony_ci		{ "sku", DMI_PRODUCT_SKU },
9962306a36Sopenharmony_ci		{ NULL,  DMI_NONE }
10062306a36Sopenharmony_ci	};
10162306a36Sopenharmony_ci
10262306a36Sopenharmony_ci	ssize_t l, left;
10362306a36Sopenharmony_ci	char *p;
10462306a36Sopenharmony_ci	const struct mafield *f;
10562306a36Sopenharmony_ci
10662306a36Sopenharmony_ci	strcpy(buffer, "dmi");
10762306a36Sopenharmony_ci	p = buffer + 3; left = buffer_size - 4;
10862306a36Sopenharmony_ci
10962306a36Sopenharmony_ci	for (f = fields; f->prefix && left > 0; f++) {
11062306a36Sopenharmony_ci		const char *c;
11162306a36Sopenharmony_ci		char *t;
11262306a36Sopenharmony_ci
11362306a36Sopenharmony_ci		c = dmi_get_system_info(f->field);
11462306a36Sopenharmony_ci		if (!c)
11562306a36Sopenharmony_ci			continue;
11662306a36Sopenharmony_ci
11762306a36Sopenharmony_ci		t = kmalloc(strlen(c) + 1, GFP_KERNEL);
11862306a36Sopenharmony_ci		if (!t)
11962306a36Sopenharmony_ci			break;
12062306a36Sopenharmony_ci		ascii_filter(t, c);
12162306a36Sopenharmony_ci		l = scnprintf(p, left, ":%s%s", f->prefix, t);
12262306a36Sopenharmony_ci		kfree(t);
12362306a36Sopenharmony_ci
12462306a36Sopenharmony_ci		p += l;
12562306a36Sopenharmony_ci		left -= l;
12662306a36Sopenharmony_ci	}
12762306a36Sopenharmony_ci
12862306a36Sopenharmony_ci	p[0] = ':';
12962306a36Sopenharmony_ci	p[1] = 0;
13062306a36Sopenharmony_ci
13162306a36Sopenharmony_ci	return p - buffer + 1;
13262306a36Sopenharmony_ci}
13362306a36Sopenharmony_ci
13462306a36Sopenharmony_cistatic ssize_t sys_dmi_modalias_show(struct device *dev,
13562306a36Sopenharmony_ci				     struct device_attribute *attr, char *page)
13662306a36Sopenharmony_ci{
13762306a36Sopenharmony_ci	ssize_t r;
13862306a36Sopenharmony_ci	r = get_modalias(page, PAGE_SIZE-1);
13962306a36Sopenharmony_ci	page[r] = '\n';
14062306a36Sopenharmony_ci	page[r+1] = 0;
14162306a36Sopenharmony_ci	return r+1;
14262306a36Sopenharmony_ci}
14362306a36Sopenharmony_ci
14462306a36Sopenharmony_cistatic struct device_attribute sys_dmi_modalias_attr =
14562306a36Sopenharmony_ci	__ATTR(modalias, 0444, sys_dmi_modalias_show, NULL);
14662306a36Sopenharmony_ci
14762306a36Sopenharmony_cistatic struct attribute *sys_dmi_attributes[DMI_STRING_MAX+2];
14862306a36Sopenharmony_ci
14962306a36Sopenharmony_cistatic struct attribute_group sys_dmi_attribute_group = {
15062306a36Sopenharmony_ci	.attrs = sys_dmi_attributes,
15162306a36Sopenharmony_ci};
15262306a36Sopenharmony_ci
15362306a36Sopenharmony_cistatic const struct attribute_group* sys_dmi_attribute_groups[] = {
15462306a36Sopenharmony_ci	&sys_dmi_attribute_group,
15562306a36Sopenharmony_ci	NULL
15662306a36Sopenharmony_ci};
15762306a36Sopenharmony_ci
15862306a36Sopenharmony_cistatic int dmi_dev_uevent(const struct device *dev, struct kobj_uevent_env *env)
15962306a36Sopenharmony_ci{
16062306a36Sopenharmony_ci	ssize_t len;
16162306a36Sopenharmony_ci
16262306a36Sopenharmony_ci	if (add_uevent_var(env, "MODALIAS="))
16362306a36Sopenharmony_ci		return -ENOMEM;
16462306a36Sopenharmony_ci	len = get_modalias(&env->buf[env->buflen - 1],
16562306a36Sopenharmony_ci			   sizeof(env->buf) - env->buflen);
16662306a36Sopenharmony_ci	if (len >= (sizeof(env->buf) - env->buflen))
16762306a36Sopenharmony_ci		return -ENOMEM;
16862306a36Sopenharmony_ci	env->buflen += len;
16962306a36Sopenharmony_ci	return 0;
17062306a36Sopenharmony_ci}
17162306a36Sopenharmony_ci
17262306a36Sopenharmony_cistatic struct class dmi_class = {
17362306a36Sopenharmony_ci	.name = "dmi",
17462306a36Sopenharmony_ci	.dev_release = (void(*)(struct device *)) kfree,
17562306a36Sopenharmony_ci	.dev_uevent = dmi_dev_uevent,
17662306a36Sopenharmony_ci};
17762306a36Sopenharmony_ci
17862306a36Sopenharmony_cistatic struct device *dmi_dev;
17962306a36Sopenharmony_ci
18062306a36Sopenharmony_ci/* Initialization */
18162306a36Sopenharmony_ci
18262306a36Sopenharmony_ci#define ADD_DMI_ATTR(_name, _field) \
18362306a36Sopenharmony_ci	if (dmi_get_system_info(_field)) \
18462306a36Sopenharmony_ci		sys_dmi_attributes[i++] = &sys_dmi_##_name##_attr.dev_attr.attr;
18562306a36Sopenharmony_ci
18662306a36Sopenharmony_ci/* In a separate function to keep gcc 3.2 happy - do NOT merge this in
18762306a36Sopenharmony_ci   dmi_id_init! */
18862306a36Sopenharmony_cistatic void __init dmi_id_init_attr_table(void)
18962306a36Sopenharmony_ci{
19062306a36Sopenharmony_ci	int i;
19162306a36Sopenharmony_ci
19262306a36Sopenharmony_ci	/* Not necessarily all DMI fields are available on all
19362306a36Sopenharmony_ci	 * systems, hence let's built an attribute table of just
19462306a36Sopenharmony_ci	 * what's available */
19562306a36Sopenharmony_ci	i = 0;
19662306a36Sopenharmony_ci	ADD_DMI_ATTR(bios_vendor,       DMI_BIOS_VENDOR);
19762306a36Sopenharmony_ci	ADD_DMI_ATTR(bios_version,      DMI_BIOS_VERSION);
19862306a36Sopenharmony_ci	ADD_DMI_ATTR(bios_date,         DMI_BIOS_DATE);
19962306a36Sopenharmony_ci	ADD_DMI_ATTR(bios_release,      DMI_BIOS_RELEASE);
20062306a36Sopenharmony_ci	ADD_DMI_ATTR(ec_firmware_release, DMI_EC_FIRMWARE_RELEASE);
20162306a36Sopenharmony_ci	ADD_DMI_ATTR(sys_vendor,        DMI_SYS_VENDOR);
20262306a36Sopenharmony_ci	ADD_DMI_ATTR(product_name,      DMI_PRODUCT_NAME);
20362306a36Sopenharmony_ci	ADD_DMI_ATTR(product_version,   DMI_PRODUCT_VERSION);
20462306a36Sopenharmony_ci	ADD_DMI_ATTR(product_serial,    DMI_PRODUCT_SERIAL);
20562306a36Sopenharmony_ci	ADD_DMI_ATTR(product_uuid,      DMI_PRODUCT_UUID);
20662306a36Sopenharmony_ci	ADD_DMI_ATTR(product_family,    DMI_PRODUCT_FAMILY);
20762306a36Sopenharmony_ci	ADD_DMI_ATTR(product_sku,       DMI_PRODUCT_SKU);
20862306a36Sopenharmony_ci	ADD_DMI_ATTR(board_vendor,      DMI_BOARD_VENDOR);
20962306a36Sopenharmony_ci	ADD_DMI_ATTR(board_name,        DMI_BOARD_NAME);
21062306a36Sopenharmony_ci	ADD_DMI_ATTR(board_version,     DMI_BOARD_VERSION);
21162306a36Sopenharmony_ci	ADD_DMI_ATTR(board_serial,      DMI_BOARD_SERIAL);
21262306a36Sopenharmony_ci	ADD_DMI_ATTR(board_asset_tag,   DMI_BOARD_ASSET_TAG);
21362306a36Sopenharmony_ci	ADD_DMI_ATTR(chassis_vendor,    DMI_CHASSIS_VENDOR);
21462306a36Sopenharmony_ci	ADD_DMI_ATTR(chassis_type,      DMI_CHASSIS_TYPE);
21562306a36Sopenharmony_ci	ADD_DMI_ATTR(chassis_version,   DMI_CHASSIS_VERSION);
21662306a36Sopenharmony_ci	ADD_DMI_ATTR(chassis_serial,    DMI_CHASSIS_SERIAL);
21762306a36Sopenharmony_ci	ADD_DMI_ATTR(chassis_asset_tag, DMI_CHASSIS_ASSET_TAG);
21862306a36Sopenharmony_ci	sys_dmi_attributes[i++] = &sys_dmi_modalias_attr.attr;
21962306a36Sopenharmony_ci}
22062306a36Sopenharmony_ci
22162306a36Sopenharmony_cistatic int __init dmi_id_init(void)
22262306a36Sopenharmony_ci{
22362306a36Sopenharmony_ci	int ret;
22462306a36Sopenharmony_ci
22562306a36Sopenharmony_ci	if (!dmi_available)
22662306a36Sopenharmony_ci		return -ENODEV;
22762306a36Sopenharmony_ci
22862306a36Sopenharmony_ci	dmi_id_init_attr_table();
22962306a36Sopenharmony_ci
23062306a36Sopenharmony_ci	ret = class_register(&dmi_class);
23162306a36Sopenharmony_ci	if (ret)
23262306a36Sopenharmony_ci		return ret;
23362306a36Sopenharmony_ci
23462306a36Sopenharmony_ci	dmi_dev = kzalloc(sizeof(*dmi_dev), GFP_KERNEL);
23562306a36Sopenharmony_ci	if (!dmi_dev) {
23662306a36Sopenharmony_ci		ret = -ENOMEM;
23762306a36Sopenharmony_ci		goto fail_class_unregister;
23862306a36Sopenharmony_ci	}
23962306a36Sopenharmony_ci
24062306a36Sopenharmony_ci	dmi_dev->class = &dmi_class;
24162306a36Sopenharmony_ci	dev_set_name(dmi_dev, "id");
24262306a36Sopenharmony_ci	dmi_dev->groups = sys_dmi_attribute_groups;
24362306a36Sopenharmony_ci
24462306a36Sopenharmony_ci	ret = device_register(dmi_dev);
24562306a36Sopenharmony_ci	if (ret)
24662306a36Sopenharmony_ci		goto fail_put_dmi_dev;
24762306a36Sopenharmony_ci
24862306a36Sopenharmony_ci	return 0;
24962306a36Sopenharmony_ci
25062306a36Sopenharmony_cifail_put_dmi_dev:
25162306a36Sopenharmony_ci	put_device(dmi_dev);
25262306a36Sopenharmony_ci
25362306a36Sopenharmony_cifail_class_unregister:
25462306a36Sopenharmony_ci	class_unregister(&dmi_class);
25562306a36Sopenharmony_ci
25662306a36Sopenharmony_ci	return ret;
25762306a36Sopenharmony_ci}
25862306a36Sopenharmony_ci
25962306a36Sopenharmony_ciarch_initcall(dmi_id_init);
260