On 7/11/2019 1:12 AM, Viresh Kumar wrote:
> On 10-07-19, 18:37, Natarajan, Janakarajan wrote:
>> diff --git a/drivers/cpufreq/amd-cpufreq.c b/drivers/cpufreq/amd-cpufreq.c
>> +#define pr_fmt(fmt) "AMD Cpufreq: " fmt
>> +
>> +#include <linux/kernel.h>
>> +#include <linux/module.h>
>> +#include <linux/cpu.h>
>> +#include <linux/vmalloc.h>
>> +#include <linux/cpufreq.h>
>> +#include <linux/acpi.h>
>> +#include <linux/delay.h>
> Please keep them in alphabetical order.


Ok.


>
>> +
>> +#include <asm/unaligned.h>
>> +
>> +#include <acpi/cppc_acpi.h>
>> +
>> +struct amd_desc {
>> +    int cpu_id;
>> +    struct cppc_ctrls ctrls;
>> +    struct kobject kobj;
>> +};
>> +
>> +struct amd_desc **all_cpu_data;
>> +
>> +static unsigned int cppc_enable;
>> +module_param(cppc_enable, uint, 0644);
>> +MODULE_PARM_DESC(cppc_enable,
>> +             "1 - enable AMD CpuFreq, create CPPC sysfs entries.");
>> +
>> +#define to_amd_desc(a) container_of(a, struct amd_desc, kobj)
>> +
>> +#define show_func(access_fn, struct_name, member_name)                      
>> \
>> +    static ssize_t show_##member_name(struct kobject *kobj,         \
>> +                                      struct kobj_attribute *attr,  \
>> +                                      char *buf)                    \
>> +    {                                                               \
>> +            struct amd_desc *desc = to_amd_desc(kobj);              \
>> +            struct struct_name st_name = {0};                       \
>> +            int ret;                                                \
>> +                                                                    \
>> +            ret = access_fn(desc->cpu_id, &st_name);                \
>> +            if (ret)                                                \
>> +                    return ret;                                     \
>> +                                                                    \
>> +            return scnprintf(buf, PAGE_SIZE, "%llu\n",              \
>> +                             (u64)st_name.member_name);             \
>> +    }                                                               \
>> +
>> +#define store_func(struct_name, member_name, reg_idx)                       
>> \
>> +    static ssize_t store_##member_name(struct kobject *kobj,        \
>> +                                       struct kobj_attribute *attr, \
>> +                                       const char *buf, size_t count)\
>> +    {                                                               \
>> +            struct amd_desc *desc = to_amd_desc(kobj);              \
>> +            struct struct_name st_name = {0};                       \
>> +            u32 val;                                                \
>> +            int ret;                                                \
>> +                                                                    \
>> +            ret = kstrtou32(buf, 0, &val);                          \
>> +            if (ret)                                                \
>> +                    return ret;                                     \
>> +                                                                    \
>> +            st_name.member_name = val;                              \
>> +                                                                    \
>> +            ret = cppc_set_reg(desc->cpu_id, &st_name, reg_idx);    \
>> +            if (ret)                                                \
>> +                    return ret;                                     \
>> +                                                                    \
>> +            return count;                                           \
>> +    }                                                               \
>> +
>> +#define define_one_rw(struct_name, access_fn, member_name, reg_idx) \
>> +    show_func(access_fn, struct_name, member_name)                  \
>> +    store_func(struct_name, member_name, reg_idx)                   \
>> +    define_one_global_rw(member_name)
>> +
>> +define_one_rw(cppc_ctrls, cppc_get_ctrls, enable, ENABLE);
>> +define_one_rw(cppc_ctrls, cppc_get_ctrls, max_perf, MAX_PERF);
>> +define_one_rw(cppc_ctrls, cppc_get_ctrls, min_perf, MIN_PERF);
>> +define_one_rw(cppc_ctrls, cppc_get_ctrls, desired_perf, DESIRED_PERF);
>> +define_one_rw(cppc_ctrls, cppc_get_ctrls, auto_sel_enable, AUTO_SEL_ENABLE);
>> +
>> +static struct attribute *amd_cpufreq_attributes[] = {
>> +    &enable.attr,
>> +    &max_perf.attr,
>> +    &min_perf.attr,
>> +    &desired_perf.attr,
>> +    &auto_sel_enable.attr,
>> +    NULL
>> +};
>> +
>> +static const struct attribute_group amd_cpufreq_attr_group = {
>> +    .attrs = amd_cpufreq_attributes,
>> +};
>> +
>> +static struct kobj_type amd_cpufreq_type = {
>> +    .sysfs_ops = &kobj_sysfs_ops,
>> +    .default_attrs = amd_cpufreq_attributes,
>> +};
>> +
>> +static int amd_cpufreq_cpu_init(struct cpufreq_policy *policy)
>> +{
>> +    return 0;
>> +}
>> +
>> +static int amd_cpufreq_cpu_exit(struct cpufreq_policy *policy)
>> +{
>> +    return 0;
>> +}
>> +
>> +static int amd_cpufreq_cpu_verify(struct cpufreq_policy *policy)
>> +{
>> +    return 0;
>> +}
>> +
>> +static int amd_cpufreq_cpu_target_index(struct cpufreq_policy *policy,
>> +                                    unsigned int index)
>> +{
>> +    return 0;
>> +}
> All empty helpers ? There is nothing you need to do ?


