mirror of the now-defunct rocklinux.org
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

423 lines
9.8 KiB

#include <stdio.h>
#include <libgen.h>
#include <unistd.h>
#include <string.h>
#include <errno.h>
#include <stdlib.h>
#include <stdint.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/socket.h>
#include <sys/time.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <fcntl.h>
#include <glob.h>
#include <signal.h>
extern char **environ;
#define LIBDIR "/ROCK/tools.cross/pseudonative.lib"
#define BINDIR_1 "/ROCK/tools.cross/pseudonative.bin"
#define BINDIR_2 "/ROCK/tools.cross/bin"
#define BINDIR_SIZE 100
#define TCPPORT 28336
#define CMD_EXEC 1
#define CMD_ADDARG 2
#define CMD_ADDENV 3
#define CMD_CHDIR 4
#define CMD_CHROOT 5
#define CMD_EXIT 6
#define CMD_BUILDID 7
#define CMD_WRITE0 1000
#define CMD_CLOSE0 2000
#define CMD_ENABLE0 3000
int config_debug = 0;
int config_remote = 0;
struct pn_pkg {
uint32_t len;
uint32_t type;
char data[];
};
int conn_open(const char *peer)
{
struct sockaddr_in sin;
int conn;
sin.sin_family = PF_INET;
sin.sin_port = htons(TCPPORT);
if ( !inet_aton(peer, &sin.sin_addr) ) {
fprintf(stderr, "pseudonative_handler: can't parse address '%s'\n", peer);
exit(-1);
}
conn = socket(PF_INET, SOCK_STREAM, 0);
if (conn < 0) {
fprintf(stderr, "pseudonative_handler: can't create socket: %s\n", strerror(errno));
exit(-1);
}
if (connect(conn, (struct sockaddr *)&sin, sizeof (sin)) < 0) {
fprintf(stderr, "pseudonative_handler: can't connect to remote host: %s\n", strerror(errno));
exit(-1);
}
return conn;
}
void conn_send(int conn, int len, int type, char *data)
{
struct pn_pkg *p = malloc( sizeof(struct pn_pkg) + len );
int i, rc;
p->len = htonl(len);
p->type = htonl(type);
memcpy(p->data, data, len);
len += sizeof(struct pn_pkg);
for (i = 0; i < len; i += rc) {
rc = write(conn, (char*)p + i, len - i);
if ( rc <= 0 ) {
if ( errno == ECONNRESET && type == CMD_CLOSE0 ) break;
fprintf(stderr, "pseudonative_handler: network write error: %s\n", strerror(errno));
exit(-1);
}
}
free(p);
}
int conn_read(int conn, char *buf, int len, int nonblock)
{
int f = -1;
if ( nonblock ) {
f = fcntl(conn, F_GETFL, 0);
fcntl(conn, F_SETFL, f | O_NONBLOCK);
}
while (len) {
int rc = read(conn, buf, len);
if ( rc < 0 ) {
if ( errno == EAGAIN ) break;
fprintf(stderr, "pseudonative_handler: network read error: %s\n", strerror(errno));
exit(-1);
}
if ( rc == 0 ) {
fprintf(stderr, "pseudonative_handler: connection closed by peer\n");
exit(-1);
}
if ( f != -1 )
fcntl(conn, F_SETFL, f);
len -= rc; buf += rc; f = -1;
}
if ( f != -1 )
fcntl(conn, F_SETFL, f);
return len;
}
struct pn_pkg *conn_recv(int conn)
{
struct pn_pkg *p = malloc( sizeof(struct pn_pkg) );
if ( conn_read(conn, (char*)p, 8, 1) ) {
free(p);
return 0;
}
p->len = ntohl(p->len);
p->type = ntohl(p->type);
p = realloc(p, sizeof(struct pn_pkg) + p->len);
conn_read(conn, p->data, p->len, 0);
return p;
}
int exec_remote(char ** argv)
{
char buf[1024], *txt;
int ret = -1, open_0 = 1;
int enable_0 = 0;
int f, i, conn, rc;
struct timeval tv;
struct pn_pkg *p;
fd_set rfds;
/* connect */
txt = getenv("ROCKCFG_PSEUDONATIVE_NATIVEHOST");
if ( txt && *txt ) {
conn = conn_open(txt);
} else {
fprintf(stderr, "pseudonative_handler: no native host configured\n");
exit(-1);
}
/* send build id */
txt = getenv("ROCKCFG_ID");
if ( txt && *txt )
conn_send(conn, strlen(txt)+1, CMD_BUILDID, txt);
/* mount and chroot */
txt = getenv("ROCKCFG_PSEUDONATIVE_NFSROOT");
if ( txt && *txt )
conn_send(conn, strlen(txt)+1, CMD_CHROOT, txt);
/* current directory */
if ( getcwd(buf, 1024) )
conn_send(conn, strlen(buf)+1, CMD_CHDIR, buf);
/* send environment */
for (i=0; environ[i]; i++)
conn_send(conn, strlen(environ[i])+1, CMD_ADDENV, environ[i]);
/* send arguments */
for (i=0; argv[i]; i++)
conn_send(conn, strlen(argv[i])+1, CMD_ADDARG, argv[i]);
/* execute command */
conn_send(conn, strlen(argv[0])+1, CMD_EXEC, argv[0]);
/* we pass sigpipe's to peer */
signal(SIGPIPE, SIG_IGN);
while (1)
{
FD_ZERO(&rfds);
if ( open_0 && enable_0 ) FD_SET(0, &rfds);
FD_SET(conn, &rfds);
nextselect:
tv.tv_sec = 1;
tv.tv_usec = 0;
rc = select(conn+1, &rfds, 0, 0, &tv);
if ( rc < 0 ) {
if ( errno == EINTR ) goto nextselect;
fprintf(stderr, "pseudonative_handler: select error: %s\n", strerror(errno));
exit(-1);
}
nextread:
if ( (p = conn_recv(conn)) != 0 ) {
char *d = p->data;
switch (p->type) {
case CMD_WRITE0+1:
while ( p->len ) {
rc = write(1, d, p->len);
if ( rc <= 0 ) {
if ( errno == EPIPE ) {
conn_send(conn, 0, CMD_CLOSE0+1, 0);
break;
}
fprintf(stderr, "pseudonative_handler: write error on fd1: %s\n", strerror(errno));
exit(-1);
}
d += rc; p->len -= rc;
}
break;
case CMD_WRITE0+2:
while ( p->len ) {
rc = write(2, d, p->len);
if ( rc <= 0 ) {
if ( errno == EPIPE ) {
conn_send(conn, 0, CMD_CLOSE0+2, 0);
break;
}
fprintf(stderr, "pseudonative_handler: write error on fd2: %s\n", strerror(errno));
exit(-1);
}
d += rc; p->len -= rc;
}
break;
case CMD_CLOSE0:
close(0);
open_0 = 0;
break;
case CMD_CLOSE0+1:
close(1);
break;
case CMD_CLOSE0+2:
close(2);
break;
case CMD_ENABLE0:
enable_0 = 1;
break;
case CMD_EXIT:
ret = (unsigned char)p->data[0];
free(p);
goto finish;
}
free(p);
goto nextread;
}
if( open_0 && enable_0 )
{
f = fcntl(0, F_GETFL, 0);
fcntl(0, F_SETFL, f | O_NONBLOCK);
rc = read(0, buf, 512);
fcntl(0, F_SETFL, f);
if ( rc > 0 ) {
conn_send(conn, rc, CMD_WRITE0, buf);
goto nextread;
} else if (rc == 0) {
conn_send(conn, 0, CMD_CLOSE0, 0);
open_0 = 0;
} else if ( errno != EAGAIN ) {
perror("Error on read from f2:");
}
}
}
finish:
close(conn);
if (ret == -1)
fprintf(stderr, "pseudonative_handler: sometimes everything goes wrong\n");
return ret;
}
void set_ld_lib_path()
{
FILE *f = fopen("/etc/ld.so.conf", "r");
char ld_lib_path[1024] = "/lib";
int written = 4, len, i;
char line[1024], *l;
glob_t globbuf;
l = getenv("LD_LIBRARY_PATH_PSEUDONATIVE_BACKUP");
if (l) {
l = strdup(l);
l = strtok(l, ":");
while ( l ) {
if ( strcmp(l, LIBDIR) && written < 1024)
written += snprintf(ld_lib_path+written, 1024-written, ":%s", l);
l = strtok(0, ":");
}
unsetenv("LD_LIBRARY_PATH_PSEUDONATIVE_BACKUP");
}
if (f) {
while ( (l=fgets(line, 1024, f)) ) {
while ( *l == ' ' || *l == '\t' || *l == '\n' ) l++;
if ( *l == '#' || !*l ) continue;
len = strcspn(l, "\t\n ");
if (len) {
l[len] = 0;
glob(l, GLOB_ONLYDIR, 0, &globbuf);
for (i=0; i<globbuf.gl_pathc && written < 1024; i++)
written += snprintf(ld_lib_path+written, 1024-written, ":%s", globbuf.gl_pathv[i]);
globfree (&globbuf);
}
}
fclose(f);
}
if (config_debug)
fprintf(stderr, "pseudonative_handler: setting LD_LIBRARY_PATH: %s\n", ld_lib_path);
setenv("LD_LIBRARY_PATH", ld_lib_path, 1);
}
void write_log(char *type, char **argv)
{
int written = 0;
char line[120] = "";
FILE *f;
if ( (f = fopen("/var/adm/rock-debug/pseudonative.log", "a")) ) {
written += snprintf(line+written, 1024-written, "%s: %s=%s",
type, getenv("ROCK_PKG"), getenv("ROCK_XPKG"));
while (*argv && written < 120)
written += snprintf(line+written, 120-written, " %s", *(argv++));
strcpy(line+100, " ..");
fprintf(f, "%s\n", line);
fclose(f);
}
}
int main(int argc, char **argv)
{
char *cmd, *mycmd, *t;
while ( *argv[1] == '-' && argc > 1 ) {
if ( !strcmp(argv[1], "-d") ) config_debug = 1;
else if ( !strcmp(argv[1], "-r") ) config_remote = 1;
else {
fprintf(stderr, "pseudonative_handler: unknown option %s.\n", argv[1]);
return -1;
}
argv++; argc--;
}
if (argc < 2) {
fprintf(stderr, "pseudonative_handler: arguments missing.\n");
return -1;
}
cmd = basename(argv[1]);
mycmd = malloc(BINDIR_SIZE+strlen(cmd)+10);
t = getenv("LD_LIBRARY_PATH");
if ( t && strcmp(t, LIBDIR) )
setenv("LD_LIBRARY_PATH_PSEUDONATIVE_BACKUP", t, 1);
setenv("LD_LIBRARY_PATH", LIBDIR, 1);
/* The target ld.so.cache would confuse bins for the build host. So
* we unlink it here and generate a LD_LIBRARY_PATH from ld.so.conf
* for bins executed on the remote machine. */
unlink("/etc/ld.so.cache");
sprintf(mycmd, "%s/%s", BINDIR_1, cmd);
if ( !access(mycmd, X_OK) && !config_remote ) {
if (config_debug)
fprintf(stderr, "pseudonative_handler: local: %s\n", mycmd);
argv[1] = mycmd;
write_log("local", argv+1);
execv(mycmd, argv+1);
goto goterror;
}
sprintf(mycmd, "%s/%s", BINDIR_2, cmd);
if ( !access(mycmd, X_OK) && !config_remote ) {
if (config_debug)
fprintf(stderr, "pseudonative_handler: local: %s\n", mycmd);
argv[1] = mycmd;
write_log("local", argv+1);
execv(mycmd, argv+1);
goto goterror;
}
/* FIXME: CMD_ENABLE0 doesn't work correctly yet (see pseudonative_daemon.c).
* So we close fd0 explicitely for some programs here. */
#if 0
if ( !strcmp(cmd, "tput") ) { close(0); open("/dev/null", O_RDONLY); }
#endif
set_ld_lib_path();
if (config_debug)
fprintf(stderr, "pseudonative_handler: remote: %s\n", argv[1]);
write_log("remote", argv+1);
return exec_remote(argv+1);
goterror:
fprintf(stderr, "pseudonative_handler: error calling '%s': %s\n", mycmd, strerror(errno));
return -1;
}