PipeStream.st
author Claus Gittinger <cg@exept.de>
Tue, 30 Apr 1996 17:39:28 +0200
changeset 1321 52e043fb7eaf
parent 1295 83f594f05c52
child 1648 faa891d2c1b0
permissions -rw-r--r--
checkin from browser
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
     1
"
5
67342904af11 *** empty log message ***
claus
parents: 3
diff changeset
     2
 COPYRIGHT (c) 1989 by Claus Gittinger
159
514c749165c3 *** empty log message ***
claus
parents: 99
diff changeset
     3
	      All Rights Reserved
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
     4
a27a279701f8 Initial revision
claus
parents:
diff changeset
     5
 This software is furnished under a license and may be used
a27a279701f8 Initial revision
claus
parents:
diff changeset
     6
 only in accordance with the terms of that license and with the
a27a279701f8 Initial revision
claus
parents:
diff changeset
     7
 inclusion of the above copyright notice.   This software may not
a27a279701f8 Initial revision
claus
parents:
diff changeset
     8
 be provided or otherwise made available to, or used by, any
a27a279701f8 Initial revision
claus
parents:
diff changeset
     9
 other person.  No title to or ownership of the software is
a27a279701f8 Initial revision
claus
parents:
diff changeset
    10
 hereby transferred.
a27a279701f8 Initial revision
claus
parents:
diff changeset
    11
"
a27a279701f8 Initial revision
claus
parents:
diff changeset
    12
269
93162487a94b *** empty log message ***
claus
parents: 255
diff changeset
    13
NonPositionableExternalStream subclass:#PipeStream
1295
83f594f05c52 documentation
Claus Gittinger <cg@exept.de>
parents: 1133
diff changeset
    14
	instanceVariableNames:'commandString'
83f594f05c52 documentation
Claus Gittinger <cg@exept.de>
parents: 1133
diff changeset
    15
	classVariableNames:'BrokenPipeSignal'
83f594f05c52 documentation
Claus Gittinger <cg@exept.de>
parents: 1133
diff changeset
    16
	poolDictionaries:''
83f594f05c52 documentation
Claus Gittinger <cg@exept.de>
parents: 1133
diff changeset
    17
	category:'Streams-External'
88
81dacba7a63a *** empty log message ***
claus
parents: 85
diff changeset
    18
!
2
claus
parents: 1
diff changeset
    19
217
a0400fdbc933 *** empty log message ***
claus
parents: 180
diff changeset
    20
!PipeStream primitiveDefinitions!
88
81dacba7a63a *** empty log message ***
claus
parents: 85
diff changeset
    21
%{
437
claus
parents: 410
diff changeset
    22
793
3d441d66beeb NT changes
Claus Gittinger <cg@exept.de>
parents: 613
diff changeset
    23
#if defined(NT) || defined(WIN32) || defined(MSDOS)
3d441d66beeb NT changes
Claus Gittinger <cg@exept.de>
parents: 613
diff changeset
    24
# undef UNIX_LIKE
3d441d66beeb NT changes
Claus Gittinger <cg@exept.de>
parents: 613
diff changeset
    25
# define MSDOS_LIKE
3d441d66beeb NT changes
Claus Gittinger <cg@exept.de>
parents: 613
diff changeset
    26
#endif
3d441d66beeb NT changes
Claus Gittinger <cg@exept.de>
parents: 613
diff changeset
    27
88
81dacba7a63a *** empty log message ***
claus
parents: 85
diff changeset
    28
#include <stdio.h>
437
claus
parents: 410
diff changeset
    29
#define _STDIO_H_INCLUDED_
claus
parents: 410
diff changeset
    30
88
81dacba7a63a *** empty log message ***
claus
parents: 85
diff changeset
    31
#include <errno.h>
437
claus
parents: 410
diff changeset
    32
#define _ERRNO_H_INCLUDED_
claus
parents: 410
diff changeset
    33
88
81dacba7a63a *** empty log message ***
claus
parents: 85
diff changeset
    34
#ifndef transputer
81dacba7a63a *** empty log message ***
claus
parents: 85
diff changeset
    35
# include <sys/types.h>
81dacba7a63a *** empty log message ***
claus
parents: 85
diff changeset
    36
# include <sys/stat.h>
81dacba7a63a *** empty log message ***
claus
parents: 85
diff changeset
    37
#endif
230
0300c6797890 interrupt & blocking close
claus
parents: 217
diff changeset
    38
0300c6797890 interrupt & blocking close
claus
parents: 217
diff changeset
    39
/*
0300c6797890 interrupt & blocking close
claus
parents: 217
diff changeset
    40
 * on some systems errno is a macro ... check for it here
0300c6797890 interrupt & blocking close
claus
parents: 217
diff changeset
    41
 */
0300c6797890 interrupt & blocking close
claus
parents: 217
diff changeset
    42
#ifndef errno
0300c6797890 interrupt & blocking close
claus
parents: 217
diff changeset
    43
 extern errno;
0300c6797890 interrupt & blocking close
claus
parents: 217
diff changeset
    44
#endif
0300c6797890 interrupt & blocking close
claus
parents: 217
diff changeset
    45
88
81dacba7a63a *** empty log message ***
claus
parents: 85
diff changeset
    46
%}
180
c488255bd0be *** empty log message ***
claus
parents: 159
diff changeset
    47
