/**************************************************************************/ /* */ /* Copyright (c) 2001, 2011 NoMachine (http://www.nomachine.com) */ /* Copyright (c) 2008-2014 Oleksandr Shneyder <o.shneyder@phoca-gmbh.de> */ /* Copyright (c) 2011-2016 Mike Gabriel <mike.gabriel@das-netzwerkteam.de>*/ /* Copyright (c) 2014-2016 Mihai Moldovan <ionic@ionic.de> */ /* Copyright (c) 2014-2016 Ulrich Sibiller <uli42@gmx.de> */ /* Copyright (c) 2015-2016 Qindel Group (http://www.qindel.com) */ /* */ /* NXAGENT, NX protocol compression and NX extensions to this software */ /* are copyright of the aforementioned persons and companies. */ /* */ /* Redistribution and use of the present software is allowed according */ /* to terms specified in the file LICENSE which comes in the source */ /* distribution. */ /* */ /* All rights reserved. */ /* */ /* NOTE: This software has received contributions from various other */ /* contributors, only the core maintainers and supporters are listed as */ /* copyright holders. Please contact us, if you feel you should be listed */ /* as copyright holder, as well. */ /* */ /**************************************************************************/ #include <stdio.h> #include <string.h> #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h> #include <errno.h> #include <unistd.h> #include "os.h" #include "scrnintstr.h" #include "Agent.h" #include "Xlib.h" #include "Xproto.h" #include "Error.h" #include "Args.h" /* * Set here the required log level. */ #define PANIC #define WARNING #undef TEST #undef DEBUG /* * Duplicated stderr file descriptor. */ static int nxagentStderrDup = -1; /* * Saved stderr file descriptor. */ static int nxagentStderrBackup = -1; /* * Clients log file descriptor. */ static int nxagentClientsLog = -1; #define DEFAULT_STRING_LENGTH 256 /* * Clients log file name. */ char nxagentClientsLogName[NXAGENTCLIENTSLOGNAMELENGTH] = { 0 }; /* * User's home. */ static char nxagentHomeDir[DEFAULT_STRING_LENGTH] = { 0 }; /* * NX root directory. */ static char nxagentRootDir[DEFAULT_STRING_LENGTH] = { 0 }; /* * Session log Directory. */ static char nxagentSessionDir[DEFAULT_STRING_LENGTH] = { 0 }; void nxagentGetClientsPath(void); static int nxagentPrintError(Display *dpy, XErrorEvent *event, FILE *fp); /* declare an error handler that does not exit when an error * event is caught. */ int nxagentErrorHandler(Display *dpy, XErrorEvent *event) { if (nxagentVerbose == 1) { nxagentPrintError(dpy, event, stderr); } return 0; } /* copied from XlibInt.c */ /* extension stuff roughly commented out */ /* FIXME: why? What's wrong with printing extension stuff? We could drop this in favour of _XprintDefaultError then! */ static int nxagentPrintError(dpy, event, fp) Display *dpy; XErrorEvent *event; FILE *fp; { char buffer[BUFSIZ]; char mesg[BUFSIZ]; char number[32]; char *mtype = "XlibMessage"; /* register _XExtension *ext = (_XExtension *)NULL; _XExtension *bext = (_XExtension *)NULL; */ XGetErrorText(dpy, event->error_code, buffer, BUFSIZ); XGetErrorDatabaseText(dpy, mtype, "XError", "X Error", mesg, BUFSIZ); (void) fprintf(fp, "%s: %s\n ", mesg, buffer); XGetErrorDatabaseText(dpy, mtype, "MajorCode", "Request Major code %d", mesg, BUFSIZ); (void) fprintf(fp, mesg, event->request_code); if (event->request_code < 128) { snprintf(number, sizeof(number), "%d", event->request_code); XGetErrorDatabaseText(dpy, "XRequest", number, "", buffer, BUFSIZ); } else { /* for (ext = dpy->ext_procs; ext && (ext->codes.major_opcode != event->request_code); ext = ext->next) ; if (ext) strncpy(buffer, ext->name, BUFSIZ); else */ buffer[0] = '\0'; } (void) fprintf(fp, " (%s)\n", buffer); if (event->request_code >= 128) { XGetErrorDatabaseText(dpy, mtype, "MinorCode", "Request Minor code %d", mesg, BUFSIZ); fputs(" ", fp); (void) fprintf(fp, mesg, event->minor_code); /* if (ext) { snprintf(mesg, sizeof(mesg), "%s.%d", ext->name, event->minor_code); XGetErrorDatabaseText(dpy, "XRequest", mesg, "", buffer, BUFSIZ); (void) fprintf(fp, " (%s)", buffer); } */ fputs("\n", fp); } if (event->error_code >= 128) { /* kludge, try to find the extension that caused it */ buffer[0] = '\0'; /* for (ext = dpy->ext_procs; ext; ext = ext->next) { if (ext->error_string) (*ext->error_string)(dpy, event->error_code, &ext->codes, buffer, BUFSIZ); if (buffer[0]) { bext = ext; break; } if (ext->codes.first_error && ext->codes.first_error < (int)event->error_code && (!bext || ext->codes.first_error > bext->codes.first_error)) bext = ext; } if (bext) snprintf(buffer, sizeof(buffer), "%s.%d", bext->name, event->error_code - bext->codes.first_error); else */ strcpy(buffer, "Value"); XGetErrorDatabaseText(dpy, mtype, buffer, "", mesg, BUFSIZ); if (mesg[0]) { fputs(" ", fp); (void) fprintf(fp, mesg, event->resourceid); fputs("\n", fp); } /* let extensions try to print the values */ /* for (ext = dpy->ext_procs; ext; ext = ext->next) { if (ext->error_values) (*ext->error_values)(dpy, event, fp); } */ } else if ((event->error_code == BadWindow) || (event->error_code == BadPixmap) || (event->error_code == BadCursor) || (event->error_code == BadFont) || (event->error_code == BadDrawable) || (event->error_code == BadColor) || (event->error_code == BadGC) || (event->error_code == BadIDChoice) || (event->error_code == BadValue) || (event->error_code == BadAtom)) { if (event->error_code == BadValue) XGetErrorDatabaseText(dpy, mtype, "Value", "Value 0x%x", mesg, BUFSIZ); else if (event->error_code == BadAtom) XGetErrorDatabaseText(dpy, mtype, "AtomID", "AtomID 0x%x", mesg, BUFSIZ); else XGetErrorDatabaseText(dpy, mtype, "ResourceID", "ResourceID 0x%x", mesg, BUFSIZ); fputs(" ", fp); (void) fprintf(fp, mesg, event->resourceid); fputs("\n", fp); } XGetErrorDatabaseText(dpy, mtype, "ErrorSerial", "Error Serial #%d", mesg, BUFSIZ); fputs(" ", fp); (void) fprintf(fp, mesg, event->serial); /* XGetErrorDatabaseText(dpy, mtype, "CurrentSerial", "Current Serial #%d", mesg, BUFSIZ); fputs("\n ", fp); (void) fprintf(fp, mesg, dpy->request); */ fputs("\n", fp); if (event->error_code == BadImplementation) return 0; return 1; } int nxagentExitHandler(const char *message) { FatalError("%s", message); return 0; } void nxagentOpenClientsLogFile(void) { if (*nxagentClientsLogName == '\0') { nxagentGetClientsPath(); } if (nxagentClientsLogName != NULL && *nxagentClientsLogName != '\0') { nxagentClientsLog = open(nxagentClientsLogName, O_RDWR | O_CREAT | O_APPEND, 0600); if (nxagentClientsLog == -1) { fprintf(stderr, "Warning: Failed to open clients log. Error is %d '%s'.\n", errno, strerror(errno)); } } else { #ifdef TEST fprintf(stderr, "nxagentOpenClientsLogFile: Cannot open clients log file. The path does not exist.\n"); #endif } } void nxagentCloseClientsLogFile(void) { close(nxagentClientsLog); } void nxagentStartRedirectToClientsLog(void) { nxagentOpenClientsLogFile(); if (nxagentClientsLog != -1) { if (nxagentStderrBackup == -1) { nxagentStderrBackup = dup(STDERR_FILENO); } if (nxagentStderrBackup != -1) { nxagentStderrDup = dup2(nxagentClientsLog, STDERR_FILENO); if (nxagentStderrDup == -1) { fprintf(stderr, "Warning: Failed to redirect stderr. Error is %d '%s'.\n", errno, strerror(errno)); } } else { fprintf(stderr, "Warning: Failed to backup stderr. Error is %d '%s'.\n", errno, strerror(errno)); } } } void nxagentEndRedirectToClientsLog(void) { if (nxagentStderrBackup != -1) { nxagentStderrDup = dup2(nxagentStderrBackup, STDERR_FILENO); if (nxagentStderrDup == -1) { fprintf(stderr, "Warning: Failed to restore stderr. Error is %d '%s'.\n", errno, strerror(errno)); } } nxagentCloseClientsLogFile(); } char *nxagentGetHomePath(void) { char *homeEnv; char *homePath; if (*nxagentHomeDir == '\0') { /* * Check the NX_HOME environment. */ homeEnv = getenv("NX_HOME"); if (homeEnv == NULL || *homeEnv == '\0') { #ifdef TEST fprintf(stderr, "nxagentGetHomePath: No environment for NX_HOME.\n"); #endif homeEnv = getenv("HOME"); if (homeEnv == NULL || *homeEnv == '\0') { #ifdef PANIC fprintf(stderr, "nxagentGetHomePath: PANIC! No environment for HOME.\n"); #endif return NULL; } } if (strlen(homeEnv) > DEFAULT_STRING_LENGTH - 1) { #ifdef PANIC fprintf(stderr, "nxagentGetHomePath: PANIC! Invalid value for the NX " "home directory '%s'.\n", homeEnv); #endif } snprintf(nxagentHomeDir, DEFAULT_STRING_LENGTH, "%s", homeEnv); #ifdef TEST fprintf(stderr, "nxagentGetHomePath: Assuming NX user's home directory '%s'.\n", nxagentHomeDir); #endif } homePath = strdup(nxagentHomeDir); if (homePath == NULL) { #ifdef PANIC fprintf(stderr, "nxagentGetHomePath: PANIC! Can't allocate memory for the home path.\n"); #endif return NULL; } return homePath; } char *nxagentGetRootPath(void) { char *rootEnv; char *homeEnv; char *rootPath; struct stat dirStat; if (*nxagentRootDir == '\0') { /* * Check the NX_ROOT environment. */ rootEnv = getenv("NX_ROOT"); if (rootEnv == NULL || *rootEnv == '\0') { #ifdef TEST fprintf(stderr, "nxagentGetRootPath: WARNING! No environment for NX_ROOT.\n"); #endif /* * We will determine the root NX directory * based on the NX_HOME or HOME directory * settings. */ homeEnv = nxagentGetHomePath(); if (homeEnv == NULL) { return NULL; } if (strlen(homeEnv) > DEFAULT_STRING_LENGTH - strlen("/.nx") - 1) { #ifdef PANIC fprintf(stderr, "nxagentGetRootPath: PANIC! Invalid value for the NX " "home directory '%s'.\n", homeEnv); #endif free(homeEnv); return NULL; } #ifdef TEST fprintf(stderr, "nxagentGetRootPath: Assuming NX root directory in '%s'.\n", homeEnv); #endif snprintf(nxagentRootDir, DEFAULT_STRING_LENGTH, "%s/.nx", homeEnv); free(homeEnv); /* * Create the NX root directory. */ if ((stat(nxagentRootDir, &dirStat) == -1) && (errno == ENOENT)) { if (mkdir(nxagentRootDir, 0777) < 0 && (errno != EEXIST)) { #ifdef PANIC fprintf(stderr, "nxagentGetRootPath: PANIC! Can't create directory '%s'. Error is %d '%s'.\n", nxagentRootDir, errno, strerror(errno)); #endif return NULL; } } } else { if (strlen(rootEnv) > DEFAULT_STRING_LENGTH - 1) { #ifdef PANIC fprintf(stderr, "nxagentGetRootPath: PANIC! Invalid value for the NX root directory '%s'.\n", rootEnv); #endif return NULL; } snprintf(nxagentRootDir, DEFAULT_STRING_LENGTH, "%s", rootEnv); } #ifdef TEST fprintf(stderr, "nxagentGetRootPath: Assuming NX root directory '%s'.\n", nxagentRootDir); #endif } rootPath = strdup(nxagentRootDir); if (rootPath == NULL) { #ifdef PANIC fprintf(stderr, "nxagentGetRootPath: Can't allocate memory for the root path.\n"); #endif return NULL; } return rootPath; } char *nxagentGetSessionPath(void) { char *rootPath; char *sessionPath; struct stat dirStat; if (*nxagentSessionDir == '\0') { /* * If nxagentSessionId does not exist we * assume that the sessionPath cannot be * realized and do not use the clients * log file. */ if (*nxagentSessionId == '\0') { #ifdef TEST fprintf(stderr, "nxagentGetSessionPath: Session id does not exist. Assuming session path NULL.\n"); #endif return NULL; } rootPath = nxagentGetRootPath(); if (rootPath == NULL) { return NULL; } /* FIXME: necessary? */ snprintf(nxagentSessionDir, DEFAULT_STRING_LENGTH, "%s", rootPath); if (strlen(nxagentSessionDir) + strlen("/C-") + strlen(nxagentSessionId) > DEFAULT_STRING_LENGTH - 1) { #ifdef PANIC fprintf(stderr, "nxagentGetSessionPath: PANIC!: Invalid value for the NX session directory '%s'.\n", nxagentSessionDir); #endif free(rootPath); return NULL; } snprintf(nxagentSessionDir, DEFAULT_STRING_LENGTH, "%s/C-%s", rootPath, nxagentSessionId); free(rootPath); if ((stat(nxagentSessionDir, &dirStat) == -1) && (errno == ENOENT)) { if (mkdir(nxagentSessionDir, 0777) < 0 && (errno != EEXIST)) { #ifdef PANIC fprintf(stderr, "nxagentGetSessionPath: PANIC! Can't create directory '%s'. Error is %d '%s'.\n", nxagentSessionDir, errno, strerror(errno)); #endif return NULL; } } #ifdef TEST fprintf(stderr, "nxagentGetSessionPath: NX session is '%s'.\n", nxagentSessionDir); #endif } sessionPath = strdup(nxagentSessionDir); if (sessionPath == NULL) { #ifdef PANIC fprintf(stderr, "nxagentGetSessionPath:: PANIC! Can't allocate memory for the session path.\n"); #endif return NULL; } return sessionPath; } void nxagentGetClientsPath(void) { if (*nxagentClientsLogName == '\0') { char *sessionPath = nxagentGetSessionPath(); if (sessionPath == NULL) { return; } if (strlen(sessionPath) + strlen("/clients") > NXAGENTCLIENTSLOGNAMELENGTH - 1) { #ifdef PANIC fprintf(stderr, "nxagentGetClientsPath: PANIC! Invalid value for the NX clients Log File Path ''.\n"); #endif free(sessionPath); return; } snprintf(nxagentClientsLogName, NXAGENTCLIENTSLOGNAMELENGTH, "%s/clients", sessionPath); free(sessionPath); } return; }