When we posted v2 of this patchset, Rafael let us know that he was 
uncomfortable

going behind the (acpi-cpufreq) drivers back by letting the user 
communicate directly

with the platform. That's the reason we have an empty driver whose 
primary purpose

is to expose sysfs entries for the user.


>
>> +
>> +static struct cpufreq_driver amd_cpufreq_driver = {
>> +    .name = "amd_cpufreq",
>> +    .init = amd_cpufreq_cpu_init,
>> +    .exit = amd_cpufreq_cpu_exit,
>> +    .verify = amd_cpufreq_cpu_verify,
>> +    .target_index = amd_cpufreq_cpu_target_index,
>> +};
>> +
>> +static void amd_cpufreq_sysfs_delete_params(void)
>> +{
>> +    int i;
>> +
>> +    for_each_possible_cpu(i) {
>> +            if (all_cpu_data[i]) {
>> +                    kobject_del(&all_cpu_data[i]->kobj);
> Shouldn't you use kobject_put() instead of this ?


Ok.


>
>> +                    kfree(all_cpu_data[i]);
>> +            }
>> +    }
>> +
>> +    kfree(all_cpu_data);
>> +}
>> +
>> +static int __init amd_cpufreq_sysfs_expose_params(void)
>> +{
>> +    struct device *cpu_dev;
>> +    int i, ret;
>> +
>> +    all_cpu_data = kcalloc(num_possible_cpus(), sizeof(void *),
>> +                           GFP_KERNEL);
>> +
>> +    if (!all_cpu_data)
>> +            return -ENOMEM;
>> +
>> +    for_each_possible_cpu(i) {
>> +            all_cpu_data[i] = kzalloc(sizeof(struct amd_desc), GFP_KERNEL);
>> +            if (!all_cpu_data[i]) {
>> +                    ret = -ENOMEM;
>> +                    goto free;
>> +            }
>> +
>> +            all_cpu_data[i]->cpu_id = i;
>> +            cpu_dev = get_cpu_device(i);
>> +            ret = kobject_init_and_add(&all_cpu_data[i]->kobj, 
>> &amd_cpufreq_type,
>> +                                       &cpu_dev->kobj, "amd_cpufreq");
>> +            if (ret)
>> +                    goto free;
>> +    }
>> +
>> +    return 0;
>> +free:
>> +    amd_cpufreq_sysfs_delete_params();
>> +    return ret;
>> +}
> Instead of doing this once for all CPUs, it would be better to do it
> every time the ->init() callback of the driver gets called. If you
> have one cpufreq policy for each CPU (i.e. no CPUs share clock lines),
> then the init() callback will get called once for each CPU.


We are trying to avoid any use of the cpufreq mechanism.


Thanks,

Janakarajan


>
>> +
>> +static int __init amd_cpufreq_init(void)
>> +{
>> +    int ret = 0;
>> +
>> +    /*
>> +     * Use only if:
>> +     * - AMD,
>> +     * - Family 17h (or) newer and,
>> +     * - Explicitly enabled
>> +     */
>> +    if (boot_cpu_data.x86_vendor != X86_VENDOR_AMD ||
>> +        boot_cpu_data.x86 < 0x17 || !cppc_enable)
>> +            return -ENODEV;
>> +
>> +    ret = cpufreq_register_driver(&amd_cpufreq_driver);
>> +    if (ret) {
>> +            pr_info("Failed to register driver\n");
>> +            goto out;
>> +    }
>> +
>> +    ret = amd_cpufreq_sysfs_expose_params();
>> +    if (ret) {
>> +            pr_info("Could not create sysfs entries\n");
>> +            cpufreq_unregister_driver(&amd_cpufreq_driver);
>> +            goto out;
>> +    }
>> +
>> +    pr_info("Using amd-cpufreq driver\n");
>> +    return ret;
>> +
>> +out:
>> +    return ret;
>> +}
>> +
>> +static void __exit amd_cpufreq_exit(void)
>> +{
>> +    amd_cpufreq_sysfs_delete_params();
>> +    cpufreq_unregister_driver(&amd_cpufreq_driver);
>> +}
>> +
>> +static const struct acpi_device_id amd_acpi_ids[] __used = {
>> +    {ACPI_PROCESSOR_DEVICE_HID, },
>> +    {}
>> +};
>> +
>> +device_initcall(amd_cpufreq_init);
>> +module_exit(amd_cpufreq_exit);
>> +MODULE_DEVICE_TABLE(acpi, amd_acpi_ids);
> All three should be placed directly below the struct/function they
> represent without any blank lines in between. As suggested in
> kernel documentation.
>
>> +
>> +MODULE_AUTHOR("Janakarajan Natarajan");
>> +MODULE_DESCRIPTION("AMD CPUFreq driver based on ACPI CPPC v6.1 spec");
>> +MODULE_LICENSE("GPL");
> Should this be "GPL v2" ?
>
>> -- 
>> 2.17.1

Reply via email to