#include <linux/ctype.h>
#include <linux/cleanup.h>
#include <linux/device.h>
#include <linux/export.h>
#include <linux/hashtable.h>
#include <linux/kstrtox.h>
#include <linux/of.h>
#include <linux/slab.h>
#include <linux/static_key.h>
#include <linux/string.h>
#include <linux/stringhash.h>
#include <linux/types.h>
#include "quirks.h"
#define SCMI_QUIRKS_HT_SZ 4
struct scmi_quirk {
bool enabled;
const char *name;
const char *vendor;
const char *sub_vendor_id;
const char *impl_ver_range;
u32 start_range;
u32 end_range;
struct static_key_false *key;
struct hlist_node hash;
unsigned int hkey;
const char *const compats[];
};
#define __DEFINE_SCMI_QUIRK_ENTRY(_qn, _ven, _sub, _impl, ...) \
static struct scmi_quirk scmi_quirk_entry_ ## _qn = { \
.name = __stringify(quirk_ ## _qn), \
.vendor = _ven, \
.sub_vendor_id = _sub, \
.impl_ver_range = _impl, \
.key = &(scmi_quirk_ ## _qn), \
.compats = { __VA_ARGS__ __VA_OPT__(,) NULL }, \
}
#define __DECLARE_SCMI_QUIRK_ENTRY(_qn) (&(scmi_quirk_entry_ ## _qn))
#define DEFINE_SCMI_QUIRK(_qn, _ven, _sub, _impl, ...) \
DEFINE_STATIC_KEY_FALSE(scmi_quirk_ ## _qn); \
__DEFINE_SCMI_QUIRK_ENTRY(_qn, _ven, _sub, _impl, ##__VA_ARGS__)
#define DEFINE_SCMI_QUIRK_EXPORTED(_qn, _ven, _sub, _impl, ...) \
DEFINE_STATIC_KEY_FALSE(scmi_quirk_ ## _qn); \
EXPORT_SYMBOL_GPL(scmi_quirk_ ## _qn); \
__DEFINE_SCMI_QUIRK_ENTRY(_qn, _ven, _sub, _impl, ##__VA_ARGS__)
DEFINE_SCMI_QUIRK(clock_rates_triplet_out_of_spec, NULL, NULL, NULL);
DEFINE_SCMI_QUIRK(perf_level_get_fc_force, "Qualcomm", NULL, "0x20000-");
static struct scmi_quirk *scmi_quirks_table[] = {
__DECLARE_SCMI_QUIRK_ENTRY(clock_rates_triplet_out_of_spec),
__DECLARE_SCMI_QUIRK_ENTRY(perf_level_get_fc_force),
NULL
};
static DEFINE_READ_MOSTLY_HASHTABLE(scmi_quirks_ht, SCMI_QUIRKS_HT_SZ);
static unsigned int scmi_quirk_signature(const char *vend, const char *sub_vend)
{
char *signature, *p;
unsigned int hash32;
unsigned long hash = 0;
signature = kasprintf(GFP_KERNEL, "|%s|%s|", vend ?: "", sub_vend ?: "");
if (!signature)
return 0;
pr_debug("SCMI Quirk Signature >>>%s<<<\n", signature);
p = signature;
while (*p)
hash = partial_name_hash(tolower(*p++), hash);
hash32 = end_name_hash(hash);
kfree(signature);
return hash32;
}
static int scmi_quirk_range_parse(struct scmi_quirk *quirk)
{
const char *last, *first __free(kfree) = NULL;
size_t len;
char *sep;
int ret;
quirk->start_range = 0;
quirk->end_range = 0xFFFFFFFF;
len = quirk->impl_ver_range ? strlen(quirk->impl_ver_range) : 0;
if (!len)
return 0;
first = kmemdup(quirk->impl_ver_range, len + 1, GFP_KERNEL);
if (!first)
return -ENOMEM;
last = first + len - 1;
sep = strchr(first, '-');
if (sep)
*sep = '\0';
if (sep == first)
ret = kstrtouint(first + 1, 0, &quirk->end_range);
else
ret = kstrtouint(first, 0, &quirk->start_range);
if (ret)
return ret;
if (!sep)
quirk->end_range = quirk->start_range;
else if (sep != last)
ret = kstrtouint(sep + 1, 0, &quirk->end_range);
if (quirk->start_range > quirk->end_range)
return -EINVAL;
return ret;
}
void scmi_quirks_initialize(void)
{
struct scmi_quirk *quirk;
int i;
for (i = 0, quirk = scmi_quirks_table[0]; quirk;
i++, quirk = scmi_quirks_table[i]) {
int ret;
ret = scmi_quirk_range_parse(quirk);
if (ret) {
pr_err("SCMI skip QUIRK [%s] - BAD RANGE - |%s|\n",
quirk->name, quirk->impl_ver_range);
continue;
}
quirk->hkey = scmi_quirk_signature(quirk->vendor,
quirk->sub_vendor_id);
hash_add(scmi_quirks_ht, &quirk->hash, quirk->hkey);
pr_debug("Registered SCMI QUIRK [%s] -- %p - Key [0x%08X] - %s/%s/[0x%08X-0x%08X]\n",
quirk->name, quirk, quirk->hkey,
quirk->vendor, quirk->sub_vendor_id,
quirk->start_range, quirk->end_range);
}
pr_debug("SCMI Quirks initialized\n");
}
void scmi_quirks_enable(struct device *dev, const char *vend,
const char *subv, const u32 impl)
{
for (int i = 3; i >= 0; i--) {
struct scmi_quirk *quirk;
unsigned int hkey;
hkey = scmi_quirk_signature(i > 1 ? vend : NULL,
i > 2 ? subv : NULL);
hash_for_each_possible(scmi_quirks_ht, quirk, hash, hkey) {
if (quirk->enabled || quirk->hkey != hkey ||
impl < quirk->start_range ||
impl > quirk->end_range)
continue;
if (quirk->compats[0] &&
!of_machine_compatible_match(quirk->compats))
continue;
dev_info(dev, "Enabling SCMI Quirk [%s]\n",
quirk->name);
dev_dbg(dev,
"Quirk matched on: %s/%s/%s/[0x%08X-0x%08X]\n",
quirk->compats[0], quirk->vendor,
quirk->sub_vendor_id,
quirk->start_range, quirk->end_range);
static_branch_enable(quirk->key);
quirk->enabled = true;
}
}
}