mirror of
https://github.com/adtools/clib2.git
synced 2025-12-08 14:59:05 +00:00
- Replaced ISO Latin 1 code #160 with a plain blank space (ASCII code #32) where necessary git-svn-id: file:///Users/olsen/Code/migration-svn-zu-git/logical-line-staging/clib2/trunk@15305 87f5fb63-7c3d-0410-a384-fd976d0f7a62
245 lines
6.0 KiB
C
245 lines
6.0 KiB
C
/*
|
|
* $Id: stdlib_mktemp.c,v 1.8 2006-01-08 12:04:26 obarthel Exp $
|
|
*
|
|
* :ts=4
|
|
*
|
|
* Portable ISO 'C' (1994) runtime library for the Amiga computer
|
|
* Copyright (c) 2002-2015 by Olaf Barthel <obarthel (at) gmx.net>
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
*
|
|
* - Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
*
|
|
* - Neither the name of Olaf Barthel nor the names of contributors
|
|
* may be used to endorse or promote products derived from this
|
|
* software without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
|
|
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
#ifndef _STDLIB_NULL_POINTER_CHECK_H
|
|
#include "stdlib_null_pointer_check.h"
|
|
#endif /* _STDLIB_NULL_POINTER_CHECK_H */
|
|
|
|
/****************************************************************************/
|
|
|
|
#ifndef _STDLIB_HEADERS_H
|
|
#include "stdlib_headers.h"
|
|
#endif /* _STDLIB_HEADERS_H */
|
|
|
|
/****************************************************************************/
|
|
|
|
#include <time.h>
|
|
#include <dos.h>
|
|
|
|
/****************************************************************************/
|
|
|
|
/* The following is not part of the ISO 'C' (1994) standard. */
|
|
|
|
/****************************************************************************/
|
|
|
|
char *
|
|
mktemp(char * name_template)
|
|
{
|
|
#if defined(UNIX_PATH_SEMANTICS)
|
|
struct name_translation_info name_template_nti;
|
|
#endif /* UNIX_PATH_SEMANTICS */
|
|
char * test_name;
|
|
struct Process * this_process;
|
|
APTR old_window_pointer;
|
|
char * result = NULL;
|
|
size_t template_offset;
|
|
size_t template_len;
|
|
size_t name_len;
|
|
size_t offset;
|
|
time_t now;
|
|
ULONG pseudo_random_number;
|
|
BPTR lock;
|
|
size_t i;
|
|
|
|
ENTER();
|
|
|
|
SHOWSTRING(name_template);
|
|
|
|
assert(name_template != NULL);
|
|
|
|
if(__check_abort_enabled)
|
|
__check_abort();
|
|
|
|
#if defined(CHECK_FOR_NULL_POINTERS)
|
|
{
|
|
if(name_template == NULL)
|
|
{
|
|
SHOWMSG("invalid name template");
|
|
|
|
__set_errno(EFAULT);
|
|
goto out;
|
|
}
|
|
}
|
|
#endif /* CHECK_FOR_NULL_POINTERS */
|
|
|
|
this_process = (struct Process *)FindTask(NULL);
|
|
|
|
SHOWSTRING(name_template);
|
|
|
|
/* So, how long is that name template? */
|
|
name_len = strlen(name_template);
|
|
if(name_len == 0)
|
|
{
|
|
SHOWMSG("invalid name template");
|
|
|
|
__set_errno(EINVAL);
|
|
goto out;
|
|
}
|
|
|
|
/* Find out how many trailing 'X' characters there are in
|
|
the template. There should be at least 6. We also want
|
|
to know where to find the first 'X' and how many of the
|
|
'X' characters there are. */
|
|
template_offset = 0;
|
|
template_len = 0;
|
|
|
|
for(i = 0 ; i < name_len ; i++)
|
|
{
|
|
assert( name_len >= (i + 1) );
|
|
|
|
offset = name_len - (i + 1);
|
|
|
|
if(name_template[offset] != 'X')
|
|
break;
|
|
|
|
template_offset = offset;
|
|
template_len++;
|
|
}
|
|
|
|
SHOWVALUE(template_offset);
|
|
SHOWVALUE(template_len);
|
|
|
|
if(template_len == 0)
|
|
{
|
|
SHOWMSG("invalid name template");
|
|
|
|
__set_errno(EINVAL);
|
|
goto out;
|
|
}
|
|
|
|
/* Generate a pseudo-random number from the current time and
|
|
the address of the current process. */
|
|
time(&now);
|
|
|
|
pseudo_random_number = (ULONG)now + (ULONG)this_process;
|
|
|
|
/* Fill the template 'X' characters with letters made up by
|
|
converting the pseudo-random number. */
|
|
for(i = 0 ; i < template_len ; i++)
|
|
{
|
|
name_template[template_offset + i] = 'A' + (pseudo_random_number % 26);
|
|
|
|
/* One more letter taken; if we run out of letters,
|
|
cook up another pseudo-random number. */
|
|
pseudo_random_number = (pseudo_random_number / 26);
|
|
if(pseudo_random_number == 0)
|
|
{
|
|
time(&now);
|
|
|
|
pseudo_random_number = (ULONG)now;
|
|
}
|
|
}
|
|
|
|
SHOWSTRING(name_template);
|
|
|
|
/* Now check if the name we picked is unique. If not, make another name. */
|
|
while(TRUE)
|
|
{
|
|
if(__check_abort_enabled)
|
|
__check_abort();
|
|
|
|
D(("checking '%s'",name_template));
|
|
|
|
test_name = name_template;
|
|
|
|
/* If necessary, quickly translate the semantics of the file name
|
|
we cooked up above. */
|
|
#if defined(UNIX_PATH_SEMANTICS)
|
|
{
|
|
if(__unix_path_semantics)
|
|
{
|
|
if(__translate_unix_to_amiga_path_name((char const **)&test_name,&name_template_nti) != 0)
|
|
goto out;
|
|
|
|
if(name_template_nti.is_root)
|
|
{
|
|
__set_errno(EACCES);
|
|
goto out;
|
|
}
|
|
}
|
|
}
|
|
#endif /* UNIX_PATH_SEMANTICS */
|
|
|
|
/* Turn off DOS error requesters. */
|
|
old_window_pointer = __set_process_window((APTR)-1);
|
|
|
|
/* Does this object exist already? */
|
|
PROFILE_OFF();
|
|
lock = Lock(test_name,SHARED_LOCK);
|
|
PROFILE_ON();
|
|
|
|
/* Restore DOS requesters. */
|
|
__set_process_window(old_window_pointer);
|
|
|
|
if(lock == ZERO)
|
|
{
|
|
/* If the object does not exist yet then we
|
|
* are finished.
|
|
*/
|
|
if(IoErr() == ERROR_OBJECT_NOT_FOUND)
|
|
{
|
|
result = name_template;
|
|
break;
|
|
}
|
|
|
|
/* Looks like a serious error. */
|
|
__set_errno(__translate_io_error_to_errno(IoErr()));
|
|
goto out;
|
|
}
|
|
|
|
/* OK, so it exists already. Start over... */
|
|
|
|
PROFILE_OFF();
|
|
UnLock(lock);
|
|
PROFILE_ON();
|
|
|
|
/* Change one letter; if that 'overflows', start
|
|
over with 'A' and move on to the next position. */
|
|
for(i = 0 ; i < template_len ; i++)
|
|
{
|
|
name_template[template_offset + i]++;
|
|
if(name_template[template_offset + i] <= 'Z')
|
|
break;
|
|
|
|
name_template[template_offset + i] = 'A';
|
|
}
|
|
}
|
|
|
|
SHOWSTRING(name_template);
|
|
|
|
out:
|
|
|
|
RETURN(result);
|
|
return(result);
|
|
}
|