29aca0e1创建于 4月20日历史提交
/****************************************************************************
 * apps/system/perf-tools/parse-options.c
 *
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.  The
 * ASF licenses this file to you 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.
 *
 ****************************************************************************/

/****************************************************************************
 * Included Files
 ****************************************************************************/

#include <ctype.h>
#include <getopt.h>
#include <stdio.h>

#include <nuttx/sched.h>

#include "builtin.h"
#include "parse-options.h"

/****************************************************************************
 * Public Functions
 ****************************************************************************/

int check_and_atoi(char *arg)
{
  int i;

  for (i = 0; arg[i] != '\0'; i++)
    {
      if (!isdigit(arg[i]))
        {
          return -EINVAL;
        }
    }

  return atoi(arg);
}

int parse_stat_options(int argc, FAR char **argv,
                       FAR struct stat_args_s *stat_args)
{
  int opt;
  int ret;

  stat_args->pid = -1;
  stat_args->cpu = -1;
  stat_args->cmd_nr = argc - 1;

  while ((opt = getopt(argc, argv, ":aC:p:e:c:m:g")) != -1)
    {
      switch (opt)
        {
          case 'a':
            if (stat_args->type == STAT_ARGS_NONE ||
                stat_args->type == STAT_ARGS_CPU)
              {
                stat_args->type = STAT_ARGS_ALL;
                stat_args->cpu = -1;
                stat_args->pid = -1;
                stat_args->events = NULL;
              }

            stat_args->cmd_nr -= 1;
            break;
          case 'C':
            if (stat_args->type == STAT_ARGS_NONE)
              {
                stat_args->type = STAT_ARGS_CPU;
                stat_args->cpu = check_and_atoi(optarg);
                stat_args->pid = -1;
                stat_args->events = NULL;
              }
            else if (stat_args->type == STAT_ARGS_EVENT)
              {
                stat_args->cpu = check_and_atoi(optarg);
                stat_args->pid = -1;
              }
            else if (stat_args->type == STAT_ARGS_TASK)
              {
                printf("PID/TID switch overriding CPU\n");
                stat_args->cpu = check_and_atoi(optarg);
              }

            stat_args->cmd_nr -= 2;
            break;
          case 'p':
            if (stat_args->type == STAT_ARGS_NONE)
              {
                stat_args->type = STAT_ARGS_TASK;
                stat_args->pid = check_and_atoi(optarg);
                stat_args->cpu = -1;
                stat_args->events = NULL;
              }
            else if (stat_args->type == STAT_ARGS_EVENT)
              {
                stat_args->pid = check_and_atoi(optarg);
                stat_args->cpu = -1;
              }
            else if (stat_args->type == STAT_ARGS_CPU)
              {
                printf("PID/TID switch overriding CPU\n");
                stat_args->type = STAT_ARGS_TASK;
                stat_args->pid = check_and_atoi(optarg);
              }

            stat_args->cmd_nr -= 2;
            break;
          case 'e':
            if (stat_args->type == STAT_ARGS_NONE ||
                stat_args->type == STAT_ARGS_ALL)
              {
                stat_args->cpu = -1;
                stat_args->pid = -1;
              }
            else if (stat_args->type == STAT_ARGS_CPU)
              {
                stat_args->pid = -1;
              }

            stat_args->type = STAT_ARGS_EVENT;
            stat_args->events = optarg;
            stat_args->cmd_nr -= 2;
            break;
          case 'h':
            stat_args->type = STAT_ARGS_HELP;
            break;
          case 'c':
            stat_args->sample_period = check_and_atoi(optarg);
            stat_args->cmd_nr -= 2;
            break;
          case 'm':
            stat_args->buffer_size = check_and_atoi(optarg);
            stat_args->cmd_nr -= 2;
            break;
          case 'g':
            stat_args->call_chain = true;
            stat_args->cmd_nr -= 1;
            break;
          default:
            return -EINVAL;
        }
    }

  stat_args->sec = 0;
  stat_args->cmd = argv[optind];
  stat_args->cmd_args = &argv[optind];

  if (argc == 1)
    {
      stat_args->type = STAT_ARGS_ALL;
      stat_args->cpu = -1;
      stat_args->pid = -1;
      stat_args->events = NULL;
      stat_args->cmd_nr -= 1;
    }
  else if (stat_args->cmd_nr > 0)
    {
      if (stat_args->cmd && !strcmp(stat_args->cmd, "sleep"))
        {
          stat_args->sec = PERF_DEFAULT_RUN_TIME;
          if (stat_args->type == STAT_ARGS_NONE)
            {
              stat_args->cpu = -1;
              stat_args->pid = -1;
              stat_args->events = NULL;
              stat_args->cmd_nr -= 1;
            }

          if (stat_args->cmd_nr == 2)
            {
              stat_args->sec = check_and_atoi(stat_args->cmd_args[1]);
              if (stat_args->sec < 0)
                {
                  return -EINVAL;
                }
            }
          else
            {
              printf("sleep: invalid time interval\n");
              return -EINVAL;
            }

          stat_args->cmd = NULL;
          stat_args->cmd_nr = 0;
        }
      else
        {
          if (stat_args->type == STAT_ARGS_NONE)
            {
              stat_args->type = STAT_ARGS_FORK;
              stat_args->cpu = -1;
              stat_args->pid = _SCHED_GETPID();
              stat_args->events = NULL;
              stat_args->cmd_nr -= 1;
            }
          else if (stat_args->type == STAT_ARGS_CPU ||
                   stat_args->type == STAT_ARGS_EVENT)
            {
              stat_args->type = STAT_ARGS_FORK;
              stat_args->pid = _SCHED_GETPID();
            }
          else if (stat_args->type == STAT_ARGS_TASK)
            {
              stat_args->cmd = NULL;
              stat_args->cmd_nr = 0;
            }
        }
    }

  if (stat_args->cpu >= CONFIG_SMP_NCPUS || stat_args->cpu < -1)
    {
      printf(" Perf can support %d CPUs.\n", CONFIG_SMP_NCPUS);
      return -EINVAL;
    }

  if (stat_args->pid > 0)
    {
      ret = kill(stat_args->pid, 0);
      if (ret == -1 && errno == ESRCH)
        {
          printf("The specified process does not exist\n");
          return -EINVAL;
        }
      else if (ret != 0)
        {
          printf("Error trying to determine whether the process exists\n");
        }
    }
  else if (stat_args->pid != -1)
    {
      return -EINVAL;
    }

  return 0;
}