mirror of
https://github.com/gnustep/libs-base.git
synced 2025-04-25 01:31:08 +00:00
git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/base/trunk@3888 72102866-910b-0410-8b05-ffd578937521
443 lines
12 KiB
Objective-C
443 lines
12 KiB
Objective-C
/* Implementation for NSProcessInfo for GNUStep
|
|
Copyright (C) 1995, 1996 Free Software Foundation, Inc.
|
|
|
|
Written by: Georg Tuparev, EMBL & Academia Naturalis,
|
|
Heidelberg, Germany
|
|
Tuparev@EMBL-Heidelberg.de
|
|
|
|
This file is part of the GNUstep Base Library.
|
|
|
|
This library is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU Library General Public
|
|
License as published by the Free Software Foundation; either
|
|
version 2 of the License, or (at your option) any later version.
|
|
|
|
This library is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
Library General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Library General Public
|
|
License along with this library; if not, write to the Free
|
|
Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
*/
|
|
|
|
/*************************************************************************
|
|
* File Name : NSProcessInfo.m
|
|
* Version : 0.6 beta
|
|
* Date : 06-aug-1995
|
|
*************************************************************************
|
|
* Notes :
|
|
* 1) The class functionality depends on the following UNIX functions and
|
|
* global variables: gethostname(), getpid(), and environ. For all system
|
|
* I had the opportunity to test them they are defined and have the same
|
|
* behavior. The same is true for the meaning of argv[0] (process name).
|
|
* 2) The global variable _gnu_sharedProcessInfoObject should NEVER be
|
|
* deallocate during the process runtime. Therefore I implemented a
|
|
* concrete NSProcessInfo subclass (_NSConcreteProcessInfo) with the only
|
|
* purpose to override the autorelease, retain, and release methods.
|
|
* To Do :
|
|
* 1) To test the class on more platforms;
|
|
* 2) To change the format of the string renurned by globallyUniqueString;
|
|
* Bugs : Not known
|
|
* Last update: 08-aug-1995
|
|
* History : 06-aug-1995 - Birth and the first beta version (v. 0.5);
|
|
* 08-aug-1995 - V. 0.6 (tested on NS, SunOS, Solaris, OSF/1
|
|
* The use of the environ global var was changed to more
|
|
* conventional env[] (main function) so now the class could be
|
|
* used on SunOS and Solaris. [GT]
|
|
*************************************************************************
|
|
* Acknowledgments:
|
|
* - Adam Fedor, Andrew McCallum, and Paul Kunz for their help;
|
|
* - To the NEXTSTEP/GNUStep community
|
|
*************************************************************************/
|
|
|
|
#include <config.h>
|
|
|
|
/* One of these two should have MAXHOSTNAMELEN */
|
|
#ifndef __WIN32__
|
|
#include <unistd.h>
|
|
#include <sys/param.h>
|
|
#include <netdb.h>
|
|
#endif /* !__WIN32__ */
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <Foundation/NSString.h>
|
|
#include <Foundation/NSArray.h>
|
|
#include <Foundation/NSSet.h>
|
|
#include <Foundation/NSDictionary.h>
|
|
#include <Foundation/NSDate.h>
|
|
#include <Foundation/NSException.h>
|
|
#include <Foundation/NSProcessInfo.h>
|
|
#include <Foundation/NSAutoreleasePool.h>
|
|
|
|
/* This error message should be called only if the private main function
|
|
* was not executed successfully. This may heppen ONLY if onother library
|
|
* or kit defines its own main function (as gnustep-base does).
|
|
*/
|
|
#define _GNU_MISSING_MAIN_FUNCTION_CALL @"GNUSTEP Internal Error: \
|
|
The private GNUstep function to establish the argv and environment \
|
|
variables was not called. Please report this error to bug-gnustep@gnu.org."
|
|
|
|
/*************************************************************************
|
|
*** _NSConcreteProcessInfo
|
|
*************************************************************************/
|
|
@interface _NSConcreteProcessInfo:NSProcessInfo
|
|
- (id)autorelease;
|
|
- (void)release;
|
|
- (id)retain;
|
|
@end
|
|
|
|
@implementation _NSConcreteProcessInfo
|
|
- (id)autorelease
|
|
{
|
|
return self;
|
|
}
|
|
|
|
- (void)release
|
|
{
|
|
return;
|
|
}
|
|
|
|
- (id)retain
|
|
{
|
|
return self;
|
|
}
|
|
@end
|
|
|
|
/*************************************************************************
|
|
*** NSProcessInfo implementation
|
|
*************************************************************************/
|
|
@implementation NSProcessInfo
|
|
/*************************************************************************
|
|
*** Static global vars
|
|
*************************************************************************/
|
|
// The shared NSProcessInfo instance
|
|
static NSProcessInfo* _gnu_sharedProcessInfoObject = nil;
|
|
|
|
// Host name of the CPU executing the process
|
|
static NSString* _gnu_hostName = nil;
|
|
|
|
// Current process name
|
|
static NSString* _gnu_processName = nil;
|
|
|
|
// Array of NSStrings (argv[1] .. argv[argc-1])
|
|
static NSArray* _gnu_arguments = nil;
|
|
|
|
// Dictionary of environment vars and their values
|
|
static NSMutableDictionary* _gnu_environment = nil;
|
|
|
|
// Array of debug levels set.
|
|
static NSMutableSet* _debug_set = nil;
|
|
|
|
/*************************************************************************
|
|
*** Implementing the Libobjects main function
|
|
*************************************************************************/
|
|
|
|
static void
|
|
_gnu_process_args(int argc, char *argv[], char *env[])
|
|
{
|
|
NSAutoreleasePool *arp = [NSAutoreleasePool new];
|
|
int i;
|
|
|
|
/* Getting the process name */
|
|
_gnu_processName = [[NSString alloc] initWithCString:argv[0]];
|
|
|
|
/* Copy the argument list */
|
|
{
|
|
NSMutableSet *mySet;
|
|
id obj_argv[argc];
|
|
int added = 0;
|
|
|
|
mySet = [[NSMutableSet alloc] init];
|
|
for (i = 0; i < argc; i++)
|
|
{
|
|
NSString *str = [NSString stringWithCString:argv[i]];
|
|
if ([str hasPrefix: @"--GNU-Debug="])
|
|
[mySet addObject: [str substringFromIndex: 12]];
|
|
else
|
|
obj_argv[added++] = str;
|
|
}
|
|
_gnu_arguments = [[NSArray alloc] initWithObjects:obj_argv count:added];
|
|
_debug_set = mySet;
|
|
}
|
|
|
|
/* Copy the evironment list */
|
|
{
|
|
char *cp;
|
|
NSMutableArray *keys = [NSMutableArray new];
|
|
NSMutableArray *values = [NSMutableArray new];
|
|
i = 0;
|
|
while (env[i])
|
|
{
|
|
cp = strchr(env[i],'=');
|
|
if (cp != NULL)
|
|
{
|
|
/* Temporary set *cp to \0 for copying purposes */
|
|
*cp = '\0';
|
|
[keys addObject: [NSString stringWithCString:env[i]]];
|
|
[values addObject: [NSString stringWithCString:cp+1]];
|
|
/* Return the original value of environ[i] */
|
|
*cp = '=';
|
|
}
|
|
i++;
|
|
}
|
|
_gnu_environment = [[NSDictionary alloc] initWithObjects:values
|
|
forKeys:keys];
|
|
/* Do this explicitly, because we probably don't have
|
|
a NSAutoreleasePool initialized yet. */
|
|
[keys release];
|
|
[values release];
|
|
}
|
|
[arp release];
|
|
}
|
|
|
|
/* Place the _gnu_process_args function in the _libc_subinit section so
|
|
that it automatically gets called before main with the argument and
|
|
environment pointers. FIXME: Would like to do something similar
|
|
for other formats besides ELF. */
|
|
#if (!defined(__FreeBSD__)) && defined(SYS_AUTOLOAD)
|
|
#ifdef linux
|
|
|
|
/* Under linux the functions in __libc_subinit are called before the
|
|
* global constructiors, therefore, we cannot send methods to any objects
|
|
*/
|
|
|
|
static int _gnu_noobjc_argc;
|
|
static char **_gnu_noobjc_argv;
|
|
static char **_gnu_noobjc_env;
|
|
|
|
static void
|
|
_gnu_process_noobjc_args(int argc, char *argv[], char *env[])
|
|
{
|
|
int i;
|
|
|
|
/* We have to copy these in case the main() modifies their values
|
|
* somehow before we get a change to use them
|
|
*/
|
|
|
|
_gnu_noobjc_argc = argc;
|
|
i = 0;
|
|
while (argv[i])
|
|
i++;
|
|
_gnu_noobjc_argv = malloc(sizeof(char *)*(i+1));
|
|
if (_gnu_noobjc_argv == NULL)
|
|
goto error;
|
|
i = 0;
|
|
while (*argv)
|
|
{
|
|
_gnu_noobjc_argv[i] = malloc(strlen(*argv)+1);
|
|
if (_gnu_noobjc_argv[i] == NULL)
|
|
goto error;
|
|
strcpy(_gnu_noobjc_argv[i],*argv);
|
|
argv++;
|
|
i++;
|
|
}
|
|
_gnu_noobjc_argv[i] = 0;
|
|
i = 0;
|
|
while (env[i])
|
|
i++;
|
|
_gnu_noobjc_env = malloc(sizeof(char *)*(i+1));
|
|
if (_gnu_noobjc_env == NULL)
|
|
goto error;
|
|
i = 0;
|
|
while(*env)
|
|
{
|
|
_gnu_noobjc_env[i] = malloc(strlen(*env)+1);
|
|
if (_gnu_noobjc_env[i] == NULL)
|
|
goto error;
|
|
strcpy(_gnu_noobjc_env[i],*env);
|
|
env++;
|
|
i++;
|
|
}
|
|
_gnu_noobjc_env[i] = 0;
|
|
return;
|
|
|
|
error:
|
|
fputs("malloc() error when starting gstep-base\n", stderr);
|
|
abort();
|
|
}
|
|
|
|
static void _gnu_noobjc_free_vars(void)
|
|
{
|
|
char **p;
|
|
|
|
p = _gnu_noobjc_argv;
|
|
while (*p)
|
|
{
|
|
free(*p);
|
|
p++;
|
|
}
|
|
free(_gnu_noobjc_argv);
|
|
_gnu_noobjc_argv = 0;
|
|
|
|
p = _gnu_noobjc_env;
|
|
while (*p)
|
|
{
|
|
free(*p);
|
|
p++;
|
|
}
|
|
free(_gnu_noobjc_env);
|
|
_gnu_noobjc_env = 0;
|
|
}
|
|
|
|
void * __gnustep_base_subinit_args__
|
|
__attribute__ ((section ("__libc_subinit"))) = &(_gnu_process_noobjc_args);
|
|
|
|
+ (void) initialize
|
|
{
|
|
if (!_gnu_processName && !_gnu_arguments && !_gnu_environment)
|
|
{
|
|
NSAssert(_gnu_noobjc_argv && _gnu_noobjc_env,
|
|
_GNU_MISSING_MAIN_FUNCTION_CALL);
|
|
_gnu_process_args(_gnu_noobjc_argc,_gnu_noobjc_argv,_gnu_noobjc_env);
|
|
_gnu_noobjc_free_vars();
|
|
}
|
|
}
|
|
|
|
#else
|
|
static void * __gnustep_base_subinit_args__
|
|
__attribute__ ((section ("_libc_subinit"))) = &(_gnu_process_args);
|
|
#endif /* linux */
|
|
|
|
#else
|
|
#ifdef __MINGW32__
|
|
/* For Windows32API Library, we know the global variables */
|
|
extern int __argc;
|
|
extern char** __argv;
|
|
extern char** _environ;
|
|
|
|
+ (void)initialize
|
|
{
|
|
if (self == [NSProcessInfo class])
|
|
_gnu_process_args(__argc, __argv, _environ);
|
|
}
|
|
|
|
#else
|
|
#undef main
|
|
int main(int argc, char *argv[], char *env[])
|
|
{
|
|
#ifdef __WIN32__
|
|
WSADATA lpWSAData;
|
|
|
|
// Initialize Windows Sockets
|
|
if (WSAStartup(MAKEWORD(1,1), &lpWSAData))
|
|
{
|
|
printf("Could not startup Windows Sockets\n");
|
|
exit(1);
|
|
}
|
|
#endif /* __WIN32__ */
|
|
|
|
#ifdef __MS_WIN32__
|
|
_MB_init_runtime();
|
|
#endif /* __MS_WIN32__ */
|
|
|
|
_gnu_process_args(argc, argv, env);
|
|
|
|
/* Call the user defined main function */
|
|
return gnustep_base_user_main (argc, argv, env);
|
|
}
|
|
#endif /* __MINGW32__ */
|
|
#endif /* __ELF__ */
|
|
|
|
/*************************************************************************
|
|
*** Getting an NSProcessInfo Object
|
|
*************************************************************************/
|
|
+ (NSProcessInfo *)processInfo
|
|
{
|
|
// Check if the main() function was successfully called
|
|
// We can't use NSAssert, which calls NSLog, which calls NSProcessInfo...
|
|
if (!(_gnu_processName && _gnu_arguments && _gnu_environment))
|
|
{
|
|
_NSLog_printf_handler(_GNU_MISSING_MAIN_FUNCTION_CALL);
|
|
[NSException raise: NSInternalInconsistencyException
|
|
format: _GNU_MISSING_MAIN_FUNCTION_CALL];
|
|
}
|
|
|
|
if (!_gnu_sharedProcessInfoObject)
|
|
_gnu_sharedProcessInfoObject = [[_NSConcreteProcessInfo alloc] init];
|
|
|
|
return _gnu_sharedProcessInfoObject;
|
|
}
|
|
|
|
/*************************************************************************
|
|
*** Returning Process Information
|
|
*************************************************************************/
|
|
- (NSArray *)arguments
|
|
{
|
|
return _gnu_arguments;
|
|
}
|
|
|
|
- (NSMutableSet*) debugSet
|
|
{
|
|
return _debug_set;
|
|
}
|
|
|
|
- (NSDictionary *)environment
|
|
{
|
|
return _gnu_environment;
|
|
}
|
|
|
|
- (NSString *)hostName
|
|
{
|
|
if (!_gnu_hostName)
|
|
{
|
|
char hn[MAXHOSTNAMELEN];
|
|
|
|
gethostname(hn, MAXHOSTNAMELEN);
|
|
_gnu_hostName = [[NSString alloc] initWithCString:hn];
|
|
}
|
|
return _gnu_hostName;
|
|
}
|
|
|
|
- (NSString *)processName
|
|
{
|
|
return _gnu_processName;
|
|
}
|
|
|
|
- (NSString *)globallyUniqueString
|
|
{
|
|
// $$$ The format of the string is not specified by the OpenStep
|
|
// specification. It could be useful to change this format after
|
|
// NeXTSTEP release 4.0 comes out.
|
|
return [NSString stringWithFormat:@"%s:%d:[%s]",
|
|
[[self hostName] cString],
|
|
(int)getpid(),
|
|
[[[NSDate date] description] cString]];
|
|
}
|
|
|
|
/*************************************************************************
|
|
*** Specifying a Process Name
|
|
*************************************************************************/
|
|
- (void)setProcessName:(NSString *)newName
|
|
{
|
|
if (newName && [newName length]) {
|
|
[_gnu_processName autorelease];
|
|
_gnu_processName = [newName copyWithZone:[self zone]];
|
|
}
|
|
return;
|
|
}
|
|
|
|
@end
|
|
|
|
/*
|
|
* Function for rapid testing to see if a debug level is set.
|
|
*/
|
|
BOOL GSDebugSet(NSString *val)
|
|
{
|
|
static SEL debugSel = @selector(member:);
|
|
static IMP debugImp = 0;
|
|
|
|
if (debugImp == 0)
|
|
{
|
|
if (_debug_set == nil)
|
|
{
|
|
[[NSProcessInfo processInfo] debugSet];
|
|
}
|
|
debugImp = [_debug_set methodForSelector: debugSel];
|
|
}
|
|
return ((*debugImp)(_debug_set, debugSel, val) == val) ? YES : NO;
|
|
}
|
|
|