Anforderungen  |   Konzepte  |   Entwurf  |   Entwicklung  |   Qualitätssicherung  |   Lebenszyklus  |   Steuerung
 
 
 
 


Quelle  record.c   Sprache: C

 
// SPDX-License-Identifier: GPL-2.0
#include "debug.h"
#include "evlist.h"
#include "evsel.h"
#include "evsel_config.h"
#include "parse-events.h"
#include <errno.h>
#include <limits.h>
#include <stdlib.h>
#include <api/fs/fs.h>
#include <subcmd/parse-options.h>
#include <perf/cpumap.h>
#include "cloexec.h"
#include "util/perf_api_probe.h"
#include "record.h"
#include "../perf-sys.h"
#include "topdown.h"
#include "map_symbol.h"
#include "mem-events.h"

/*
 * evsel__config_leader_sampling() uses special rules for leader sampling.
 * However, if the leader is an AUX area event, then assume the event to sample
 * is the next event.
 */

static struct evsel *evsel__read_sampler(struct evsel *evsel, struct evlist *evlist)
{
 struct evsel *leader = evsel__leader(evsel);

 if (evsel__is_aux_event(leader) || arch_topdown_sample_read(leader) ||
     is_mem_loads_aux_event(leader)) {
  evlist__for_each_entry(evlist, evsel) {
   if (evsel__leader(evsel) == leader && evsel != evsel__leader(evsel))
    return evsel;
  }
 }

 return leader;
}

static u64 evsel__config_term_mask(struct evsel *evsel)
{
 struct evsel_config_term *term;
 struct list_head *config_terms = &evsel->config_terms;
 u64 term_types = 0;

 list_for_each_entry(term, config_terms, list) {
  term_types |= 1 << term->type;
 }
 return term_types;
}

static void evsel__config_leader_sampling(struct evsel *evsel, struct evlist *evlist)
{
 struct perf_event_attr *attr = &evsel->core.attr;
 struct evsel *leader = evsel__leader(evsel);
 struct evsel *read_sampler;
 u64 term_types, freq_mask;

 if (!leader->sample_read)
  return;

 read_sampler = evsel__read_sampler(evsel, evlist);

 if (evsel == read_sampler)
  return;

 term_types = evsel__config_term_mask(evsel);
 /*
 * Disable sampling for all group members except those with explicit
 * config terms or the leader. In the case of an AUX area event, the 2nd
 * event in the group is the one that 'leads' the sampling.
 */

 freq_mask = (1 << EVSEL__CONFIG_TERM_FREQ) | (1 << EVSEL__CONFIG_TERM_PERIOD);
 if ((term_types & freq_mask) == 0) {
  attr->freq           = 0;
  attr->sample_freq    = 0;
  attr->sample_period  = 0;
 }
 if ((term_types & (1 << EVSEL__CONFIG_TERM_OVERWRITE)) == 0)
  attr->write_backward = 0;

 /*
 * We don't get a sample for slave events, we make them when delivering
 * the group leader sample. Set the slave event to follow the master
 * sample_type to ease up reporting.
 * An AUX area event also has sample_type requirements, so also include
 * the sample type bits from the leader's sample_type to cover that
 * case.
 */

 attr->sample_type = read_sampler->core.attr.sample_type |
       leader->core.attr.sample_type;
}

void evlist__config(struct evlist *evlist, struct record_opts *opts, struct callchain_param *callchain)
{
 struct evsel *evsel;
 bool use_sample_identifier = false;
 bool use_comm_exec;
 bool sample_id = opts->sample_id;

 if (perf_cpu_map__cpu(evlist->core.user_requested_cpus, 0).cpu < 0)
  opts->no_inherit = true;

 use_comm_exec = perf_can_comm_exec();

 evlist__for_each_entry(evlist, evsel) {
  evsel__config(evsel, opts, callchain);
  if (evsel->tracking && use_comm_exec)
   evsel->core.attr.comm_exec = 1;
 }

 /* Configure leader sampling here now that the sample type is known */
 evlist__for_each_entry(evlist, evsel)
  evsel__config_leader_sampling(evsel, evlist);

 if (opts->full_auxtrace || opts->sample_identifier) {
  /*
 * Need to be able to synthesize and parse selected events with
 * arbitrary sample types, which requires always being able to
 * match the id.
 */

  use_sample_identifier = perf_can_sample_identifier();
  sample_id = true;
 } else if (evlist->core.nr_entries > 1) {
  struct evsel *first = evlist__first(evlist);

  evlist__for_each_entry(evlist, evsel) {
   if (evsel->core.attr.sample_type == first->core.attr.sample_type)
    continue;
   use_sample_identifier = perf_can_sample_identifier();
   break;
  }
  sample_id = true;
 }

 if (sample_id) {
  evlist__for_each_entry(evlist, evsel)
   evsel__set_sample_id(evsel, use_sample_identifier);
 }

 evlist__set_id_pos(evlist);
}

static int get_max_rate(unsigned int *rate)
{
 return sysctl__read_int("kernel/perf_event_max_sample_rate", (int *)rate);
}