! !
88
81dacba7a63a *** empty log message ***
claus
parents: 85
diff changeset
    48
325
claus
parents: 308
diff changeset
    49
!PipeStream primitiveFunctions!
claus
parents: 308
diff changeset
    50
%{
claus
parents: 308
diff changeset
    51
claus
parents: 308
diff changeset
    52
/*
claus
parents: 308
diff changeset
    53
 * some systems (i.e. ultrix) use fork;
claus
parents: 308
diff changeset
    54
 * were better off with a popen based on vfork ...
claus
parents: 308
diff changeset
    55
 */
claus
parents: 308
diff changeset
    56
#ifdef NEED_POPEN_WITH_VFORK
claus
parents: 308
diff changeset
    57
claus
parents: 308
diff changeset
    58
static int popen_pid = 0;
claus
parents: 308
diff changeset
    59
claus
parents: 308
diff changeset
    60
FILE *
claus
parents: 308
diff changeset
    61
popen(command, type)
claus
parents: 308
diff changeset
    62
/* const */ char *command;
claus
parents: 308
diff changeset
    63
/* const */ char *type;
claus
parents: 308
diff changeset
    64
{
claus
parents: 308
diff changeset
    65
    int pipes[2];
claus
parents: 308
diff changeset
    66
    int itype = (strcmp(type, "w") == 0 ? 1 : 0);
claus
parents: 308
diff changeset
    67
claus
parents: 308
diff changeset
    68
    if (pipe(pipes) == -1)
326
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
    69
	return NULL;
325
claus
parents: 308
diff changeset
    70
claus
parents: 308
diff changeset
    71
    switch (popen_pid = vfork()) {
claus
parents: 308
diff changeset
    72
    case -1:
326
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
    73
	(void)close(pipes[0]);
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
    74
	(void)close(pipes[1]);
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
    75
	return NULL;
325
claus
parents: 308
diff changeset
    76
claus
parents: 308
diff changeset
    77
    case 0:
326
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
    78
	if (itype) {
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
    79
	    dup2(pipes[0], fileno(stdin));
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
    80
	    close(pipes[1]);
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
    81
	} else {
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
    82
	    dup2(pipes[1], fileno(stdout));
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
    83
	    close(pipes[0]);
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
    84
	}
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
    85
	execl("/bin/sh", "/bin/sh", "-c", command, 0);
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
    86
	fprintf(stderr, "XRN Error: failed the execlp\n");
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
    87
	_exit(-1);
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
    88
	/* NOTREACHED */
325
claus
parents: 308
diff changeset
    89
claus
parents: 308
diff changeset
    90
    default:
326
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
    91
	    if (itype) {
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
    92
		close(pipes[0]);
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
    93
		return fdopen(pipes[1], "w");
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
    94
	    } else {
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
    95
		close(pipes[1]);
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
    96
		return fdopen(pipes[0], "r");
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
    97
	    }
325
claus
parents: 308
diff changeset
    98
    }
claus
parents: 308
diff changeset
    99
}
claus
parents: 308
diff changeset
   100
claus
parents: 308
diff changeset
   101
int
claus
parents: 308
diff changeset
   102
pclose(str)
claus
parents: 308
diff changeset
   103
FILE *str;
claus
parents: 308
diff changeset
   104
{
claus
parents: 308
diff changeset
   105
    int pd = 0;
claus
parents: 308
diff changeset
   106
    int status;
claus
parents: 308
diff changeset
   107
    int err;
claus
parents: 308
diff changeset
   108
claus
parents: 308
diff changeset
   109
    err = fclose(str);
claus
parents: 308
diff changeset
   110
claus
parents: 308
diff changeset
   111
    do {
326
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
   112
	if ((pd = wait(&status)) == -1)
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
   113
	{
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
   114
		err = EOF;
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
   115
		break;
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
   116
	}
325
claus
parents: 308
diff changeset
   117
    } while (pd !=  popen_pid);
claus
parents: 308
diff changeset
   118
claus
parents: 308
diff changeset
   119
    if (err == EOF)
326
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
   120
	return  -1;
325
claus
parents: 308
diff changeset
   121
claus
parents: 308
diff changeset
   122
    if (status)
326
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
   123
	status >>= 8;   /* exit status in high byte */
325
claus
parents: 308
diff changeset
   124
claus
parents: 308
diff changeset
   125
    return status;
claus
parents: 308
diff changeset
   126
}
claus
parents: 308
diff changeset
   127
claus
parents: 308
diff changeset
   128
#endif
claus
parents: 308
diff changeset
   129
claus
parents: 308
diff changeset
   130
%}
claus
parents: 308
diff changeset
   131
