如何通过QNXCpp代码中的进程名称将SIGTERM发送到进程

How to send SIGTERM to a process by process name in QNX cpp code?

本文关键字:SIGTERM 进程 进程名 QNXCpp 何通过 代码      更新时间:2023-10-16

E.g:我有一个名为foo的进程。

通常在控制台中,我可以键入slay foo,然后foo终止。此外,在cpp代码中,我可以使用system("slay foo"); 发出系统调用

我知道system((是一个很重的fork调用,应该避免。<csignal><cstdlib>中还有其他功能我可以选择吗?

我读过SignalKill((和SignalKill_r((,两者都需要pid,但我无法提供。

这并不像人们想象的那么简单。Linux没有提供一个syscall,它通过进程的名称为您提供进程的PID

假设QNX的文件系统类似于标准UNI,您可能需要阅读本文来了解如何使用进程名称查找进程的PID,然后将该PIDSignalKillSignalKill_r 一起使用

以下是使用C中的名称查找进程的PID的代码。我不能在QNX上测试它,但它可以在Ubuntu上工作。

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <ctype.h>
#include <dirent.h>
#include <libgen.h>
/* checks if the string is purely an integer
 * we can do it with `strtol' also
 */
int check_if_number (char *str)
{
  int i;
  for (i=0; str[i] != ''; i++)
  {
    if (!isdigit (str[i]))
    {
      return 0;
    }
  }
  return 1;
}
#define MAX_BUF 1024
#define PID_LIST_BLOCK 32
int *pidof (char *pname)
{
  DIR *dirp;
  FILE *fp;
  struct dirent *entry;
  int *pidlist, pidlist_index = 0, pidlist_realloc_count = 1;
  char path[MAX_BUF], read_buf[MAX_BUF];
  dirp = opendir ("/proc/");
  if (dirp == NULL)
  {
    perror ("Fail");
    return NULL;
  }
  pidlist = malloc (sizeof (int) * PID_LIST_BLOCK);
  if (pidlist == NULL)
  {
    return NULL;
  }
  while ((entry = readdir (dirp)) != NULL)
  {
    if (check_if_number (entry->d_name))
    {
      strcpy (path, "/proc/");
      strcat (path, entry->d_name);
      strcat (path, "/comm");
      /* A file may not exist, it may have been removed.
       * dut to termination of the process. Actually we need to
       * make sure the error is actually file does not exist to
       * be accurate.
       */
      fp = fopen (path, "r");
      if (fp != NULL)
      {
        fscanf (fp, "%s", read_buf);
        if (strcmp (read_buf, pname) == 0)
        {
          /* add to list and expand list if needed */
          pidlist[pidlist_index++] = atoi (entry->d_name);
          if (pidlist_index == PID_LIST_BLOCK * pidlist_realloc_count)
          {
            pidlist_realloc_count++;
            pidlist = realloc (pidlist, sizeof (int) * PID_LIST_BLOCK * pidlist_realloc_count); //Error check todo
            if (pidlist == NULL)
            {
              return NULL;
            }
          }
        }
        fclose (fp);
      }
    }
  }

  closedir (dirp);
  pidlist[pidlist_index] = -1; /* indicates end of list */
  return pidlist;
}
int main (int argc, char *argv[])
{
  int *list, i;
  if (argc != 2)
  {
    printf ("Usage: %s proc_namen", argv[0]);
    return 0;
  }
  list = pidof (argv[1]);
  for (i=0; list[i] != -1; i++)
  {
    printf ("%d ", list[i]);
  }
  free (list);
  if (list[0] != -1)
  {
    printf ("n");
  }
  return 0;
}

为了将进程名称转换为pid,您需要挖掘QNX的/proc文件系统。我写了一本名为《QNX食谱》的书,现在可以在QNX的网站上免费获得(http://www.qnx.com/download/feature.html?programid=26184)。转到第222页"遍历进程列表",复制遍历进程列表的代码。这将允许您在所有进程中搜索要杀死的进程(它将为您提供所需的PID(。

void
iterate_processes (void)
{
  struct dirent   *dirent;
  DIR             *dir;
  int             r;
  int             pid;
  // 1) find all processes
  if (!(dir = opendir ("/proc"))) {
    fprintf (stderr, "%s:  couldn't open /proc, errno %dn",
             progname, errno);
    perror (NULL);
    exit (EXIT_FAILURE);
  }
  while (dirent = readdir (dir)) {
    // 2) we are only interested in process IDs
    if (isdigit (*dirent -> d_name)) {
      pid = atoi (dirent -> d_name);
      iterate_process (pid);
    }
  }
  closedir (dir);
}
void
iterate_process (int pid)
{
  char      paths [PATH_MAX];
  int       fd;
  // 1) set up structure
  static struct {
    procfs_debuginfo    info;
    char                buff [PATH_MAX];
  } name;
  sprintf (paths, "/proc/%d/as", pid);
  if ((fd = open (paths, O_RDONLY)) == -1) {
    return;
  }
  // 2) ask for the name
  if (devctl (fd, DCMD_PROC_MAPDEBUG_BASE, &name,
              sizeof (name), 0) != EOK) {
    if (pid == 1) {
      strcpy (name.info.path, "(procnto)");
    } else {
      strcpy (name.info.path, "(n/a)");
    }
  }
  // 3) we can compare against name.info.path here...
  do_process (pid, fd, name.info.path);
  close (fd);
}

通过为"do_process(("提供您想要的任何操作,您可以例如通过名称等进行kill。