cdesktopenv/cde/programs/dtprintinfo/libUI/MotifUI/MotifThread.C

184 lines
4.4 KiB
C

/*
* CDE - Common Desktop Environment
*
* Copyright (c) 1993-2012, The Open Group. All rights reserved.
*
* These libraries and programs are free software; you can
* redistribute them and/or modify them under the terms of the GNU
* Lesser General Public License as published by the Free Software
* Foundation; either version 2 of the License, or (at your option)
* any later version.
*
* These libraries and programs are distributed in the hope that
* they will be useful, but WITHOUT ANY WARRANTY; without even the
* implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
* PURPOSE. See the GNU Lesser General Public License for more
* details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with these libraries and programs; if not, write
* to the Free Software Foundation, Inc., 51 Franklin Street, Fifth
* Floor, Boston, MA 02110-1301 USA
*/
/* $TOG: MotifThread.C /main/3 1998/07/24 16:15:54 mgreess $ */
/* *
* (c) Copyright 1993, 1994 Hewlett-Packard Company *
* (c) Copyright 1993, 1994 International Business Machines Corp. *
* (c) Copyright 1993, 1994 Sun Microsystems, Inc. *
* (c) Copyright 1993, 1994 Novell, Inc. *
*/
#include "MotifThread.h"
#include <unistd.h>
#include <stdlib.h>
#include <sys/wait.h>
#include <string.h>
#include <fcntl.h>
#include <errno.h>
MotifThread::MotifThread(MotifUI *obj, const char *cmd,
MotifThreadCallback cb, ThreadCallback cb1,
int buf_len)
{
CreateThread(obj, cmd, -1, -1, cb, cb1, buf_len);
}
MotifThread::MotifThread(MotifUI *obj, int pid, int fd,
MotifThreadCallback cb, ThreadCallback cb1,
int buf_len)
{
CreateThread(obj, NULL, pid, fd, cb, cb1, buf_len);
}
MotifThread::MotifThread(MotifUI *obj, int socket,
MotifThreadCallback cb, ThreadCallback cb1,
int buf_len)
{
CreateThread(obj, NULL, -1, socket, cb, cb1, buf_len);
}
void MotifThread::CreateThread(MotifUI *_obj, const char *cmd, int _pid,
int _fd, MotifThreadCallback _cb,
ThreadCallback _cb1, int _buf_len)
{
int m_stdout[2];
cb = _cb;
cb1 = _cb1;
obj = _obj;
output = NULL;
out1 = NULL;
inputID = 0;
if (_buf_len < 0)
buf_len = 512;
else
buf_len = _buf_len;
if (cmd)
{
fd = -1;
if (pipe(m_stdout) < 0)
{
output = strdup(strerror(errno));
status = -1;
Halt();
return;
}
if ((pid = fork()) == 0) // In Child
{
close(m_stdout[0]);
close(1);
dup(m_stdout[1]);
close(m_stdout[1]);
execlp(KORNSHELL, "ksh", "-c", cmd, NULL);
char *msg = strerror(errno);
write(1, msg, strlen(msg));
exit(-1);
}
else if (pid == -1)
{
close(m_stdout[0]);
output = strdup(strerror(errno));
status = -1;
Halt();
return;
}
close(m_stdout[1]);
fd = m_stdout[0];
}
else
{
pid = (pid_t)_pid;
fd = _fd;
}
fcntl(fd, F_SETFL, O_NDELAY);
len = buf_len;
ctr = 1;
output = (char *)malloc(buf_len);
out1 = output;
out2 = out1 + buf_len - 1;
long mask = XtInputReadMask | XtInputExceptMask;
inputID = XtAppAddInput(obj->appContext, fd, (XtPointer)mask,
&(MotifThread::GetOutputCB), this);
}
MotifThread::~MotifThread()
{
if (fd >= 0)
Halt();
if (inputID)
XtRemoveInput(inputID);
free(output);
}
void MotifThread::Halt()
{
if (out1)
*out1 = '\0';
close(fd);
fd = -1;
if (pid != -1)
{
pid_t w;
while ((w = wait(&status)) != pid && w != -1);
status = (status >> 8) & 0xFF;
}
else
status = 0;
(*cb)(this, obj, cb1);
}
void MotifThread::GetOutput()
{
int n = read(fd, out1, len);
if (n == len)
{
len = buf_len;
ctr++;
output = (char *)realloc(output, (ctr * len));
out1 = output + ((ctr - 1) * len);
out2 = out1 + len - 1;
}
else if (n > 0)
{
out1 += n;
len = out2 - out1 + 1;
}
else
Halt();
}
void MotifThread::GetOutputCB(XtPointer closure, int * /*fid*/,
XtInputId * /*id*/)
{
MotifThread *obj = (MotifThread *)closure;
obj->GetOutput();
}