! !
claus
parents: 308
diff changeset
   132
613
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   133
!PipeStream class methodsFor:'documentation'!
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   134
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   135
copyright
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   136
"
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   137
 COPYRIGHT (c) 1989 by Claus Gittinger
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   138
	      All Rights Reserved
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   139
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   140
 This software is furnished under a license and may be used
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   141
 only in accordance with the terms of that license and with the
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   142
 inclusion of the above copyright notice.   This software may not
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   143
 be provided or otherwise made available to, or used by, any
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   144
 other person.  No title to or ownership of the software is
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   145
 hereby transferred.
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   146
"
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   147
!
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   148
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   149
documentation
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   150
"
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   151
    Pipestreams allow reading or writing from/to a unix command.
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   152
    For example, to get a stream reading the output of an 'ls -l'
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   153
    command, a PipeStream can be created with:
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   154
1295
83f594f05c52 documentation
Claus Gittinger <cg@exept.de>
parents: 1133
diff changeset
   155
        PipeStream readingFrom:'ls -l'
613
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   156
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   157
    the characters of the commands output can be read using the
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   158
    standard stream messages as next, nextLine etc.
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   159
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   160
    If a writing pipeStream is written to, after the command has finished,
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   161
    UNIX will generate an error-signal (SIGPIPE), which will raise the
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   162
    BrokenPipeSignal. 
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   163
    Thus, to handle this condition correctly, the following code is suggested:
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   164
1295
83f594f05c52 documentation
Claus Gittinger <cg@exept.de>
parents: 1133
diff changeset
   165
        |p|
83f594f05c52 documentation
Claus Gittinger <cg@exept.de>
parents: 1133
diff changeset
   166
        p := PipeStream writingTo:'echo hello'.
83f594f05c52 documentation
Claus Gittinger <cg@exept.de>
parents: 1133
diff changeset
   167
        PipeStream brokenPipeSignal handle:[:ex |
83f594f05c52 documentation
Claus Gittinger <cg@exept.de>
parents: 1133
diff changeset
   168
            'broken pipe' printNewline.
83f594f05c52 documentation
Claus Gittinger <cg@exept.de>
parents: 1133
diff changeset
   169
            p shutDown.
83f594f05c52 documentation
Claus Gittinger <cg@exept.de>
parents: 1133
diff changeset
   170
            ex return
83f594f05c52 documentation
Claus Gittinger <cg@exept.de>
parents: 1133
diff changeset
   171
        ] do:[
83f594f05c52 documentation
Claus Gittinger <cg@exept.de>
parents: 1133
diff changeset
   172
            p nextPutLine:'oops'.
83f594f05c52 documentation
Claus Gittinger <cg@exept.de>
parents: 1133
diff changeset
   173
           'after write' printNewline.
83f594f05c52 documentation
Claus Gittinger <cg@exept.de>
parents: 1133
diff changeset
   174
            p close.
83f594f05c52 documentation
Claus Gittinger <cg@exept.de>
parents: 1133
diff changeset
   175
           'after close' printNewline
83f594f05c52 documentation
Claus Gittinger <cg@exept.de>
parents: 1133
diff changeset
   176
        ]
