File:  [ELWIX - Embedded LightWeight unIX -] / embedaddon / mtr / packet / command_cygwin.c
Revision 1.1.1.1 (vendor branch): download - view: text, annotated - select for diffs - revision graph
Mon Oct 21 14:25:31 2019 UTC (5 years, 5 months ago) by misho
Branches: mtr, MAIN
CVS tags: v0_92, HEAD
mtr ver 0.92

    1: /*
    2:     mtr  --  a network diagnostic tool
    3:     Copyright (C) 2016  Matt Kimball
    4: 
    5:     This program is free software; you can redistribute it and/or modify
    6:     it under the terms of the GNU General Public License version 2 as
    7:     published by the Free Software Foundation.
    8: 
    9:     This program is distributed in the hope that it will be useful,
   10:     but WITHOUT ANY WARRANTY; without even the implied warranty of
   11:     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
   12:     GNU General Public License for more details.
   13: 
   14:     You should have received a copy of the GNU General Public License
   15:     along with this program; if not, write to the Free Software
   16:     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
   17: */
   18: 
   19: #include "command.h"
   20: 
   21: #include <errno.h>
   22: #include <io.h>
   23: #include <stdio.h>
   24: 
   25: /*
   26:     A completion routine to be called by Windows when a read from
   27:     the command stream has completed.
   28: */
   29: static
   30: void CALLBACK finish_read_command(
   31:     DWORD status,
   32:     DWORD size_read,
   33:     OVERLAPPED * overlapped)
   34: {
   35:     struct command_buffer_t *buffer;
   36:     char *read_position;
   37: 
   38:     /*
   39:        hEvent is unusuaed by ReadFileEx, so we use it to pass
   40:        our command_buffer structure.
   41:      */
   42:     buffer = (struct command_buffer_t *) overlapped->hEvent;
   43: 
   44:     if (status) {
   45:         /*  When the stream is closed ERROR_BROKEN_PIPE will be the result  */
   46:         if (status == ERROR_BROKEN_PIPE) {
   47:             buffer->platform.pipe_open = false;
   48:             return;
   49:         }
   50: 
   51:         fprintf(stderr, "ReadFileEx completion failure %d\n", status);
   52:         exit(EXIT_FAILURE);
   53:     }
   54: 
   55:     /*  Copy from the overlapped I/O buffer to the incoming command buffer  */
   56:     read_position =
   57:         &buffer->incoming_buffer[buffer->incoming_read_position];
   58:     memcpy(read_position, buffer->platform.overlapped_buffer, size_read);
   59: 
   60:     /*  Account for the newly read data  */
   61:     buffer->incoming_read_position += size_read;
   62:     buffer->platform.read_active = false;
   63: }
   64: 
   65: /*
   66:     An APC which does nothing, to be used only to wake from the current
   67:     alertable wait.
   68: */
   69: static
   70: void CALLBACK empty_apc(
   71:     ULONG * param)
   72: {
   73: }
   74: 
   75: /*  Wake from the next alertable wait without waiting for newly read data  */
   76: static
   77: void queue_empty_apc(
   78:     void)
   79: {
   80:     if (QueueUserAPC((PAPCFUNC) empty_apc, GetCurrentThread(), 0) == 0) {
   81:         fprintf(stderr, "Unexpected QueueUserAPC failure %d\n",
   82:                 GetLastError());
   83:         exit(EXIT_FAILURE);
   84:     }
   85: }
   86: 
   87: /*  Start a new overlapped I/O read from the command stream  */
   88: void start_read_command(
   89:     struct command_buffer_t *buffer)
   90: {
   91:     HANDLE command_stream = (HANDLE) get_osfhandle(buffer->command_stream);
   92:     int space_remaining =
   93:         COMMAND_BUFFER_SIZE - buffer->incoming_read_position - 1;
   94:     int err;
   95: 
   96:     /*  If a read is already active, or the pipe is closed, do nothing  */
   97:     if (!buffer->platform.pipe_open || buffer->platform.read_active) {
   98:         return;
   99:     }
  100: 
  101:     memset(&buffer->platform.overlapped, 0, sizeof(OVERLAPPED));
  102:     buffer->platform.overlapped.hEvent = (HANDLE) buffer;
  103: 
  104:     if (!ReadFileEx
  105:         (command_stream, buffer->platform.overlapped_buffer,
  106:          space_remaining, &buffer->platform.overlapped,
  107:          finish_read_command)) {
  108: 
  109:         err = GetLastError();
  110: 
  111:         if (err == ERROR_BROKEN_PIPE) {
  112:             /*  If the command stream has been closed, we need to wake from
  113:                the next altertable wait to exit the main loop  */
  114:             buffer->platform.pipe_open = false;
  115:             queue_empty_apc();
  116: 
  117:             return;
  118:         } else if (err != WAIT_IO_COMPLETION) {
  119:             fprintf(stderr, "Unexpected ReadFileEx failure %d\n",
  120:                     GetLastError());
  121:             exit(EXIT_FAILURE);
  122:         }
  123:     }
  124: 
  125:     /*  Remember that we have started an overlapped read already  */
  126:     buffer->platform.read_active = true;
  127: }
  128: 
  129: /*  Initialize the command buffer, and start the first overlapped read  */
  130: void init_command_buffer(
  131:     struct command_buffer_t *command_buffer,
  132:     int command_stream)
  133: {
  134:     memset(command_buffer, 0, sizeof(struct command_buffer_t));
  135:     command_buffer->command_stream = command_stream;
  136:     command_buffer->platform.pipe_open = true;
  137: }
  138: 
  139: /*
  140:     Return with errno EPIPE if the command stream has been closed.
  141:     Otherwise, not much to do for Cygwin, since we are using Overlapped I/O
  142:     to read commands.
  143: */
  144: int read_commands(
  145:     struct command_buffer_t *buffer)
  146: {
  147:     if (!buffer->platform.pipe_open) {
  148:         errno = EPIPE;
  149:         return -1;
  150:     }
  151: 
  152:     return 0;
  153: }

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>