/*
|
* Copyright (c) 2016 Facebook, Inc.
|
*
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
* you may not use this file except in compliance with the License.
|
* You may obtain a copy of the License at
|
*
|
* http://www.apache.org/licenses/LICENSE-2.0
|
*
|
* Unless required by applicable law or agreed to in writing, software
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
* See the License for the specific language governing permissions and
|
* limitations under the License.
|
*/
|
#include <ctype.h>
|
#include <errno.h>
|
#include <limits.h>
|
#include <stdio.h>
|
#include <stdlib.h>
|
#include <string.h>
|
|
#include "bcc_perf_map.h"
|
|
bool bcc_is_perf_map(const char *path) {
|
char* pos = strstr(path, ".map");
|
// Path ends with ".map"
|
return (pos != NULL) && (*(pos + 4)== 0);
|
}
|
|
bool bcc_is_valid_perf_map(const char *path) {
|
return bcc_is_perf_map(path) && (access(path, R_OK) == 0);
|
}
|
|
int bcc_perf_map_nstgid(int pid) {
|
char status_path[64];
|
FILE *status;
|
|
snprintf(status_path, sizeof(status_path), "/proc/%d/status", pid);
|
status = fopen(status_path, "r");
|
|
if (!status)
|
return -1;
|
|
// return the original PID if we fail to work out the TGID
|
int nstgid = pid;
|
|
size_t size = 0;
|
char *line = NULL;
|
while (getline(&line, &size, status) != -1) {
|
// check Tgid line first in case CONFIG_PID_NS is off
|
if (strstr(line, "Tgid:") != NULL)
|
nstgid = (int)strtol(strrchr(line, '\t'), NULL, 10);
|
if (strstr(line, "NStgid:") != NULL)
|
// PID namespaces can be nested -- last number is innermost PID
|
nstgid = (int)strtol(strrchr(line, '\t'), NULL, 10);
|
}
|
free(line);
|
fclose(status);
|
|
return nstgid;
|
}
|
|
bool bcc_perf_map_path(char *map_path, size_t map_len, int pid) {
|
char source[64];
|
snprintf(source, sizeof(source), "/proc/%d/root", pid);
|
|
char target[4096];
|
ssize_t target_len = readlink(source, target, sizeof(target) - 1);
|
if (target_len == -1)
|
return false;
|
|
target[target_len] = '\0';
|
if (strcmp(target, "/") == 0)
|
target[0] = '\0';
|
|
int nstgid = bcc_perf_map_nstgid(pid);
|
|
snprintf(map_path, map_len, "%s/tmp/perf-%d.map", target, nstgid);
|
return true;
|
}
|
|
int bcc_perf_map_foreach_sym(const char *path, bcc_perf_map_symcb callback,
|
void* payload) {
|
FILE* file = fopen(path, "r");
|
if (!file)
|
return -1;
|
|
char *line = NULL;
|
size_t size = 0;
|
long long begin, len;
|
while (getline(&line, &size, file) != -1) {
|
char *cursor = line;
|
char *newline, *sep;
|
|
begin = strtoull(cursor, &sep, 16);
|
if (begin == 0 || *sep != ' ' || (begin == ULLONG_MAX && errno == ERANGE))
|
continue;
|
cursor = sep;
|
while (*cursor && isspace(*cursor)) cursor++;
|
|
len = strtoull(cursor, &sep, 16);
|
if (*sep != ' ' ||
|
(sep == cursor && len == 0) ||
|
(len == ULLONG_MAX && errno == ERANGE))
|
continue;
|
cursor = sep;
|
while (*cursor && isspace(*cursor)) cursor++;
|
|
newline = strchr(cursor, '\n');
|
if (newline)
|
newline[0] = '\0';
|
|
callback(cursor, begin, len, payload);
|
}
|
|
free(line);
|
fclose(file);
|
|
return 0;
|
}
|