613
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   177
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   178
    Notice, that if the Stream is buffered, the Signal may occur some time after
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   179
    the write - or even at close time; to avoid a recursive signal in the exception
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   180
    handler, a shutDown is useful there.
1295
83f594f05c52 documentation
Claus Gittinger <cg@exept.de>
parents: 1133
diff changeset
   181
83f594f05c52 documentation
Claus Gittinger <cg@exept.de>
parents: 1133
diff changeset
   182
    [author:]
83f594f05c52 documentation
Claus Gittinger <cg@exept.de>
parents: 1133
diff changeset
   183
        Claus Gittinger
613
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   184
"
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   185
! !
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   186
2
claus
parents: 1
diff changeset
   187
!PipeStream class methodsFor:'initialization'!
claus
parents: 1
diff changeset
   188
claus
parents: 1
diff changeset
   189
initialize
claus
parents: 1
diff changeset
   190
    "setup the signal"
claus
parents: 1
diff changeset
   191
57
db9677479d35 *** empty log message ***
claus
parents: 49
diff changeset
   192
    BrokenPipeSignal isNil ifTrue:[
582
21f08116b28d BrokenPipeSignal now a child of WriteErrorSignal
Claus Gittinger <cg@exept.de>
parents: 530
diff changeset
   193
	BrokenPipeSignal := WriteErrorSignal newSignalMayProceed:true.
159
514c749165c3 *** empty log message ***
claus
parents: 99
diff changeset
   194
	BrokenPipeSignal nameClass:self message:#brokenPipeSignal.
514c749165c3 *** empty log message ***
claus
parents: 99
diff changeset
   195
	BrokenPipeSignal notifierString:'write on a pipe with no one to read'.
57
db9677479d35 *** empty log message ***
claus
parents: 49
diff changeset
   196
    ]
2
claus
parents: 1
diff changeset
   197
! !
claus
parents: 1
diff changeset
   198
613
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   199
!PipeStream class methodsFor:'instance creation'!
2
claus
parents: 1
diff changeset
   200
613
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   201
readingFrom:commandString
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   202
    "create and return a new pipeStream which can read from the unix command
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   203
     given by command."
2
claus
parents: 1
diff changeset
   204
613
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   205
    ^ (self basicNew) readingFrom:commandString
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   206
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   207
    "PipeStream readingFrom:'ls -l'"
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   208
!
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   209
2
claus
parents: 1
diff changeset
   210
writingTo:commandString
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   211
    "create and return a new pipeStream which can write to the unix command
a27a279701f8 Initial revision
claus
parents:
diff changeset
   212
     given by command."
a27a279701f8 Initial revision
claus
parents:
diff changeset
   213
2
claus
parents: 1
diff changeset
   214
    ^ (self basicNew) writingTo:commandString
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   215
a27a279701f8 Initial revision
claus
parents:
diff changeset
   216
    "PipeStream writingTo:'sort'"
613
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   217
! !
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   218
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   219
!PipeStream class methodsFor:'Signal constants'!
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   220
613
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   221
brokenPipeSignal
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   222
    "return the signal used to handle SIGPIPE unix-signals"
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   223
613
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   224
    ^ BrokenPipeSignal
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   225
! !
a27a279701f8 Initial revision
claus
parents:
diff changeset
   226
99
afba01fbe15c added access method for commandString
claus
parents: 93
diff changeset
   227
!PipeStream methodsFor:'accessing'!
afba01fbe15c added access method for commandString
claus
parents: 93
diff changeset
   228
afba01fbe15c added access method for commandString
claus
parents: 93
diff changeset
   229
commandString
afba01fbe15c added access method for commandString
claus
parents: 93
diff changeset
   230
    "return the command string"
afba01fbe15c added access method for commandString
claus
parents: 93
diff changeset
   231
afba01fbe15c added access method for commandString
claus
parents: 93
diff changeset
   232
    ^ commandString
afba01fbe15c added access method for commandString
claus
parents: 93
diff changeset
   233
! !
afba01fbe15c added access method for commandString
claus
parents: 93
diff changeset
   234
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   235
!PipeStream methodsFor:'instance release'!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   236
a27a279701f8 Initial revision
claus
parents:
diff changeset
   237
