Annotation of embedaddon/mtr/packet/command_cygwin.c, revision 1.1

1.1     ! misho       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>