Commit 8b4e6dea authored by Anju T Sudhakar's avatar Anju T Sudhakar Committed by Michael Ellerman

powerpc/perf: Pass struct imc_events as a parameter to imc_parse_event()

Remove the allocation of struct imc_events from imc_parse_event().
Instead pass imc_events as a parameter to imc_parse_event(), which is
a pointer to a slot in the array allocated in
update_events_in_group().

Reported-by: Dan Carpenter ("powerpc/perf: Fix a sizeof() typo so we allocate less memory")
Suggested-by: default avatarMichael Ellerman <mpe@ellerman.id.au>
Signed-off-by: default avatarAnju T Sudhakar <anju@linux.vnet.ibm.com>
Signed-off-by: default avatarMichael Ellerman <mpe@ellerman.id.au>
parent ed8e443f
...@@ -71,7 +71,7 @@ struct imc_events { ...@@ -71,7 +71,7 @@ struct imc_events {
struct imc_pmu { struct imc_pmu {
struct pmu pmu; struct pmu pmu;
struct imc_mem_info *mem_info; struct imc_mem_info *mem_info;
struct imc_events **events; struct imc_events *events;
/* /*
* Attribute groups for the PMU. Slot 0 used for * Attribute groups for the PMU. Slot 0 used for
* format attribute, slot 1 used for cpusmask attribute, * format attribute, slot 1 used for cpusmask attribute,
......
...@@ -116,17 +116,13 @@ static struct attribute *device_str_attr_create(const char *name, const char *st ...@@ -116,17 +116,13 @@ static struct attribute *device_str_attr_create(const char *name, const char *st
return &attr->attr.attr; return &attr->attr.attr;
} }
struct imc_events *imc_parse_event(struct device_node *np, const char *scale, static int imc_parse_event(struct device_node *np, const char *scale,
const char *unit, const char *prefix, u32 base) const char *unit, const char *prefix,
u32 base, struct imc_events *event)
{ {
struct imc_events *event;
const char *s; const char *s;
u32 reg; u32 reg;
event = kzalloc(sizeof(struct imc_events), GFP_KERNEL);
if (!event)
return NULL;
if (of_property_read_u32(np, "reg", &reg)) if (of_property_read_u32(np, "reg", &reg))
goto error; goto error;
/* Add the base_reg value to the "reg" */ /* Add the base_reg value to the "reg" */
...@@ -157,14 +153,32 @@ struct imc_events *imc_parse_event(struct device_node *np, const char *scale, ...@@ -157,14 +153,32 @@ struct imc_events *imc_parse_event(struct device_node *np, const char *scale,
goto error; goto error;
} }
return event; return 0;
error: error:
kfree(event->unit); kfree(event->unit);
kfree(event->scale); kfree(event->scale);
kfree(event->name); kfree(event->name);
kfree(event); return -EINVAL;
}
/*
* imc_free_events: Function to cleanup the events list, having
* "nr_entries".
*/
static void imc_free_events(struct imc_events *events, int nr_entries)
{
int i;
/* Nothing to clean, return */
if (!events)
return;
for (i = 0; i < nr_entries; i++) {
kfree(events[i].unit);
kfree(events[i].scale);
kfree(events[i].name);
}
return NULL; kfree(events);
} }
/* /*
...@@ -176,9 +190,8 @@ static int update_events_in_group(struct device_node *node, struct imc_pmu *pmu) ...@@ -176,9 +190,8 @@ static int update_events_in_group(struct device_node *node, struct imc_pmu *pmu)
struct attribute_group *attr_group; struct attribute_group *attr_group;
struct attribute **attrs, *dev_str; struct attribute **attrs, *dev_str;
struct device_node *np, *pmu_events; struct device_node *np, *pmu_events;
struct imc_events *ev;
u32 handle, base_reg; u32 handle, base_reg;
int i=0, j=0, ct; int i = 0, j = 0, ct, ret;
const char *prefix, *g_scale, *g_unit; const char *prefix, *g_scale, *g_unit;
const char *ev_val_str, *ev_scale_str, *ev_unit_str; const char *ev_val_str, *ev_scale_str, *ev_unit_str;
...@@ -216,15 +229,17 @@ static int update_events_in_group(struct device_node *node, struct imc_pmu *pmu) ...@@ -216,15 +229,17 @@ static int update_events_in_group(struct device_node *node, struct imc_pmu *pmu)
ct = 0; ct = 0;
/* Parse the events and update the struct */ /* Parse the events and update the struct */
for_each_child_of_node(pmu_events, np) { for_each_child_of_node(pmu_events, np) {
ev = imc_parse_event(np, g_scale, g_unit, prefix, base_reg); ret = imc_parse_event(np, g_scale, g_unit, prefix, base_reg, &pmu->events[ct]);
if (ev) if (!ret)
pmu->events[ct++] = ev; ct++;
} }
/* Allocate memory for attribute group */ /* Allocate memory for attribute group */
attr_group = kzalloc(sizeof(*attr_group), GFP_KERNEL); attr_group = kzalloc(sizeof(*attr_group), GFP_KERNEL);
if (!attr_group) if (!attr_group) {
imc_free_events(pmu->events, ct);
return -ENOMEM; return -ENOMEM;
}
/* /*
* Allocate memory for attributes. * Allocate memory for attributes.
...@@ -237,31 +252,31 @@ static int update_events_in_group(struct device_node *node, struct imc_pmu *pmu) ...@@ -237,31 +252,31 @@ static int update_events_in_group(struct device_node *node, struct imc_pmu *pmu)
attrs = kcalloc(((ct * 3) + 1), sizeof(struct attribute *), GFP_KERNEL); attrs = kcalloc(((ct * 3) + 1), sizeof(struct attribute *), GFP_KERNEL);
if (!attrs) { if (!attrs) {
kfree(attr_group); kfree(attr_group);
kfree(pmu->events); imc_free_events(pmu->events, ct);
return -ENOMEM; return -ENOMEM;
} }
attr_group->name = "events"; attr_group->name = "events";
attr_group->attrs = attrs; attr_group->attrs = attrs;
do { do {
ev_val_str = kasprintf(GFP_KERNEL, "event=0x%x", pmu->events[i]->value); ev_val_str = kasprintf(GFP_KERNEL, "event=0x%x", pmu->events[i].value);
dev_str = device_str_attr_create(pmu->events[i]->name, ev_val_str); dev_str = device_str_attr_create(pmu->events[i].name, ev_val_str);
if (!dev_str) if (!dev_str)
continue; continue;
attrs[j++] = dev_str; attrs[j++] = dev_str;
if (pmu->events[i]->scale) { if (pmu->events[i].scale) {
ev_scale_str = kasprintf(GFP_KERNEL, "%s.scale",pmu->events[i]->name); ev_scale_str = kasprintf(GFP_KERNEL, "%s.scale", pmu->events[i].name);
dev_str = device_str_attr_create(ev_scale_str, pmu->events[i]->scale); dev_str = device_str_attr_create(ev_scale_str, pmu->events[i].scale);
if (!dev_str) if (!dev_str)
continue; continue;
attrs[j++] = dev_str; attrs[j++] = dev_str;
} }
if (pmu->events[i]->unit) { if (pmu->events[i].unit) {
ev_unit_str = kasprintf(GFP_KERNEL, "%s.unit",pmu->events[i]->name); ev_unit_str = kasprintf(GFP_KERNEL, "%s.unit", pmu->events[i].name);
dev_str = device_str_attr_create(ev_unit_str, pmu->events[i]->unit); dev_str = device_str_attr_create(ev_unit_str, pmu->events[i].unit);
if (!dev_str) if (!dev_str)
continue; continue;
...@@ -272,7 +287,6 @@ static int update_events_in_group(struct device_node *node, struct imc_pmu *pmu) ...@@ -272,7 +287,6 @@ static int update_events_in_group(struct device_node *node, struct imc_pmu *pmu)
/* Save the event attribute */ /* Save the event attribute */
pmu->attr_groups[IMC_EVENT_ATTR] = attr_group; pmu->attr_groups[IMC_EVENT_ATTR] = attr_group;
kfree(pmu->events);
return 0; return 0;
} }
......
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