closeFile
230
0300c6797890 interrupt & blocking close
claus
parents: 217
diff changeset
   238
    "low level close - redefined since we close a pipe here.
0300c6797890 interrupt & blocking close
claus
parents: 217
diff changeset
   239
     This waits for the command to finish. 
0300c6797890 interrupt & blocking close
claus
parents: 217
diff changeset
   240
     Use shutDown for a fast (nonBlocking) close."
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   241
13
62303f84ff5f *** empty log message ***
claus
parents: 10
diff changeset
   242
%{  /* UNLIMITEDSTACK */
793
3d441d66beeb NT changes
Claus Gittinger <cg@exept.de>
parents: 613
diff changeset
   243
#if !defined(transputer) && !defined(MSDOS_LIKE)
360
claus
parents: 345
diff changeset
   244
    OBJ fp;
13
62303f84ff5f *** empty log message ***
claus
parents: 10
diff changeset
   245
1133
961f2b095c22 underline cleanup
Claus Gittinger <cg@exept.de>
parents: 793
diff changeset
   246
    if ((fp = __INST(filePointer)) != nil) {
961f2b095c22 underline cleanup
Claus Gittinger <cg@exept.de>
parents: 793
diff changeset
   247
	__INST(filePointer) = nil;
159
514c749165c3 *** empty log message ***
claus
parents: 99
diff changeset
   248
	/*
514c749165c3 *** empty log message ***
claus
parents: 99
diff changeset
   249
	 * allow interrupt even when blocking here ...
514c749165c3 *** empty log message ***
claus
parents: 99
diff changeset
   250
	 */
362
claus
parents: 360
diff changeset
   251
	__BEGIN_INTERRUPTABLE__
475
b57530aa1b0a use new FILE* wrapper macros (based on externalAddress)
Claus Gittinger <cg@exept.de>
parents: 437
diff changeset
   252
	pclose(__FILEVal(fp));
362
claus
parents: 360
diff changeset
   253
	__END_INTERRUPTABLE__
13
62303f84ff5f *** empty log message ***
claus
parents: 10
diff changeset
   254
    }
793
3d441d66beeb NT changes
Claus Gittinger <cg@exept.de>
parents: 613
diff changeset
   255
#endif /* not transputer && not MSDOS_LIKE */
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   256
%}
369
claus
parents: 362
diff changeset
   257
!
claus
parents: 362
diff changeset
   258
claus
parents: 362
diff changeset
   259
closeFileDescriptor
claus
parents: 362
diff changeset
   260
    "alternative very low level close 
claus
parents: 362
diff changeset
   261
     This closes the underlying OS-fileDescriptor 
claus
parents: 362
diff changeset
   262
     - and will NOT write any buffered data to the stream.
claus
parents: 362
diff changeset
   263
     You have been warned."
claus
parents: 362
diff changeset
   264
claus
parents: 362
diff changeset
   265
%{  /* NOCONTEXT */
793
3d441d66beeb NT changes
Claus Gittinger <cg@exept.de>
parents: 613
diff changeset
   266
#if !defined(transputer) && !defined(MSDOS_LIKE)
369
claus
parents: 362
diff changeset
   267
claus
parents: 362
diff changeset
   268
    OBJ fp;
claus
parents: 362
diff changeset
   269
    FILE *f;
claus
parents: 362
diff changeset
   270
1133
961f2b095c22 underline cleanup
Claus Gittinger <cg@exept.de>
parents: 793
diff changeset
   271
    if ((fp = __INST(filePointer)) != nil) {
961f2b095c22 underline cleanup
Claus Gittinger <cg@exept.de>
parents: 793
diff changeset
   272
	__INST(filePointer) = nil;
369
claus
parents: 362
diff changeset
   273
	__BEGIN_INTERRUPTABLE__
475
b57530aa1b0a use new FILE* wrapper macros (based on externalAddress)
Claus Gittinger <cg@exept.de>
parents: 437
diff changeset
   274
	f = __FILEVal(fp);
369
claus
parents: 362
diff changeset
   275
	close(fileno(f));
claus
parents: 362
diff changeset
   276
	__END_INTERRUPTABLE__
claus
parents: 362
diff changeset
   277
    }
793
3d441d66beeb NT changes
Claus Gittinger <cg@exept.de>
parents: 613
diff changeset
   278
#endif /* not transputer && not MSDOS_LIKE */
369
claus
parents: 362
diff changeset
   279
%}
613
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   280
!
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   281
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   282
disposed
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   283
    "redefined to avoid blocking in close."
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   284
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   285
    self shutDown
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   286
!
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   287
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   288
shutDown
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   289
    "close the Stream, ignoring any broken-pipe errors"
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   290
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   291
    BrokenPipeSignal catch:[
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   292
	Lobby unregister:self.
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   293
	self closeFileDescriptor
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   294
    ]
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   295
! !
a27a279701f8 Initial revision
claus
parents:
diff changeset
   296
