/* * pfmlib_sparc.c : support for SPARC processors * * Copyright (c) 2011 Google, Inc * Contributed by Stephane Eranian * * Permission is hereby granted, free of charge, to any person obtaining a * copy of this software and associated documentation files (the "Software"), * to deal in the Software without restriction, including without limitation * the rights to use, copy, modify, merge, publish, distribute, sublicense, * and/or sell copies of the Software, and to permit persons to whom the * Software is furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS * IN THE SOFTWARE. * */ #include #include #include #include #include /* private headers */ #include "pfmlib_priv.h" /* library private */ #include "pfmlib_sparc_priv.h" const pfmlib_attr_desc_t sparc_mods[]={ PFM_ATTR_B("k", "monitor at priv level 0"), /* monitor priv level 0 */ PFM_ATTR_B("u", "monitor at priv level 1, 2, 3"), /* monitor priv level 1, 2, 3 */ PFM_ATTR_B("h", "monitor in hypervisor"), /* monitor in hypervisor*/ PFM_ATTR_NULL /* end-marker to avoid exporting number of entries */ }; #define SPARC_NUM_MODS (sizeof(sparc_mods)/sizeof(pfmlib_attr_desc_t) - 1) #ifdef CONFIG_PFMLIB_OS_LINUX /* * helper function to retrieve one value from /proc/cpuinfo * for internal libpfm use only * attr: the attribute (line) to look for * ret_buf: a buffer to store the value of the attribute (as a string) * maxlen : number of bytes of capacity in ret_buf * * ret_buf is null terminated. * * Return: * 0 : attribute found, ret_buf populated * -1: attribute not found */ static int pfmlib_getcpuinfo_attr(const char *attr, char *ret_buf, size_t maxlen) { FILE *fp = NULL; int ret = -1; size_t attr_len, buf_len = 0; char *p, *value = NULL; char *buffer = NULL; if (attr == NULL || ret_buf == NULL || maxlen < 1) return -1; attr_len = strlen(attr); fp = fopen("/proc/cpuinfo", "r"); if (fp == NULL) return -1; while(pfmlib_getl(&buffer, &buf_len, fp) != -1){ /* skip blank lines */ if (*buffer == '\n') continue; p = strchr(buffer, ':'); if (p == NULL) goto error; /* * p+2: +1 = space, +2= firt character * strlen()-1 gets rid of \n */ *p = '\0'; value = p+2; value[strlen(value)-1] = '\0'; if (!strncmp(attr, buffer, attr_len)) break; } strncpy(ret_buf, value, maxlen-1); ret_buf[maxlen-1] = '\0'; ret = 0; error: free(buffer); fclose(fp); return ret; } #else static int pfmlib_getcpuinfo_attr(const char *attr, char *ret_buf, size_t maxlen) { return -1; } #endif static pfm_pmu_t pmu_name_to_pmu_type(char *name) { if (!strcmp(name, "ultra12")) return PFM_PMU_SPARC_ULTRA12; if (!strcmp(name, "ultra3")) return PFM_PMU_SPARC_ULTRA3; if (!strcmp(name, "ultra3i")) return PFM_PMU_SPARC_ULTRA3I; if (!strcmp(name, "ultra3+")) return PFM_PMU_SPARC_ULTRA3PLUS; if (!strcmp(name, "ultra4+")) return PFM_PMU_SPARC_ULTRA4PLUS; if (!strcmp(name, "niagara2")) return PFM_PMU_SPARC_NIAGARA2; if (!strcmp(name, "niagara")) return PFM_PMU_SPARC_NIAGARA1; return PFM_PMU_NONE; } int pfm_sparc_detect(void *this) { pfmlib_pmu_t *pmu = this; pfm_pmu_t model; int ret; char buffer[32]; ret = pfmlib_getcpuinfo_attr("pmu", buffer, sizeof(buffer)); if (ret == -1) return PFM_ERR_NOTSUPP; model = pmu_name_to_pmu_type(buffer); return model == pmu->pmu ? PFM_SUCCESS : PFM_ERR_NOTSUPP; } void pfm_sparc_display_reg(void *this, pfmlib_event_desc_t *e, pfm_sparc_reg_t reg) { __pfm_vbprintf("[0x%x umask=0x%x code=0x%x ctrl_s1=%d ctrl_s0=%d] %s\n", reg.val, reg.config.umask, reg.config.code, reg.config.ctrl_s1, reg.config.ctrl_s0, e->fstr); } int pfm_sparc_get_encoding(void *this, pfmlib_event_desc_t *e) { const sparc_entry_t *pe = this_pe(this); pfmlib_event_attr_info_t *a; pfm_sparc_reg_t reg; int i; //reg.val = pe[e->event].code << 16 | pe[e->event].ctrl; reg.val = pe[e->event].code; for (i = 0; i < e->nattrs; i++) { a = attr(e, i); if (a->ctrl != PFM_ATTR_CTRL_PMU) continue; if (a->type == PFM_ATTR_UMASK) reg.config.umask |= 1 << pe[e->event].umasks[a->idx].ubit; } e->count = 2; e->codes[0] = reg.val; e->codes[1] = pe[e->event].ctrl; evt_strcat(e->fstr, "%s", pe[e->event].name); pfmlib_sort_attr(e); for (i = 0; i < e->nattrs; i++) { a = attr(e, i); if (a->ctrl != PFM_ATTR_CTRL_PMU) continue; if (a->type == PFM_ATTR_UMASK) evt_strcat(e->fstr, ":%s", pe[e->event].umasks[a->idx].uname); } pfm_sparc_display_reg(this, e, reg); return PFM_SUCCESS; } int pfm_sparc_get_event_first(void *this) { return 0; } int pfm_sparc_get_event_next(void *this, int idx) { pfmlib_pmu_t *p = this; if (idx >= (p->pme_count-1)) return -1; return idx+1; } int pfm_sparc_event_is_valid(void *this, int pidx) { pfmlib_pmu_t *p = this; return pidx >= 0 && pidx < p->pme_count; } int pfm_sparc_validate_table(void *this, FILE *fp) { pfmlib_pmu_t *pmu = this; const sparc_entry_t *pe = this_pe(this); int i, j, error = 0; for(i=0; i < pmu->pme_count; i++) { if (!pe[i].name) { fprintf(fp, "pmu: %s event%d: :: no name (prev event was %s)\n", pmu->name, i, i > 1 ? pe[i-1].name : "??"); error++; } if (!pe[i].desc) { fprintf(fp, "pmu: %s event%d: %s :: no description\n", pmu->name, i, pe[i].name); error++; } for(j=i+1; j < pmu->pme_count; j++) { if (pe[i].code == pe[j].code && pe[i].ctrl == pe[j].ctrl) { fprintf(fp, "pmu: %s event%d: %s code: 0x%x is duplicated in event%d : %s\n", pmu->name, i, pe[i].name, pe[i].code, j, pe[j].name); error++; } } } return error ? PFM_ERR_INVAL : PFM_SUCCESS; } int pfm_sparc_get_event_attr_info(void *this, int pidx, int attr_idx, pfmlib_event_attr_info_t *info) { const sparc_entry_t *pe = this_pe(this); int idx; if (attr_idx < pe[pidx].numasks) { info->name = pe[pidx].umasks[attr_idx].uname; info->desc = pe[pidx].umasks[attr_idx].udesc; info->name = pe[pidx].umasks[attr_idx].uname; info->equiv= NULL; info->code = 1 << pe[pidx].umasks[attr_idx].ubit; info->type = PFM_ATTR_UMASK; info->idx = attr_idx; } else { /* * all mods implemented by ALL events */ idx = attr_idx - pe[pidx].numasks; info->name = sparc_mods[idx].name; info->desc = sparc_mods[idx].desc; info->type = sparc_mods[idx].type; info->code = idx; info->type = sparc_mods[idx].type; } info->is_dfl = 0; info->is_precise = 0; info->ctrl = PFM_ATTR_CTRL_PMU;; return PFM_SUCCESS; } int pfm_sparc_get_event_info(void *this, int idx, pfm_event_info_t *info) { pfmlib_pmu_t *pmu = this; const sparc_entry_t *pe = this_pe(this); /* * pmu and idx filled out by caller */ info->name = pe[idx].name; info->desc = pe[idx].desc; info->code = pe[idx].code; info->equiv = NULL; info->idx = idx; /* private index */ info->pmu = pmu->pmu; info->is_precise = 0; info->nattrs = pe[idx].numasks; return PFM_SUCCESS; } unsigned int pfm_sparc_get_event_nattrs(void *this, int pidx) { const sparc_entry_t *pe = this_pe(this); return SPARC_NUM_MODS + pe[pidx].numasks; }