diff options
Diffstat (limited to 'arch/powerpc/perf')
| -rw-r--r-- | arch/powerpc/perf/core-book3s.c | 12 | ||||
| -rw-r--r-- | arch/powerpc/perf/imc-pmu.c | 17 | 
2 files changed, 24 insertions, 5 deletions
| diff --git a/arch/powerpc/perf/core-book3s.c b/arch/powerpc/perf/core-book3s.c index 9e3da168d54c..fce545774d50 100644 --- a/arch/powerpc/perf/core-book3s.c +++ b/arch/powerpc/perf/core-book3s.c @@ -410,8 +410,12 @@ static __u64 power_pmu_bhrb_to(u64 addr)  	int ret;  	__u64 target; -	if (is_kernel_addr(addr)) -		return branch_target((unsigned int *)addr); +	if (is_kernel_addr(addr)) { +		if (probe_kernel_read(&instr, (void *)addr, sizeof(instr))) +			return 0; + +		return branch_target(&instr); +	}  	/* Userspace: need copy instruction here then translate it */  	pagefault_disable(); @@ -1415,7 +1419,7 @@ static int collect_events(struct perf_event *group, int max_count,  	int n = 0;  	struct perf_event *event; -	if (!is_software_event(group)) { +	if (group->pmu->task_ctx_nr == perf_hw_context) {  		if (n >= max_count)  			return -1;  		ctrs[n] = group; @@ -1423,7 +1427,7 @@ static int collect_events(struct perf_event *group, int max_count,  		events[n++] = group->hw.config;  	}  	list_for_each_entry(event, &group->sibling_list, group_entry) { -		if (!is_software_event(event) && +		if (event->pmu->task_ctx_nr == perf_hw_context &&  		    event->state != PERF_EVENT_STATE_OFF) {  			if (n >= max_count)  				return -1; diff --git a/arch/powerpc/perf/imc-pmu.c b/arch/powerpc/perf/imc-pmu.c index 0ead3cd73caa..be4e7f84f70a 100644 --- a/arch/powerpc/perf/imc-pmu.c +++ b/arch/powerpc/perf/imc-pmu.c @@ -310,6 +310,19 @@ static int ppc_nest_imc_cpu_offline(unsigned int cpu)  		return 0;  	/* +	 * Check whether nest_imc is registered. We could end up here if the +	 * cpuhotplug callback registration fails. i.e, callback invokes the +	 * offline path for all successfully registered nodes. At this stage, +	 * nest_imc pmu will not be registered and we should return here. +	 * +	 * We return with a zero since this is not an offline failure. And +	 * cpuhp_setup_state() returns the actual failure reason to the caller, +	 * which in turn will call the cleanup routine. +	 */ +	if (!nest_pmus) +		return 0; + +	/*  	 * Now that this cpu is one of the designated,  	 * find a next cpu a) which is online and b) in same chip.  	 */ @@ -1171,6 +1184,7 @@ static void imc_common_cpuhp_mem_free(struct imc_pmu *pmu_ptr)  		if (nest_pmus == 1) {  			cpuhp_remove_state(CPUHP_AP_PERF_POWERPC_NEST_IMC_ONLINE);  			kfree(nest_imc_refc); +			kfree(per_nest_pmu_arr);  		}  		if (nest_pmus > 0) @@ -1195,7 +1209,6 @@ static void imc_common_cpuhp_mem_free(struct imc_pmu *pmu_ptr)  		kfree(pmu_ptr->attr_groups[IMC_EVENT_ATTR]->attrs);  	kfree(pmu_ptr->attr_groups[IMC_EVENT_ATTR]);  	kfree(pmu_ptr); -	kfree(per_nest_pmu_arr);  	return;  } @@ -1309,6 +1322,8 @@ int init_imc_pmu(struct device_node *parent, struct imc_pmu *pmu_ptr, int pmu_id  			ret = nest_pmu_cpumask_init();  			if (ret) {  				mutex_unlock(&nest_init_lock); +				kfree(nest_imc_refc); +				kfree(per_nest_pmu_arr);  				goto err_free;  			}  		} |