a27a279701f8 Initial revision
claus
parents:
diff changeset
   297
!PipeStream methodsFor:'private'!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   298
379
5b5a130ccd09 revision added
claus
parents: 369
diff changeset
   299
atEnd
93
e31220cb391f *** empty log message ***
claus
parents: 88
diff changeset
   300
    "return true, if position is at end"
e31220cb391f *** empty log message ***
claus
parents: 88
diff changeset
   301
e31220cb391f *** empty log message ***
claus
parents: 88
diff changeset
   302
%{  /* NOCONTEXT */
e31220cb391f *** empty log message ***
claus
parents: 88
diff changeset
   303
    FILE *f;
e31220cb391f *** empty log message ***
claus
parents: 88
diff changeset
   304
    OBJ t;
e31220cb391f *** empty log message ***
claus
parents: 88
diff changeset
   305
    OBJ _true = true;
e31220cb391f *** empty log message ***
claus
parents: 88
diff changeset
   306
    int c;
e31220cb391f *** empty log message ***
claus
parents: 88
diff changeset
   307
1133
961f2b095c22 underline cleanup
Claus Gittinger <cg@exept.de>
parents: 793
diff changeset
   308
    if (__INST(hitEOF) == _true) {
159
514c749165c3 *** empty log message ***
claus
parents: 99
diff changeset
   309
	RETURN (_true);
93
e31220cb391f *** empty log message ***
claus
parents: 88
diff changeset
   310
    }
1133
961f2b095c22 underline cleanup
Claus Gittinger <cg@exept.de>
parents: 793
diff changeset
   311
    if ((t = __INST(filePointer)) != nil) {
475
b57530aa1b0a use new FILE* wrapper macros (based on externalAddress)
Claus Gittinger <cg@exept.de>
parents: 437
diff changeset
   312
	f = __FILEVal(t);
159
514c749165c3 *** empty log message ***
claus
parents: 99
diff changeset
   313
	if (feof(f)) {
1133
961f2b095c22 underline cleanup
Claus Gittinger <cg@exept.de>
parents: 793
diff changeset
   314
	    __INST(hitEOF) = true;
159
514c749165c3 *** empty log message ***
claus
parents: 99
diff changeset
   315
	    RETURN (true);
514c749165c3 *** empty log message ***
claus
parents: 99
diff changeset
   316
	}
401
claus
parents: 384
diff changeset
   317
	clearerr(f);
159
514c749165c3 *** empty log message ***
claus
parents: 99
diff changeset
   318
	RETURN ( false );
93
e31220cb391f *** empty log message ***
claus
parents: 88
diff changeset
   319
    }
e31220cb391f *** empty log message ***
claus
parents: 88
diff changeset
   320
%}
e31220cb391f *** empty log message ***
claus
parents: 88
diff changeset
   321
.
e31220cb391f *** empty log message ***
claus
parents: 88
diff changeset
   322
    ^ super atEnd
613
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   323
!
93
e31220cb391f *** empty log message ***
claus
parents: 88
diff changeset
   324
49
f1c2d75f2eb6 *** empty log message ***
claus
parents: 32
diff changeset
   325
openPipeFor:aCommandString withMode:mode
255
2b2c5c0facab *** empty log message ***
claus
parents: 230
diff changeset
   326
    "open a pipe to the unix command in aCcommandString; 
2b2c5c0facab *** empty log message ***
claus
parents: 230
diff changeset
   327
     mode may be 'r' or 'w'"
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   328
a27a279701f8 Initial revision
claus
parents:
diff changeset
   329
    |retVal|
a27a279701f8 Initial revision
claus
parents:
diff changeset
   330