static int record_opts__config_freq(struct record_opts *opts)
{
 bool user_freq = opts->user_freq != UINT_MAX;
 bool user_interval = opts->user_interval != ULLONG_MAX;
 unsigned int max_rate;

 if (user_interval && user_freq) {
  pr_err("cannot set frequency and period at the same time\n");
  return -1;
 }

 if (user_interval)
  opts->default_interval = opts->user_interval;
 if (user_freq)
  opts->freq = opts->user_freq;

 /*
 * User specified count overrides default frequency.
 */

 if (opts->default_interval)
  opts->freq = 0;
 else if (opts->freq) {
  opts->default_interval = opts->freq;
 } else {
  pr_err("frequency and count are zero, aborting\n");
  return -1;
 }

 if (get_max_rate(&max_rate))
  return 0;

 /*
 * User specified frequency is over current maximum.
 */

 if (user_freq && (max_rate < opts->freq)) {
  if (opts->strict_freq) {
   pr_err("error: Maximum frequency rate (%'u Hz) exceeded.\n"
          " Please use -F freq option with a lower value or consider\n"
          " tweaking /proc/sys/kernel/perf_event_max_sample_rate.\n",
          max_rate);
   return -1;
  } else {
   pr_warning("warning: Maximum frequency rate (%'u Hz) exceeded, throttling from %'u Hz to %'u Hz.\n"
       " The limit can be raised via /proc/sys/kernel/perf_event_max_sample_rate.\n"
       " The kernel will lower it when perf's interrupts take too long.\n"
       " Use --strict-freq to disable this throttling, refusing to record.\n",
       max_rate, opts->freq, max_rate);

   opts->freq = max_rate;
  }
 }

 /*
 * Default frequency is over current maximum.
 */

 if (max_rate < opts->freq) {
  pr_warning("Lowering default frequency rate from %u to %u.\n"
      "Please consider tweaking "
      "/proc/sys/kernel/perf_event_max_sample_rate.\n",
      opts->freq, max_rate);
  opts->freq = max_rate;
 }

 return 0;
}

int record_opts__config(struct record_opts *opts)
{
 return record_opts__config_freq(opts);
}

bool evlist__can_select_event(struct evlist *evlist, const char *str)
{
 struct evlist *temp_evlist;
 struct evsel *evsel;
 int err, fd;
 struct perf_cpu cpu = { .cpu = 0 };
 bool ret = false;
 pid_t pid = -1;

 temp_evlist = evlist__new();
 if (!temp_evlist)
  return false;

 err = parse_event(temp_evlist, str);
 if (err)
  goto out_delete;

 evsel = evlist__last(temp_evlist);

 if (!evlist || perf_cpu_map__is_any_cpu_or_is_empty(evlist->core.user_requested_cpus)) {
  struct perf_cpu_map *cpus = perf_cpu_map__new_online_cpus();

  if (cpus)
   cpu =  perf_cpu_map__cpu(cpus, 0);

  perf_cpu_map__put(cpus);
 } else {
  cpu = perf_cpu_map__cpu(evlist->core.user_requested_cpus, 0);
 }

 while (1) {
  fd = sys_perf_event_open(&evsel->core.attr, pid, cpu.cpu, -1,
      perf_event_open_cloexec_flag());
  if (fd < 0) {
   if (pid == -1 && errno == EACCES) {
    pid = 0;
    continue;
   }
   goto out_delete;
  }
  break;
 }
 close(fd);
 ret = true;

out_delete:
 evlist__delete(temp_evlist);
 return ret;
}

int record__parse_freq(const struct option *opt, const char *str, int unset __maybe_unused)
{
 unsigned int freq;
 struct record_opts *opts = opt->value;

 if (!str)
  return -EINVAL;

 if (strcasecmp(str, "max") == 0) {
  if (get_max_rate(&freq)) {
   pr_err("couldn't read /proc/sys/kernel/perf_event_max_sample_rate\n");
   return -1;
  }
  pr_info("info: Using a maximum frequency rate of %'d Hz\n", freq);
 } else {
  freq = atoi(str);
 }

 opts->user_freq = freq;
 return 0;
}

Messung V0.5
C=95 H=95 G=94

¤ Dauer der Verarbeitung: 0.10 Sekunden  (vorverarbeitet)  ¤

*© Formatika GbR, Deutschland






Wurzel

Suchen

Beweissystem der NASA

Beweissystem Isabelle

NIST Cobol Testsuite

Cephes Mathematical Library

Wiener Entwicklungsmethode

Haftungshinweis

Die Informationen auf dieser Webseite wurden nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit, noch Qualität der bereit gestellten Informationen zugesichert.

Bemerkung:

Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.






                                                                                                                                                                                                                                                                                                                                                                                                     


Neuigkeiten

     Aktuelles
     Motto des Tages

Software

     Produkte
     Quellcodebibliothek

Aktivitäten

     Artikel über Sicherheit
     Anleitung zur Aktivierung von SSL

Muße

     Gedichte
     Musik
     Bilder

Jenseits des Üblichen ....
    

Besucherstatistik

Besucherstatistik

Monitoring

Montastic status badge