Quellcodebibliothek Statistik Leitseite products/sources/formale Sprachen/C/Linux/tools/virtio/virtio-trace/   (Open Source Betriebssystem Version 6.17.9©)  Datei vom 24.10.2025 mit Größe 4 kB image not shown  

Quelle  trace-agent-rw.c   Sprache: C

 
// SPDX-License-Identifier: GPL-2.0-only
/*
 * Read/write thread of a guest agent for virtio-trace
 *
 * Copyright (C) 2012 Hitachi, Ltd.
 * Created by Yoshihiro Yunomae <yoshihiro.yunomae.ez@hitachi.com>
 *            Masami Hiramatsu <masami.hiramatsu.pt@hitachi.com>
 */


#define _GNU_SOURCE
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/syscall.h>
#include "trace-agent.h"

#define READ_WAIT_USEC 100000

void *rw_thread_info_new(void)
{
 struct rw_thread_info *rw_ti;

 rw_ti = zalloc(sizeof(struct rw_thread_info));
 if (rw_ti == NULL) {
  pr_err("rw_thread_info zalloc error\n");
  exit(EXIT_FAILURE);
 }

 rw_ti->cpu_num = -1;
 rw_ti->in_fd = -1;
 rw_ti->out_fd = -1;
 rw_ti->read_pipe = -1;
 rw_ti->write_pipe = -1;
 rw_ti->pipe_size = PIPE_INIT;

 return rw_ti;
}

void *rw_thread_init(int cpu, const char *in_path, const char *out_path,
    bool stdout_flag, unsigned long pipe_size,
    struct rw_thread_info *rw_ti)
{
 int data_pipe[2];

 rw_ti->cpu_num = cpu;

 /* set read(input) fd */
 rw_ti->in_fd = open(in_path, O_RDONLY);
 if (rw_ti->in_fd == -1) {
  pr_err("Could not open in_fd (CPU:%d)\n", cpu);
  goto error;
 }

 /* set write(output) fd */
 if (!stdout_flag) {
  /* virtio-serial output mode */
  rw_ti->out_fd = open(out_path, O_WRONLY);
  if (rw_ti->out_fd == -1) {
   pr_err("Could not open out_fd (CPU:%d)\n", cpu);
   goto error;
  }
 } else
  /* stdout mode */
  rw_ti->out_fd = STDOUT_FILENO;

 if (pipe2(data_pipe, O_NONBLOCK) < 0) {
  pr_err("Could not create pipe in rw-thread(%d)\n", cpu);
  goto error;
 }

 /*
 * Size of pipe is 64kB in default based on fs/pipe.c.
 * To read/write trace data speedy, pipe size is changed.
 */

 if (fcntl(*data_pipe, F_SETPIPE_SZ, pipe_size) < 0) {
  pr_err("Could not change pipe size in rw-thread(%d)\n", cpu);
  goto error;
 }

 rw_ti->read_pipe = data_pipe[1];
 rw_ti->write_pipe = data_pipe[0];
 rw_ti->pipe_size = pipe_size;

 return NULL;

error:
 exit(EXIT_FAILURE);
}

/* Bind a thread to a cpu */
static void bind_cpu(int cpu_num)
{
 cpu_set_t mask;

 CPU_ZERO(&mask);
 CPU_SET(cpu_num, &mask);

 /* bind my thread to cpu_num by assigning zero to the first argument */
 if (sched_setaffinity(0, sizeof(mask), &mask) == -1)
  pr_err("Could not set CPU#%d affinity\n", (int)cpu_num);
}

static void *rw_thread_main(void *thread_info)
{
 ssize_t rlen, wlen;
 ssize_t ret;
 struct rw_thread_info *ts = (struct rw_thread_info *)thread_info;

 bind_cpu(ts->cpu_num);

 while (1) {
  /* Wait for a read order of trace data by Host OS */
  if (!global_run_operation) {
   pthread_mutex_lock(&mutex_notify);
   pthread_cond_wait(&cond_wakeup, &mutex_notify);
   pthread_mutex_unlock(&mutex_notify);
  }

  if (global_sig_receive)
   break;

  /*
 * Each thread read trace_pipe_raw of each cpu bounding the
 * thread, so contention of multi-threads does not occur.
 */

  rlen = splice(ts->in_fd, NULL, ts->read_pipe, NULL,
    ts->pipe_size, SPLICE_F_MOVE | SPLICE_F_MORE);

  if (rlen < 0) {
   pr_err("Splice_read in rw-thread(%d)\n", ts->cpu_num);
   goto error;
  } else if (rlen == 0) {
   /*
 * If trace data do not exist or are unreadable not
 * for exceeding the page size, splice_read returns
 * NULL. Then, this waits for being filled the data in a
 * ring-buffer.
 */

   usleep(READ_WAIT_USEC);
   pr_debug("Read retry(cpu:%d)\n", ts->cpu_num);
   continue;
  }

  wlen = 0;

  do {
   ret = splice(ts->write_pipe, NULL, ts->out_fd, NULL,
     rlen - wlen,
     SPLICE_F_MOVE | SPLICE_F_MORE);

   if (ret < 0) {
    pr_err("Splice_write in rw-thread(%d)\n",
        ts->cpu_num);
    goto error;
   } else if (ret == 0)
    /*
 * When host reader is not in time for reading
 * trace data, guest will be stopped. This is
 * because char dev in QEMU is not supported
 * non-blocking mode. Then, writer might be
 * sleep in that case.
 * This sleep will be removed by supporting
 * non-blocking mode.
 */

    sleep(1);
   wlen += ret;
  } while (wlen < rlen);
 }

 return NULL;

error:
 exit(EXIT_FAILURE);
}


pthread_t rw_thread_run(struct rw_thread_info *rw_ti)
{
 int ret;
 pthread_t rw_thread_per_cpu;

 ret = pthread_create(&rw_thread_per_cpu, NULL, rw_thread_main, rw_ti);
 if (ret != 0) {
  pr_err("Could not create a rw thread(%d)\n", rw_ti->cpu_num);
  exit(EXIT_FAILURE);
 }

 return rw_thread_per_cpu;
}

Messung V0.5
C=94 H=72 G=83

¤ Dauer der Verarbeitung: 0.3 Sekunden  ¤

*© 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.