// SPDX-License-Identifier: GPL-2.0-or-later /* * AMD 3D V-Cache Performance Optimizer Driver * * Copyright (c) 2024, Advanced Micro Devices, Inc. * All Rights Reserved. * * Authors: Basavaraj Natikar * Perry Yuan * Mario Limonciello */ #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt #include #include #include #include #include #include #include #include #include #include static char *x3d_mode = "frequency"; module_param(x3d_mode, charp, 0); MODULE_PARM_DESC(x3d_mode, "Initial 3D-VCache mode; 'frequency' (default) or 'cache'"); #define DSM_REVISION_ID 0 #define DSM_SET_X3D_MODE 1 static guid_t x3d_guid = GUID_INIT(0xdff8e55f, 0xbcfd, 0x46fb, 0xba, 0x0a, 0xef, 0xd0, 0x45, 0x0f, 0x34, 0xee); enum amd_x3d_mode_type { MODE_INDEX_FREQ, MODE_INDEX_CACHE, }; static const char * const amd_x3d_mode_strings[] = { [MODE_INDEX_FREQ] = "frequency", [MODE_INDEX_CACHE] = "cache", }; struct amd_x3d_dev { struct device *dev; acpi_handle ahandle; /* To protect x3d mode setting */ struct mutex lock; enum amd_x3d_mode_type curr_mode; }; static int amd_x3d_get_mode(struct amd_x3d_dev *data) { guard(mutex)(&data->lock); return data->curr_mode; } static int amd_x3d_mode_switch(struct amd_x3d_dev *data, int new_state) { union acpi_object *out, argv; guard(mutex)(&data->lock); argv.type = ACPI_TYPE_INTEGER; argv.integer.value = new_state; out = acpi_evaluate_dsm(data->ahandle, &x3d_guid, DSM_REVISION_ID, DSM_SET_X3D_MODE, &argv); if (!out) { dev_err(data->dev, "failed to evaluate _DSM\n"); return -EINVAL; } data->curr_mode = new_state; kfree(out); return 0; } static ssize_t amd_x3d_mode_store(struct device *dev, struct device_attribute *attr, const char *buf, size_t count) { struct amd_x3d_dev *data = dev_get_drvdata(dev); int ret; ret = sysfs_match_string(amd_x3d_mode_strings, buf); if (ret < 0) return ret; ret = amd_x3d_mode_switch(data, ret); if (ret < 0) return ret; return count; } static ssize_t amd_x3d_mode_show(struct device *dev, struct device_attribute *attr, char *buf) { struct amd_x3d_dev *data = dev_get_drvdata(dev); int mode = amd_x3d_get_mode(data); return sysfs_emit(buf, "%s\n", amd_x3d_mode_strings[mode]); } static DEVICE_ATTR_RW(amd_x3d_mode); static struct attribute *amd_x3d_attrs[] = { &dev_attr_amd_x3d_mode.attr, NULL }; ATTRIBUTE_GROUPS(amd_x3d); static int amd_x3d_resume_handler(struct device *dev) { struct amd_x3d_dev *data = dev_get_drvdata(dev); int ret = amd_x3d_get_mode(data); return amd_x3d_mode_switch(data, ret); } static DEFINE_SIMPLE_DEV_PM_OPS(amd_x3d_pm, NULL, amd_x3d_resume_handler); static const struct acpi_device_id amd_x3d_acpi_ids[] = { {"AMDI0101"}, { }, }; MODULE_DEVICE_TABLE(acpi, amd_x3d_acpi_ids); static int amd_x3d_probe(struct platform_device *pdev) { struct amd_x3d_dev *data; acpi_handle handle; int ret; handle = ACPI_HANDLE(&pdev->dev); if (!handle) return -ENODEV; if (!acpi_check_dsm(handle, &x3d_guid, DSM_REVISION_ID, BIT(DSM_SET_X3D_MODE))) return -ENODEV; data = devm_kzalloc(&pdev->dev, sizeof(*data), GFP_KERNEL); if (!data) return -ENOMEM; data->dev = &pdev->dev; ret = devm_mutex_init(data->dev, &data->lock); if (ret) return ret; data->ahandle = handle; platform_set_drvdata(pdev, data); ret = match_string(amd_x3d_mode_strings, ARRAY_SIZE(amd_x3d_mode_strings), x3d_mode); if (ret < 0) return dev_err_probe(&pdev->dev, -EINVAL, "invalid mode %s\n", x3d_mode); return amd_x3d_mode_switch(data, ret); } static struct platform_driver amd_3d_vcache_driver = { .driver = { .name = "amd_x3d_vcache", .dev_groups = amd_x3d_groups, .acpi_match_table = amd_x3d_acpi_ids, .pm = pm_sleep_ptr(&amd_x3d_pm), }, .probe = amd_x3d_probe, }; module_platform_driver(amd_3d_vcache_driver); MODULE_DESCRIPTION("AMD 3D V-Cache Performance Optimizer Driver"); MODULE_LICENSE("GPL");