326
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
   331
    filePointer notNil ifTrue:[
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
   332
	"the pipe was already open ...
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
   333
	 this should (can) not happen."
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
   334
	^ self errorOpen
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
   335
    ].
255
2b2c5c0facab *** empty log message ***
claus
parents: 230
diff changeset
   336
13
62303f84ff5f *** empty log message ***
claus
parents: 10
diff changeset
   337
%{  /* STACK: 32000 */
793
3d441d66beeb NT changes
Claus Gittinger <cg@exept.de>
parents: 613
diff changeset
   338
#if !defined(transputer) && !defined(MSDOS_LIKE)
255
2b2c5c0facab *** empty log message ***
claus
parents: 230
diff changeset
   339
    FILE *f;
477
8710aba7876b oops - making id's real objects requires a store macro
Claus Gittinger <cg@exept.de>
parents: 475
diff changeset
   340
    OBJ fp;
255
2b2c5c0facab *** empty log message ***
claus
parents: 230
diff changeset
   341
1133
961f2b095c22 underline cleanup
Claus Gittinger <cg@exept.de>
parents: 793
diff changeset
   342
    __INST(lastErrorNumber) = nil;
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   343
255
2b2c5c0facab *** empty log message ***
claus
parents: 230
diff changeset
   344
    if (__isString(aCommandString) && __isString(mode)) {
362
claus
parents: 360
diff changeset
   345
	__BEGIN_INTERRUPTABLE__
255
2b2c5c0facab *** empty log message ***
claus
parents: 230
diff changeset
   346
	do {
793
3d441d66beeb NT changes
Claus Gittinger <cg@exept.de>
parents: 613
diff changeset
   347
# ifdef LINUX
255
2b2c5c0facab *** empty log message ***
claus
parents: 230
diff changeset
   348
	    /* LINUX returns a non-NULL f even when interrupted */
2b2c5c0facab *** empty log message ***
claus
parents: 230
diff changeset
   349
	    errno = 0;
1133
961f2b095c22 underline cleanup
Claus Gittinger <cg@exept.de>
parents: 793
diff changeset
   350
	    f = (FILE *)popen((char *) __stringVal(aCommandString),
961f2b095c22 underline cleanup
Claus Gittinger <cg@exept.de>
parents: 793
diff changeset
   351
			      (char *) __stringVal(mode));
255
2b2c5c0facab *** empty log message ***
claus
parents: 230
diff changeset
   352
	    if (errno == EINTR)
2b2c5c0facab *** empty log message ***
claus
parents: 230
diff changeset
   353
		f = NULL;
793
3d441d66beeb NT changes
Claus Gittinger <cg@exept.de>
parents: 613
diff changeset
   354
# else
1133
961f2b095c22 underline cleanup
Claus Gittinger <cg@exept.de>
parents: 793
diff changeset
   355
	    f = (FILE *)popen((char *) __stringVal(aCommandString),
961f2b095c22 underline cleanup
Claus Gittinger <cg@exept.de>
parents: 793
diff changeset
   356
			      (char *) __stringVal(mode));
793
3d441d66beeb NT changes
Claus Gittinger <cg@exept.de>
parents: 613
diff changeset
   357
# endif /* LINUX */
255
2b2c5c0facab *** empty log message ***
claus
parents: 230
diff changeset
   358
	} while ((f == NULL) && (errno == EINTR));
362
claus
parents: 360
diff changeset
   359
	__END_INTERRUPTABLE__
claus
parents: 360
diff changeset
   360
255
2b2c5c0facab *** empty log message ***
claus
parents: 230
diff changeset
   361
	if (f == NULL) {
1133
961f2b095c22 underline cleanup
Claus Gittinger <cg@exept.de>
parents: 793
diff changeset
   362
	    __INST(lastErrorNumber) = __MKSMALLINT(errno);
255
2b2c5c0facab *** empty log message ***
claus
parents: 230
diff changeset
   363
	} else {
401
claus
parents: 384
diff changeset
   364
	    clearerr(f);
1133
961f2b095c22 underline cleanup
Claus Gittinger <cg@exept.de>
parents: 793
diff changeset
   365
	    __INST(filePointer) = fp = __MKOBJ(f); __STORE(self, fp);
255
2b2c5c0facab *** empty log message ***
claus
parents: 230
diff changeset
   366
	    retVal = self;
159
514c749165c3 *** empty log message ***
claus
parents: 99
diff changeset
   367
	}
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   368
    }
793
3d441d66beeb NT changes
Claus Gittinger <cg@exept.de>
parents: 613
diff changeset
   369
#endif /* not transputer && not MSDOS_LIKE */
255
2b2c5c0facab *** empty log message ***
claus
parents: 230
diff changeset
   370
%}.
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   371
    retVal notNil ifTrue:[
159
514c749165c3 *** empty log message ***
claus
parents: 99
diff changeset
   372
	commandString := aCommandString.
514c749165c3 *** empty log message ***
claus
parents: 99
diff changeset
   373
	buffered := true.
514c749165c3 *** empty log message ***
claus
parents: 99
diff changeset
   374
	hitEOF := false.
406
claus
parents: 401
diff changeset
   375
	binary := false.
159
514c749165c3 *** empty log message ***
claus
parents: 99
diff changeset
   376
	Lobby register:self
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   377
    ].
255
2b2c5c0facab *** empty log message ***
claus
parents: 230
diff changeset
   378
    lastErrorNumber notNil ifTrue:[
2b2c5c0facab *** empty log message ***
claus
parents: 230
diff changeset
   379
	"
308
f04744ef7b5d *** empty log message ***
claus
parents: 302
diff changeset
   380
	 the pipe open failed for some reason ...
f04744ef7b5d *** empty log message ***
claus
parents: 302
diff changeset
   381
	 ... this may be either due to an invalid command string,
f04744ef7b5d *** empty log message ***
claus
parents: 302
diff changeset
   382
	 or due to the system running out of memory (when forking
f04744ef7b5d *** empty log message ***
claus
parents: 302
diff changeset
   383
	 the unix process)
255
2b2c5c0facab *** empty log message ***
claus
parents: 230
diff changeset
   384
	"
2b2c5c0facab *** empty log message ***
claus
parents: 230
diff changeset
   385
	^ self openError
2b2c5c0facab *** empty log message ***
claus
parents: 230
diff changeset
   386
    ].
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   387
    ^ retVal
a27a279701f8 Initial revision
claus
parents:
diff changeset
   388
!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   389
a27a279701f8 Initial revision
claus
parents:
diff changeset
   390
readingFrom:command
a27a279701f8 Initial revision
claus
parents:
diff changeset
   391
    "setup the receiver to read from command"
a27a279701f8 Initial revision
claus
parents:
diff changeset
   392
326
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
   393
    mode := #readonly. didWrite := false.
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   394
    ^ self openPipeFor:command withMode:'r'
a27a279701f8 Initial revision
claus
parents:
diff changeset
   395
!
a27a279701f8 Initial revision
claus
parents:
diff changeset
   396
a27a279701f8 Initial revision
claus
parents:
diff changeset
   397
writingTo:command
a27a279701f8 Initial revision
claus
parents:
diff changeset
   398
    "setup the receiver to write to command"
a27a279701f8 Initial revision
claus
parents:
diff changeset
   399
326
d2902942491d *** empty log message ***
claus
parents: 325
diff changeset
   400
    mode := #writeonly. didWrite := true.
1
a27a279701f8 Initial revision
claus
parents:
diff changeset
   401
    ^ self openPipeFor:command withMode:'w'
a27a279701f8 Initial revision
claus
parents:
diff changeset
   402
! !
613
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   403
1295
83f594f05c52 documentation
Claus Gittinger <cg@exept.de>
parents: 1133
diff changeset
   404
!PipeStream class methodsFor:'documentation'!
83f594f05c52 documentation
Claus Gittinger <cg@exept.de>
parents: 1133
diff changeset
   405
83f594f05c52 documentation
Claus Gittinger <cg@exept.de>
parents: 1133
diff changeset
   406
version
83f594f05c52 documentation
Claus Gittinger <cg@exept.de>
parents: 1133
diff changeset
   407
    ^ '$Header: /cvs/stx/stx/libbasic/PipeStream.st,v 1.43 1996-04-25 17:01:50 cg Exp $'
83f594f05c52 documentation
Claus Gittinger <cg@exept.de>
parents: 1133
diff changeset
   408
! !
613
0af19c3594fc checkin from browser
Claus Gittinger <cg@exept.de>
parents: 582
diff changeset
   409
PipeStream initialize!