1
0
mirror of https://github.com/adtools/clib2.git synced 2025-12-08 14:59:05 +00:00

21 Commits

Author SHA1 Message Date
obarthel
4c54ee3f2d Version bump 2016-12-04 11:14:17 +01:00
obarthel
f491e38b38 Added code to temporarily disable profiling 2016-12-04 11:14:00 +01:00
obarthel
734ce4c1a9 Added code to temporarily disable profiling 2016-12-04 11:13:37 +01:00
obarthel
ce345df9da Hide warnings for deprecated functions 2016-12-04 11:13:00 +01:00
obarthel
5e0fa78d61 Only builds with GCC now 2016-12-04 11:12:42 +01:00
obarthel
bc3e19abe5 Only uses MEMF_PRIVATE on OS4 now 2016-12-04 11:12:32 +01:00
obarthel
29e02775fb Cleaned up the build rules, added missing files 2016-12-04 11:11:53 +01:00
obarthel
5cb27db203 Slab allocator is exercised much more 2016-12-04 11:11:28 +01:00
obarthel
4fc1b13945 Added a rogue malloc/free pair 2016-12-04 11:11:05 +01:00
obarthel
8beaabac4f Removed unused local variable 2016-12-04 11:10:46 +01:00
obarthel
271572ed56 Fixed so that it builds correctly with SAS/C again 2016-12-04 11:10:33 +01:00
obarthel
e0feef8932 Moved __CXV54 into sas_cxv.asm
sas_cxv54.asm is no longer needed.
2016-12-04 11:10:13 +01:00
obarthel
07259ed7eb This is needed when building with math=ieee option 2016-12-04 11:09:24 +01:00
obarthel
3203fcf96a Removed unused result variable 2016-12-04 11:08:54 +01:00
obarthel
bfba44bf83 Rewritten to use setjmp()/longjmp()
Also contains new code which prints the number of times a slab was reused.
2016-12-04 11:08:41 +01:00
obarthel
35434bdedc Updated to build correctly with SAS/C again 2016-12-04 11:07:40 +01:00
obarthel
17ba18c731 Added code which temporarily disables profiling 2016-12-04 11:06:50 +01:00
obarthel
78a8c7655e Added __decay_unused_slabs() 2016-12-04 11:05:50 +01:00
obarthel
184a127860 Added stdlib_decay_unused_slabs() 2016-12-04 11:05:30 +01:00
obarthel
5617c0eacf Slab allocator update
Unused slabs which get recycled are no longer reinitialized from scratch if their chunk size matches what the allocator needed. If the chunk size matches, the list of available chunks is left unchanged, and just the various counters are reset.

Added __get_slab_stats() function.

Added support for global __slab_purge_threshold tuning variable.

Added a short test program for the slab allocator.

The malloc-test program was linked against the wrong object file in GNUmakefile.68k. Fixed.
2016-11-27 15:53:40 +01:00
Olaf Barthel
ac710b333e Accidentally omitted from version 1.211 2016-11-24 09:45:35 +01:00
44 changed files with 1973 additions and 1073 deletions

View File

@@ -327,6 +327,7 @@ C_LIB = \
stdlib_exit.o \ stdlib_exit.o \
stdlib_free.o \ stdlib_free.o \
stdlib_free_unused_slabs.o \ stdlib_free_unused_slabs.o \
stdlib_decay_unused_slabs.o \
stdlib_getdefstacksize.o \ stdlib_getdefstacksize.o \
stdlib_getenv.o \ stdlib_getenv.o \
stdlib_getmemstats.o \ stdlib_getmemstats.o \
@@ -334,6 +335,7 @@ C_LIB = \
stdlib_get_errno.o \ stdlib_get_errno.o \
stdlib_get_slab_usage.o \ stdlib_get_slab_usage.o \
stdlib_get_slab_allocations.o \ stdlib_get_slab_allocations.o \
stdlib_get_slab_stats.o \
stdlib_isresident.o \ stdlib_isresident.o \
stdlib_labs.o \ stdlib_labs.o \
stdlib_llabs.o \ stdlib_llabs.o \
@@ -373,6 +375,7 @@ C_LIB = \
stdlib_showerror.o \ stdlib_showerror.o \
stdlib_slab.o \ stdlib_slab.o \
stdlib_slab_max_size.o \ stdlib_slab_max_size.o \
stdlib_slab_purge_threshold.o \
stdlib_srand.o \ stdlib_srand.o \
stdlib_stacksize.o \ stdlib_stacksize.o \
stdlib_stack_usage.o \ stdlib_stack_usage.o \
@@ -1124,25 +1127,31 @@ $(LIBC_OBJS)/stdlib_getdefstacksize.o : stdlib_getdefstacksize.c stdlib_gcc_help
$(LIBC_OBJS)/stdlib_shell_escape.o : stdlib_shell_escape.c stdlib_gcc_help.h $(LIBC_OBJS)/stdlib_shell_escape.o : stdlib_shell_escape.c stdlib_gcc_help.h
$(LIBC_OBJS)/stdlib_alloca.o : stdlib_alloca.c stdlib_memory.h $(LIBC_OBJS)/stdlib_alloca.o : stdlib_alloca.c stdlib_memory.h include/stdlib.h
$(LIBC_OBJS)/stdlib_calloc.o : stdlib_calloc.c stdlib_memory.h $(LIBC_OBJS)/stdlib_calloc.o : stdlib_calloc.c stdlib_memory.h include/stdlib.h
$(LIBC_OBJS)/stdlib_free.o : stdlib_free.c stdlib_memory.h $(LIBC_OBJS)/stdlib_free.o : stdlib_free.c stdlib_memory.h include/stdlib.h
$(LIBC_OBJS)/stdlib_malloc.o : stdlib_malloc.c stdlib_memory.h $(LIBC_OBJS)/stdlib_malloc.o : stdlib_malloc.c stdlib_memory.h include/stdlib.h
$(LIBC_OBJS)/stdlib_slab.o : stdlib_slab.c stdlib_memory.h $(LIBC_OBJS)/stdlib_slab.o : stdlib_slab.c stdlib_memory.h include/stdlib.h
$(LIBC_OBJS)/stdlib_free_unused_slabs.o : stdlib_free_unused_slabs.c stdlib_memory.h $(LIBC_OBJS)/stdlib_slab_purge_threshold.o : stdlib_slab_purge_threshold.c stdlib_memory.h include/stdlib.h
$(LIBC_OBJS)/stdlib_get_slab_usage.o : stdlib_get_slab_usage.c stdlib_memory.h $(LIBC_OBJS)/stdlib_get_slab_stats.o : stdlib_get_slab_stats.c stdlib_memory.h include/stdlib.h
$(LIBC_OBJS)/stdlib_get_slab_allocations.o : stdlib_get_slab_allocations.c stdlib_memory.h $(LIBC_OBJS)/stdlib_free_unused_slabs.o : stdlib_free_unused_slabs.c stdlib_memory.h include/stdlib.h
$(LIBC_OBJS)/stdlib_realloc.o : stdlib_realloc.c stdlib_memory.h $(LIBC_OBJS)/stdlib_decay_unused_slabs.o : stdlib_decay_unused_slabs.c stdlib_memory.h include/stdlib.h
$(LIBC_OBJS)/stdlib_red_black.o : stdlib_red_black.c stdlib_memory.h $(LIBC_OBJS)/stdlib_get_slab_usage.o : stdlib_get_slab_usage.c stdlib_memory.h include/stdlib.h
$(LIBC_OBJS)/stdlib_get_slab_allocations.o : stdlib_get_slab_allocations.c stdlib_memory.h include/stdlib.h
$(LIBC_OBJS)/stdlib_realloc.o : stdlib_realloc.c stdlib_memory.h include/stdlib.h
$(LIBC_OBJS)/stdlib_red_black.o : stdlib_red_black.c stdlib_memory.h include/stdlib.h
############################################################################## ##############################################################################

View File

@@ -54,7 +54,8 @@ LOG_COMMAND := 2>&1 | tee -a compiler.log
WARNINGS := \ WARNINGS := \
-Wall -W -Wpointer-arith -Wsign-compare -Wmissing-prototypes \ -Wall -W -Wpointer-arith -Wsign-compare -Wmissing-prototypes \
-Wundef -Wbad-function-cast -Wmissing-declarations -Wunused -Wwrite-strings -Wundef -Wbad-function-cast -Wmissing-declarations -Wunused -Wwrite-strings \
-Wno-deprecated-declarations \
# -Wconversion -Wshadow # -Wconversion -Wshadow

View File

@@ -1,6 +1,6 @@
#define VERSION 1 #define VERSION 1
#define REVISION 211 #define REVISION 213
#define DATE "23.11.2016" #define DATE "4.12.2016"
#define VERS "amiga.lib 1.211" #define VERS "amiga.lib 1.213"
#define VSTRING "amiga.lib 1.211 (23.11.2016)\r\n" #define VSTRING "amiga.lib 1.213 (4.12.2016)\r\n"
#define VERSTAG "\0$VER: amiga.lib 1.211 (23.11.2016)" #define VERSTAG "\0$VER: amiga.lib 1.213 (4.12.2016)"

View File

@@ -1 +1 @@
211 213

View File

@@ -79,6 +79,8 @@ DoTimer(struct timeval *tv,LONG unit,LONG command)
struct MsgPort * mp; struct MsgPort * mp;
LONG error; LONG error;
PROFILE_OFF();
assert( tv != NULL ); assert( tv != NULL );
#if defined(__amigaos4__) #if defined(__amigaos4__)
@@ -129,14 +131,10 @@ DoTimer(struct timeval *tv,LONG unit,LONG command)
tr->tr_time.tv_secs = tv->tv_secs; tr->tr_time.tv_secs = tv->tv_secs;
tr->tr_time.tv_micro = tv->tv_micro; tr->tr_time.tv_micro = tv->tv_micro;
PROFILE_OFF();
SetSignal(0,(1UL << mp->mp_SigBit)); SetSignal(0,(1UL << mp->mp_SigBit));
error = DoIO((struct IORequest *)tr); error = DoIO((struct IORequest *)tr);
PROFILE_ON();
tv->tv_secs = tr->tr_time.tv_secs; tv->tv_secs = tr->tr_time.tv_secs;
tv->tv_micro = tr->tr_time.tv_micro; tv->tv_micro = tr->tr_time.tv_micro;
@@ -161,5 +159,7 @@ DoTimer(struct timeval *tv,LONG unit,LONG command)
} }
#endif /* __amigaos4__ */ #endif /* __amigaos4__ */
PROFILE_ON();
return(error); return(error);
} }

View File

@@ -262,9 +262,8 @@ STATIC VOID
_SetValue(struct Environment * env,struct NexxStr * value,struct Node * symbol_table_node) _SetValue(struct Environment * env,struct NexxStr * value,struct Node * symbol_table_node)
{ {
STATIC CONST UWORD code[] = { 0x4EAE,0xFFAC,0x4E75 }; /* jsr -84(a6) ; rts */ STATIC CONST UWORD code[] = { 0x4EAE,0xFFAC,0x4E75 }; /* jsr -84(a6) ; rts */
struct Node * result;
result = (struct Node *)EmulateTags(code, EmulateTags(code,
ET_RegisterA0,env, ET_RegisterA0,env,
ET_RegisterA1,value, ET_RegisterA1,value,
ET_RegisterD0,symbol_table_node, ET_RegisterD0,symbol_table_node,

View File

@@ -1,6 +1,6 @@
#define VERSION 1 #define VERSION 1
#define REVISION 211 #define REVISION 213
#define DATE "23.11.2016" #define DATE "4.12.2016"
#define VERS "c.lib 1.211" #define VERS "c.lib 1.213"
#define VSTRING "c.lib 1.211 (23.11.2016)\r\n" #define VSTRING "c.lib 1.213 (4.12.2016)\r\n"
#define VERSTAG "\0$VER: c.lib 1.211 (23.11.2016)" #define VERSTAG "\0$VER: c.lib 1.213 (4.12.2016)"

View File

@@ -1 +1 @@
211 213

View File

@@ -1,3 +1,49 @@
c.lib 1.213 (4.12.2016)
- Added the __decay_unused_slabs() function which brings all currently
empty slabs which are still protected from reuse closer to getting
reused or released.
- The slab-test program now exercises the memory allocation functions
to a greater degree. Memory is allocated in random chunk sizes,
the allocations are resized (to other random chunk sizes),
33% of all allocations are randomly freed, empty slabs readied for
reuse then discarded. The output in JSON format now shows a bit
more information as to what is being done.
- Rewrote __get_slab_stats() to use setjmp() and longjmp() in the
print() callback invocation.
- __get_slab_stats() now reports how many times a slab was reused
after having stuck around in the "empty slab" list.
- Changing the slab size through an environment variable is now
a feature of the debug build.
- Small changes to allow the library to be built with SAS/C again.
This includes adding code to disable/re-enable profiling,
fixing "stdlib_profile.h" and updating the smakefiles.
- Still not sure what it does, but _CXV45 now sits along with _CX25
and _CX35 in "sas_cxv.asm". "sas_cxv54.asm" is not needed any
more.
- Found the last use of MEMF_PRIVATE which should have been compiled
only for the OS4 version.
c.lib 1.212 (27.11.2016)
- Unused slabs which get recycled are no longer reinitialized from
scratch if their chunk size matches what the allocator needed.
If the chunk size matches, the list of available chunks is
left unchanged, and just the various counters are reset.
- Added __get_slab_stats() function.
- Added support for global __slab_purge_threshold tuning variable.
c.lib 1.211 (23.11.2016) c.lib 1.211 (23.11.2016)
- Added more consistency checking to the slab allocator, which is - Added more consistency checking to the slab allocator, which is

View File

@@ -1,6 +1,6 @@
#define VERSION 1 #define VERSION 1
#define REVISION 211 #define REVISION 213
#define DATE "23.11.2016" #define DATE "4.12.2016"
#define VERS "debug.lib 1.211" #define VERS "debug.lib 1.213"
#define VSTRING "debug.lib 1.211 (23.11.2016)\r\n" #define VSTRING "debug.lib 1.213 (4.12.2016)\r\n"
#define VERSTAG "\0$VER: debug.lib 1.211 (23.11.2016)" #define VERSTAG "\0$VER: debug.lib 1.213 (4.12.2016)"

View File

@@ -1 +1 @@
211 213

View File

@@ -173,6 +173,19 @@ extern int rand_r(unsigned int * seed);
extern unsigned long __slab_max_size; extern unsigned long __slab_max_size;
/*
* The slab allocator will periodically free all currently unused memory.
* You can control how much memory should be released, instead of
* releasing everything.
*
* This would make the slab allocator release only up to 512 KBytes of
* unused memory at a time:
*
* unsigned long __slab_purge_threshold = 512 * 1024;
*/
extern unsigned long __slab_purge_threshold;
/****************************************************************************/ /****************************************************************************/
/* /*
@@ -188,6 +201,21 @@ extern void __free_unused_slabs(void);
/****************************************************************************/ /****************************************************************************/
/*
* You can accelerate the reuse of currently unused slabs by calling
* the __decay_unused_slabs() function. Each call decrements the decay
* counter until it reaches 0, at which point an unused slab can be
* reused instead of allocating a new slab. Also, at 0 unused slabs
* will be freed by the allocator.
*
* Please note that this function works within the context of the memory
* allocation system and is not safe to call from interrupt code. It may
* break a Forbid() or Disable() condition.
*/
extern void __decay_unused_slabs(void);
/****************************************************************************/
/* /*
* You can obtain runtime statistics about the slab allocator by * You can obtain runtime statistics about the slab allocator by
* invoking the __get_slab_usage() function which in turn invokes * invoking the __get_slab_usage() function which in turn invokes
@@ -250,6 +278,11 @@ struct __slab_usage_information
/* How many memory chunks in this slab are being used? */ /* How many memory chunks in this slab are being used? */
size_t sui_num_chunks_used; size_t sui_num_chunks_used;
/* How many time was this slab reused without reinitializing
* it all over again from scratch?
*/
size_t sui_num_reused;
}; };
/****************************************************************************/ /****************************************************************************/
@@ -324,6 +357,31 @@ void __get_slab_allocations(__slab_allocation_callback callback);
/****************************************************************************/ /****************************************************************************/
/*
* You can obtain information about the memory managed by the slab allocator,
* as well as additional information about the slab allocator's performance
* in JSON format. This format can be used for more detailed analysis.
*
* You supply a function which will be called for each line of the JSON
* data produced. You can store this data in a file, or in the clipboard,
* for later use. Your function must return 0 if it wants to be called
* again, or return -1 if it wants to stop (e.g. if an error occured
* when writing the JSON data to disk). The same "user_data" pointer which
* you pass to __get_slab_stats() will be passed to your callback function.
*
* Please note that this function works within the context of the memory
* allocation system and is not safe to call from interrupt code. It may
* break a Forbid() or Disable() condition.
*/
typedef int (* __slab_status_callback)(void * user_data, const char * line, size_t line_length);
/****************************************************************************/
extern void __get_slab_stats(void * user_data, __slab_status_callback callback);
/****************************************************************************/
/* /*
* You can request to use the alloca() variant that actually does allocate * You can request to use the alloca() variant that actually does allocate
* memory from the system rather than the current stack frame, which will * memory from the system rather than the current stack frame, which will

View File

@@ -212,6 +212,7 @@ C_LIB := \
stdlib_dosbase.o \ stdlib_dosbase.o \
stdlib_exit.o \ stdlib_exit.o \
stdlib_free.o \ stdlib_free.o \
stdlib_decay_unused_slabs.o \
stdlib_free_unused_slabs.o \ stdlib_free_unused_slabs.o \
stdlib_getdefstacksize.o \ stdlib_getdefstacksize.o \
stdlib_getenv.o \ stdlib_getenv.o \
@@ -220,6 +221,7 @@ C_LIB := \
stdlib_get_errno.o \ stdlib_get_errno.o \
stdlib_get_slab_usage.o \ stdlib_get_slab_usage.o \
stdlib_get_slab_allocations.o \ stdlib_get_slab_allocations.o \
stdlib_get_slab_stats.o \
stdlib_isresident.o \ stdlib_isresident.o \
stdlib_labs.o \ stdlib_labs.o \
stdlib_llabs.o \ stdlib_llabs.o \
@@ -260,6 +262,7 @@ C_LIB := \
stdlib_showerror.o \ stdlib_showerror.o \
stdlib_slab.o \ stdlib_slab.o \
stdlib_slab_max_size.o \ stdlib_slab_max_size.o \
stdlib_slab_purge_threshold.o \
stdlib_srand.o \ stdlib_srand.o \
stdlib_stacksize.o \ stdlib_stacksize.o \
stdlib_stack_usage.o \ stdlib_stack_usage.o \

View File

@@ -1,6 +1,6 @@
#define VERSION 1 #define VERSION 1
#define REVISION 211 #define REVISION 213
#define DATE "23.11.2016" #define DATE "4.12.2016"
#define VERS "m.lib 1.211" #define VERS "m.lib 1.213"
#define VSTRING "m.lib 1.211 (23.11.2016)\r\n" #define VSTRING "m.lib 1.213 (4.12.2016)\r\n"
#define VERSTAG "\0$VER: m.lib 1.211 (23.11.2016)" #define VERSTAG "\0$VER: m.lib 1.213 (4.12.2016)"

View File

@@ -1 +1 @@
211 213

View File

@@ -1,6 +1,6 @@
#define VERSION 1 #define VERSION 1
#define REVISION 211 #define REVISION 213
#define DATE "23.11.2016" #define DATE "4.12.2016"
#define VERS "m881.lib 1.211" #define VERS "m881.lib 1.213"
#define VSTRING "m881.lib 1.211 (23.11.2016)\r\n" #define VSTRING "m881.lib 1.213 (4.12.2016)\r\n"
#define VERSTAG "\0$VER: m881.lib 1.211 (23.11.2016)" #define VERSTAG "\0$VER: m881.lib 1.213 (4.12.2016)"

View File

@@ -1 +1 @@
211 213

View File

@@ -1,6 +1,6 @@
#define VERSION 1 #define VERSION 1
#define REVISION 211 #define REVISION 213
#define DATE "23.11.2016" #define DATE "4.12.2016"
#define VERS "net.lib 1.211" #define VERS "net.lib 1.213"
#define VSTRING "net.lib 1.211 (23.11.2016)\r\n" #define VSTRING "net.lib 1.213 (4.12.2016)\r\n"
#define VERSTAG "\0$VER: net.lib 1.211 (23.11.2016)" #define VERSTAG "\0$VER: net.lib 1.213 (4.12.2016)"

View File

@@ -1 +1 @@
211 213

View File

@@ -37,6 +37,7 @@
xdef __CXV25 xdef __CXV25
xdef __CXV35 xdef __CXV35
xdef __CXV45
xdef __CXNRM5 xdef __CXNRM5
xdef __CXTAB5 xdef __CXTAB5
@@ -74,6 +75,43 @@ L44: MOVE.W D0,D1
MOVEM.L (SP)+,D2-D5/A1 MOVEM.L (SP)+,D2-D5/A1
RTS RTS
__CXV45:
MOVE.L D0,D1
SWAP D1
AND.W #$7FFF,D1
CMP.W #$80,D1
BLT .1
CMP.W #$7F80,D1
BGE .3
ASR.L #3,D0
AND.L #$8FFFFFFF,D0
ADD.L #$38000000,D0
SWAP D1
AND.L #7,D1
ROR.L #3,D1
.2 RTS
.1 TST.L D1
BEQ.S .2
MOVEM.L D2-D5,-(SP)
SWAP D0
MOVE.W D0,D4
AND.W #$8000,D4
MOVE.W #$39D0,D5
MOVEQ #0,D0
SWAP D1
JSR __CXNRM5(PC)
MOVEM.L (SP)+,D2-D5
RTS
.3 ASR.L #3,D0
OR.L #$7FF00000,D0
SWAP D1
AND.L #7,D1
ROR.L #3,D1
RTS
__CXNRM5: __CXNRM5:
CMP.L #$20,D0 CMP.L #$20,D0

106
library/sas_cxv52.asm Normal file
View File

@@ -0,0 +1,106 @@
*
* :ts=8
*
* Adapted from reassembled SAS/C runtime library code.
*
* 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.
*
xdef __CXV52
xdef __CXV53
xref __CXFERR
section text,code
__CXV52:
MOVEM.L D2/D3,-(SP)
MOVEQ #-1,D3
MOVE.L D0,D2
BPL.W lab04A
CMPI.L #$BFF00000,D0
BCS.W lab07A
MOVEM.L D0/D1/A0/A1,-(SP)
PEA (2).L
JSR __CXFERR
ADDQ.W #4,SP
MOVEM.L (SP)+,D0/D1/A0/A1
BRA.W lab07A
__CXV53:
MOVEM.L D2/D3,-(SP)
MOVE.L #$7FFFFFFF,D3
MOVE.L D0,D2
BPL.W lab04A
ADDQ.L #1,D3
EOR.L D3,D0
lab04A: SWAP D0
MOVE.W D0,D2
ANDI.W #$7FF0,D2
EOR.W D2,D0
SUBI.W #$3FF0,D2
BLT.W lab07A
EORI.W #$10,D0
SWAP D0
ASR.W #4,D2
SUBI.W #$14,D2
BGT.W lab09A
NEG.W D2
LSR.L D2,D0
TST.L D2
BMI.W lab0B2
BRA.W lab0B4
lab07A: MOVEQ #0,D0
BRA.W lab0B4
lab080: MOVEM.L D0/D1/A0/A1,-(SP)
PEA (2).L
JSR __CXFERR
ADDQ.W #4,SP
MOVEM.L (SP)+,D0/D1/A0/A1
MOVE.L D3,D0
BRA.W lab0B4
lab09A: CMPI.W #11,D2
BGT.B lab080
EOR.L D1,D0
ROL.L D2,D0
LSL.L D2,D1
EOR.L D1,D0
CMP.L D3,D0
BHI.B lab080
TST.L D2
BPL.W lab0B4
lab0B2: NEG.L D0
lab0B4: MOVEM.L (SP)+,D2/D3
RTS
end

View File

@@ -1,6 +1,4 @@
* *
* $Id: sas_cxv54.asm,v 1.1.1.1 2004-07-26 16:31:04 obarthel Exp $
*
* :ts=8 * :ts=8
* *
* Adapted from reassembled SAS/C runtime library code. * Adapted from reassembled SAS/C runtime library code.
@@ -38,7 +36,8 @@
xdef __CXV54 xdef __CXV54
xref __CXFERR xref __CXFERR
__CXV54 __CXV54:
MOVEM.L A0/A1,-(SP) MOVEM.L A0/A1,-(SP)
MOVE.L D4,A0 MOVE.L D4,A0
SWAP D0 SWAP D0
@@ -47,11 +46,11 @@ __CXV54
EOR.W D4,D0 EOR.W D4,D0
SUB.W #$3800,D0 SUB.W #$3800,D0
CMP.W #$10,D0 CMP.W #$10,D0
BLT lbC000098 BLT lab098
CMP.W #$FEF,D0 CMP.W #$FEF,D0
BLT lbC000102 BLT lab102
CMP.W #$47F0,D0 CMP.W #$47F0,D0
BLT lbC000058 BLT lab058
SWAP D0 SWAP D0
LSL.L #3,D0 LSL.L #3,D0
ROL.L #3,D1 ROL.L #3,D1
@@ -59,17 +58,16 @@ __CXV54
EOR.L D1,D0 EOR.L D1,D0
SWAP D0 SWAP D0
OR.W #$7F80,D0 OR.W #$7F80,D0
BRA lbC000112 BRA lab112
lbC000058 lab058: CMP.W #$FF0,D0
CMP.W #$FF0,D0 BGE lab074
BGE lbC000074
CMP.L #$FFFF0FEF,D0 CMP.L #$FFFF0FEF,D0
BNE lbC000102 BNE lab102
CMP.L #$F0000000,D1 CMP.L #$F0000000,D1
BCS lbC000102 BCS lab102
lbC000074
MOVEM.L D0/D1/A0/A1,-(SP) lab074: MOVEM.L D0/D1/A0/A1,-(SP)
PEA 2.L PEA 2.L
JSR __CXFERR JSR __CXFERR
ADDQ.W #4,SP ADDQ.W #4,SP
@@ -78,63 +76,56 @@ lbC000074
EOR.W D4,D0 EOR.W D4,D0
SWAP D0 SWAP D0
MOVEQ #0,D1 MOVEQ #0,D1
BRA lbC000116 BRA lab116
lbC000098 lab098: CMP.W #$FE90,D0
CMP.W #$FE90,D0 BGE lab0C4
BGE lbC0000C4
ADD.W #$3800,D0 ADD.W #$3800,D0
OR.L D1,D0 OR.L D1,D0
BEQ lbC000112 BEQ lab112
MOVEM.L D0/D1/A0/A1,-(SP) MOVEM.L D0/D1/A0/A1,-(SP)
PEA 1.L PEA 1.L
JSR __CXFERR JSR __CXFERR
ADDQ.W #4,SP ADDQ.W #4,SP
MOVEM.L (SP)+,D0/D1/A0/A1 MOVEM.L (SP)+,D0/D1/A0/A1
MOVEQ #0,D0 MOVEQ #0,D0
BRA lbC000112 BRA lab112
lbC0000C4 lab0C4: MOVE.L D5,A1
MOVE.L D5,A1
MOVE.W D0,D5 MOVE.W D0,D5
AND.W #15,D0 AND.W #15,D0
EOR.W #$10,D0 EOR.W #$10,D0
SWAP D0 SWAP D0
ASR.W #4,D5 ASR.W #4,D5
ADDQ.W #2,D5 ADDQ.W #2,D5
BGE lbC0000E6 BGE lab0E6
NEG.W D5 NEG.W D5
LSR.L D5,D0 LSR.L D5,D0
MOVEQ #0,D5 MOVEQ #0,D5
ADDX.L D5,D0 ADDX.L D5,D0
BRA lbC0000F6 BRA lab0F6
lbC0000E6 lab0E6: CLR.W D1
CLR.W D1
LSL.L D5,D0 LSL.L D5,D0
ADDQ.W #1,D5 ADDQ.W #1,D5
ROXL.L D5,D1 ROXL.L D5,D1
AND.L #15,D1 AND.L #15,D1
ADDX.L D1,D0 ADDX.L D1,D0
lbC0000F6 lab0F6: MOVE.L A1,D5
MOVE.L A1,D5
SWAP D0 SWAP D0
EOR.W D4,D0 EOR.W D4,D0
SWAP D0 SWAP D0
BRA lbC000116 BRA lab116
lbC000102 lab102: SWAP D0
SWAP D0
LSL.L #3,D0 LSL.L #3,D0
ROXL.L #4,D1 ROXL.L #4,D1
AND.L #7,D1 AND.L #7,D1
ADDX.L D1,D0 ADDX.L D1,D0
SWAP D0 SWAP D0
lbC000112 lab112: EOR.W D4,D0
EOR.W D4,D0
SWAP D0 SWAP D0
lbC000116 lab116: MOVE.L A0,D4
MOVE.L A0,D4
MOVEM.L (SP)+,A0/A1 MOVEM.L (SP)+,A0/A1
RTS RTS

File diff suppressed because it is too large Load Diff

View File

@@ -0,0 +1,75 @@
/*
* :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_HEADERS_H
#include "stdlib_headers.h"
#endif /* _STDLIB_HEADERS_H */
/****************************************************************************/
#ifndef _STDLIB_MEMORY_H
#include "stdlib_memory.h"
#endif /* _STDLIB_MEMORY_H */
/****************************************************************************/
/* Look at all currently unused slabs, decrementing the decay
* counter which prevents them from being reused.
*/
void
__decay_unused_slabs(void)
{
if(__slab_data.sd_InUse)
{
struct MinNode * free_node;
struct MinNode * free_node_next;
struct SlabNode * sn;
__memory_lock();
for(free_node = (struct MinNode *)__slab_data.sd_EmptySlabs.mlh_Head ;
free_node->mln_Succ != NULL ;
free_node = free_node_next)
{
free_node_next = (struct MinNode *)free_node->mln_Succ;
/* free_node points to SlabNode.sn_EmptyLink, which
* directly follows the SlabNode.sn_MinNode.
*/
sn = (struct SlabNode *)&free_node[-1];
if(sn->sn_EmptyDecay > 0)
sn->sn_EmptyDecay--;
}
__memory_unlock();
}
}

View File

@@ -378,15 +378,21 @@ remove_and_free_memory_node(struct MemoryNode * mn)
{ {
__slab_free(mn,allocation_size); __slab_free(mn,allocation_size);
} }
else if (__memory_pool != NULL) else
{ {
if(__memory_pool != NULL)
{
PROFILE_OFF();
FreePooled(__memory_pool,mn,allocation_size); FreePooled(__memory_pool,mn,allocation_size);
PROFILE_ON();
} }
else else
{ {
#if defined(__MEM_DEBUG) #if defined(__MEM_DEBUG)
{ {
PROFILE_OFF();
FreeMem(mn,allocation_size); FreeMem(mn,allocation_size);
PROFILE_ON();
} }
#else #else
{ {
@@ -396,22 +402,29 @@ remove_and_free_memory_node(struct MemoryNode * mn)
Remove((struct Node *)mln); Remove((struct Node *)mln);
PROFILE_OFF();
FreeMem(mln,sizeof(*mln) + allocation_size); FreeMem(mln,sizeof(*mln) + allocation_size);
PROFILE_ON();
} }
#endif /* __MEM_DEBUG */ #endif /* __MEM_DEBUG */
} }
} }
}
#else #else
{ {
if(__memory_pool != NULL) if(__memory_pool != NULL)
{ {
PROFILE_OFF();
FreePooled(__memory_pool,mn,allocation_size); FreePooled(__memory_pool,mn,allocation_size);
PROFILE_ON();
} }
else else
{ {
#if defined(__MEM_DEBUG) #if defined(__MEM_DEBUG)
{ {
PROFILE_OFF();
FreeMem(mn,allocation_size); FreeMem(mn,allocation_size);
PROFILE_ON();
} }
#else #else
{ {
@@ -421,7 +434,9 @@ remove_and_free_memory_node(struct MemoryNode * mn)
Remove((struct Node *)mln); Remove((struct Node *)mln);
PROFILE_OFF();
FreeMem(mln,sizeof(*mln) + allocation_size); FreeMem(mln,sizeof(*mln) + allocation_size);
PROFILE_ON();
} }
#endif /* __MEM_DEBUG */ #endif /* __MEM_DEBUG */
} }

View File

@@ -72,7 +72,9 @@ __free_unused_slabs(void)
/* Unlink from list of slabs of the same size. */ /* Unlink from list of slabs of the same size. */
Remove((struct Node *)sn); Remove((struct Node *)sn);
PROFILE_OFF();
FreeVec(sn); FreeVec(sn);
PROFILE_ON();
} }
__memory_unlock(); __memory_unlock();

View File

@@ -0,0 +1,206 @@
/*
* :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_HEADERS_H
#include "stdlib_headers.h"
#endif /* _STDLIB_HEADERS_H */
/****************************************************************************/
#ifndef _STDLIB_MEMORY_H
#include "stdlib_memory.h"
#endif /* _STDLIB_MEMORY_H */
/****************************************************************************/
#include <setjmp.h>
/****************************************************************************/
struct context
{
jmp_buf abort_buf;
void * user_data;
__slab_status_callback callback;
char * buffer;
size_t buffer_size;
};
/****************************************************************************/
static void print(struct context * ct, const char * format, ...)
{
va_list args;
int len;
va_start(args,format);
len = vsnprintf(ct->buffer, ct->buffer_size, format, args);
va_end(args);
/* This shouldn't happen: the buffer ought to be large enough
* to hold every single line.
*/
if(len >= (int)ct->buffer_size)
len = strlen(ct->buffer);
if((*ct->callback)(ct->user_data, ct->buffer, len) != 0)
longjmp(ct->abort_buf,-1);
}
/****************************************************************************/
void
__get_slab_stats(void * user_data, __slab_status_callback callback)
{
if(__slab_data.sd_InUse)
{
static int times_checked = 1;
const struct SlabNode * sn;
volatile size_t num_empty_slabs = 0;
volatile size_t num_full_slabs = 0;
volatile size_t num_slabs = 0;
volatile size_t slab_allocation_size = 0;
volatile size_t total_slab_allocation_size = 0;
struct context ct;
char line[1024];
char time_buffer[40];
time_t now;
struct tm when;
int i;
memset(&ct, 0, sizeof(ct));
ct.user_data = user_data;
ct.callback = callback;
ct.buffer = line;
ct.buffer_size = sizeof(line);
__memory_lock();
if(setjmp(ct.abort_buf) == 0)
{
now = time(NULL);
localtime_r(&now, &when);
strftime(time_buffer, sizeof(time_buffer), "%Y-%m-%dT%H:%M:%S", &when);
print(&ct,"{\n");
print(&ct,"\t\"when\": \"%s\",\n", time_buffer);
print(&ct,"\t\"times_checked\": %d,\n", times_checked++);
print(&ct,"\t\"slab_size\": %zu,\n", __slab_data.sd_StandardSlabSize);
print(&ct,"\t\"num_single_allocations\": %zu,\n", __slab_data.sd_NumSingleAllocations);
print(&ct,"\t\"total_single_allocation_size\": %zu,\n", __slab_data.sd_TotalSingleAllocationSize);
if(__slab_data.sd_SingleAllocations.mlh_Head->mln_Succ != NULL)
{
const struct SlabSingleAllocation * ssa;
print(&ct,"\t\"single_allocations\": [\n");
for(ssa = (struct SlabSingleAllocation *)__slab_data.sd_SingleAllocations.mlh_Head ;
ssa->ssa_MinNode.mln_Succ != NULL ;
ssa = (struct SlabSingleAllocation *)ssa->ssa_MinNode.mln_Succ)
{
print(&ct,"\t\t{ \"size\": %lu, \"total_size\": %lu }%s\n",
ssa->ssa_Size - sizeof(*ssa), ssa->ssa_Size,
ssa->ssa_MinNode.mln_Succ->mln_Succ != NULL ? "," : "");
}
print(&ct,"\t],\n");
}
else
{
print(&ct,"\t\"single_allocations\": [],\n");
}
for(i = 0 ; i < (int)NUM_ENTRIES(__slab_data.sd_Slabs) ; i++)
{
for(sn = (struct SlabNode *)__slab_data.sd_Slabs[i].mlh_Head ;
sn->sn_MinNode.mln_Succ != NULL ;
sn = (struct SlabNode *)sn->sn_MinNode.mln_Succ)
{
if (sn->sn_UseCount == 0)
num_empty_slabs++;
else if (sn->sn_UseCount == sn->sn_Count)
num_full_slabs++;
num_slabs++;
slab_allocation_size += sn->sn_ChunkSize * sn->sn_UseCount;
total_slab_allocation_size += sizeof(*sn) + __slab_data.sd_StandardSlabSize;
}
}
print(&ct,"\t\"num_slabs\": %zu,\n", num_slabs);
print(&ct,"\t\"num_empty_slabs\": %zu,\n", num_empty_slabs);
print(&ct,"\t\"num_full_slabs\": %zu,\n", num_full_slabs);
print(&ct,"\t\"slab_allocation_size\": %zu,\n", slab_allocation_size);
print(&ct,"\t\"total_slab_allocation_size\": %zu,\n", total_slab_allocation_size);
if(num_slabs > 0)
{
const char * eol = "";
print(&ct,"\t\"slabs\": [\n");
for(i = 0 ; i < (int)NUM_ENTRIES(__slab_data.sd_Slabs) ; i++)
{
for(sn = (struct SlabNode *)__slab_data.sd_Slabs[i].mlh_Head ;
sn->sn_MinNode.mln_Succ != NULL ;
sn = (struct SlabNode *)sn->sn_MinNode.mln_Succ)
{
print(&ct,"%s\t\t{ \"size\": %lu, \"chunks\": %lu, \"chunks_in_use\": %lu, \"times_reused\": %lu, \"empty_decay\": %lu }",
eol,
sn->sn_ChunkSize,
sn->sn_Count,
sn->sn_UseCount,
sn->sn_NumReused,
sn->sn_EmptyDecay);
eol = ",\n";
}
}
print(&ct,"\n\t]\n");
}
else
{
print(&ct,"\t\"slabs\": []\n");
}
print(&ct,"}\n");
}
__memory_unlock();
}
}

View File

@@ -87,6 +87,7 @@ __get_slab_usage(__slab_usage_callback callback)
sui.sui_chunk_size = sn->sn_ChunkSize; sui.sui_chunk_size = sn->sn_ChunkSize;
sui.sui_num_chunks = sn->sn_Count; sui.sui_num_chunks = sn->sn_Count;
sui.sui_num_chunks_used = sn->sn_UseCount; sui.sui_num_chunks_used = sn->sn_UseCount;
sui.sui_num_reused = sn->sn_NumReused;
sui.sui_slab_index++; sui.sui_slab_index++;

View File

@@ -37,6 +37,10 @@
/****************************************************************************/ /****************************************************************************/
#if defined(__GNUC__) || (defined(__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L))
/****************************************************************************/
lldiv_t lldiv_t
lldiv(long long n,long long d) lldiv(long long n,long long d)
{ {
@@ -49,3 +53,7 @@ lldiv(long long n,long long d)
return(result); return(result);
} }
/****************************************************************************/
#endif

View File

@@ -117,9 +117,20 @@ call_main(void)
struct Process * this_process = (struct Process *)FindTask(NULL); struct Process * this_process = (struct Process *)FindTask(NULL);
UBYTE * arg_str = GetArgStr(); UBYTE * arg_str = GetArgStr();
size_t arg_str_len = strlen(arg_str); size_t arg_str_len = strlen(arg_str);
UBYTE * arg_str_copy = AllocVec(arg_str_len+1,MEMF_PRIVATE); UBYTE * arg_str_copy;
UBYTE current_dir_name[256]; UBYTE current_dir_name[256];
#if defined(__amigaos4__)
{
arg_str_copy = AllocVec(arg_str_len+1,MEMF_PRIVATE);
}
#else
{
arg_str_copy = AllocVec(arg_str_len+1,MEMF_ANY);
}
#endif /* __amigaos4__ */
if(arg_str_copy != NULL && NameFromLock(this_process->pr_CurrentDir,current_dir_name,sizeof(current_dir_name))) if(arg_str_copy != NULL && NameFromLock(this_process->pr_CurrentDir,current_dir_name,sizeof(current_dir_name)))
{ {
strcpy(arg_str_copy,arg_str); strcpy(arg_str_copy,arg_str);

View File

@@ -151,21 +151,30 @@ __allocate_memory(size_t size,BOOL never_free,const char * UNUSED debug_file_nam
{ {
mn = __slab_allocate(allocation_size); mn = __slab_allocate(allocation_size);
} }
else if (__memory_pool != NULL) else
{ {
if (__memory_pool != NULL)
{
PROFILE_OFF();
mn = AllocPooled(__memory_pool,allocation_size); mn = AllocPooled(__memory_pool,allocation_size);
PROFILE_ON();
} }
else else
{ {
#ifdef __MEM_DEBUG #ifdef __MEM_DEBUG
{ {
PROFILE_OFF();
mn = AllocMem(allocation_size,MEMF_ANY); mn = AllocMem(allocation_size,MEMF_ANY);
PROFILE_ON();
} }
#else #else
{ {
struct MinNode * mln; struct MinNode * mln;
PROFILE_OFF();
mln = AllocMem(sizeof(*mln) + allocation_size,MEMF_ANY); mln = AllocMem(sizeof(*mln) + allocation_size,MEMF_ANY);
PROFILE_ON();
if(mln != NULL) if(mln != NULL)
{ {
AddTail((struct List *)&__memory_list,(struct Node *)mln); AddTail((struct List *)&__memory_list,(struct Node *)mln);
@@ -180,23 +189,31 @@ __allocate_memory(size_t size,BOOL never_free,const char * UNUSED debug_file_nam
#endif /* __MEM_DEBUG */ #endif /* __MEM_DEBUG */
} }
} }
}
#else #else
{ {
if(__memory_pool != NULL) if(__memory_pool != NULL)
{ {
PROFILE_OFF();
mn = AllocPooled(__memory_pool,allocation_size); mn = AllocPooled(__memory_pool,allocation_size);
PROFILE_ON();
} }
else else
{ {
#ifdef __MEM_DEBUG #ifdef __MEM_DEBUG
{ {
PROFILE_OFF();
mn = AllocMem(allocation_size,MEMF_ANY); mn = AllocMem(allocation_size,MEMF_ANY);
PROFILE_ON();
} }
#else #else
{ {
struct MinNode * mln; struct MinNode * mln;
PROFILE_OFF();
mln = AllocMem(sizeof(*mln) + allocation_size,MEMF_ANY); mln = AllocMem(sizeof(*mln) + allocation_size,MEMF_ANY);
PROFILE_ON();
if(mln != NULL) if(mln != NULL)
{ {
AddTail((struct List *)&__memory_list,(struct Node *)mln); AddTail((struct List *)&__memory_list,(struct Node *)mln);
@@ -355,8 +372,12 @@ static struct SignalSemaphore * memory_semaphore;
void void
__memory_lock(void) __memory_lock(void)
{ {
PROFILE_OFF();
if(memory_semaphore != NULL) if(memory_semaphore != NULL)
ObtainSemaphore(memory_semaphore); ObtainSemaphore(memory_semaphore);
PROFILE_ON();
} }
/****************************************************************************/ /****************************************************************************/
@@ -364,8 +385,12 @@ __memory_lock(void)
void void
__memory_unlock(void) __memory_unlock(void)
{ {
PROFILE_OFF();
if(memory_semaphore != NULL) if(memory_semaphore != NULL)
ReleaseSemaphore(memory_semaphore); ReleaseSemaphore(memory_semaphore);
PROFILE_ON();
} }
/****************************************************************************/ /****************************************************************************/
@@ -503,7 +528,7 @@ STDLIB_CONSTRUCTOR(stdlib_memory_init)
#if defined(__USE_SLAB_ALLOCATOR) #if defined(__USE_SLAB_ALLOCATOR)
{ {
/* ZZZ this is just for the purpose of testing */ /* ZZZ this is just for the purpose of testing */
#if 0 #if DEBUG
{ {
TEXT slab_size_var[20]; TEXT slab_size_var[20];

View File

@@ -235,6 +235,11 @@ struct SlabNode
/* How many chunks of this slab are currently in use? */ /* How many chunks of this slab are currently in use? */
ULONG sn_UseCount; ULONG sn_UseCount;
/* How many times was this slab reused instead of allocating
* it from system memory?
*/
ULONG sn_NumReused;
/* This contains all the chunks of memory which are available /* This contains all the chunks of memory which are available
* for allocation. * for allocation.
*/ */
@@ -286,7 +291,7 @@ struct SlabData
*/ */
size_t sd_StandardSlabSize; size_t sd_StandardSlabSize;
/* These fields kees track of how many entries there are in /* These fields keep track of how many entries there are in
* the sd_SingleAllocations list, and how much memory these * the sd_SingleAllocations list, and how much memory these
* allocations occupy. * allocations occupy.
*/ */
@@ -303,6 +308,7 @@ struct SlabData
extern struct SlabData NOCOMMON __slab_data; extern struct SlabData NOCOMMON __slab_data;
extern unsigned long NOCOMMON __slab_max_size; extern unsigned long NOCOMMON __slab_max_size;
extern unsigned long NOCOMMON __slab_purge_threshold;
/****************************************************************************/ /****************************************************************************/

View File

@@ -45,8 +45,8 @@
/****************************************************************************/ /****************************************************************************/
extern void ASM _PROLOG(REG(a0,char *)); extern void __asm _PROLOG(register __a0 char *);
extern void ASM _EPILOG(REG(a0,char *)); extern void __asm _EPILOG(register __a0 char *);
#if _PROFILE #if _PROFILE
#define PROFILE_OFF() _PROLOG(0L) #define PROFILE_OFF() _PROLOG(0L)

View File

@@ -31,6 +31,8 @@
* POSSIBILITY OF SUCH DAMAGE. * POSSIBILITY OF SUCH DAMAGE.
*/ */
/*#define DEBUG*/
#ifndef _STDLIB_HEADERS_H #ifndef _STDLIB_HEADERS_H
#include "stdlib_headers.h" #include "stdlib_headers.h"
#endif /* _STDLIB_HEADERS_H */ #endif /* _STDLIB_HEADERS_H */
@@ -76,6 +78,7 @@ __realloc(void *ptr,size_t size,const char * file,int line)
#endif /* UNIX_PATH_SEMANTICS */ #endif /* UNIX_PATH_SEMANTICS */
else else
{ {
size_t old_size;
struct MemoryNode * mn; struct MemoryNode * mn;
BOOL reallocate; BOOL reallocate;
@@ -108,29 +111,23 @@ __realloc(void *ptr,size_t size,const char * file,int line)
} }
#endif /* __MEM_DEBUG */ #endif /* __MEM_DEBUG */
if(mn == NULL || mn->mn_NeverFree) if(mn == NULL || FLAG_IS_SET(mn->mn_Size, MN_SIZE_NEVERFREE))
{ {
SHOWMSG("cannot free this chunk"); SHOWMSG("cannot free this chunk");
goto out; goto out;
} }
old_size = GET_MN_SIZE(mn);
/* Don't do anything unless the size of the allocation /* Don't do anything unless the size of the allocation
has really changed. */ has really changed. */
#if defined(__MEM_DEBUG) #if defined(__MEM_DEBUG)
{ {
reallocate = (mn->mn_Size != size); reallocate = (old_size != size);
} }
#else #else
{ {
size_t rounded_allocation_size; if(size > old_size)
/* Round the total allocation size to the operating system
granularity. */
rounded_allocation_size = __get_allocation_size(size);
assert( rounded_allocation_size >= size );
if(rounded_allocation_size > mn->mn_Size)
{ {
/* Allocation size should grow. */ /* Allocation size should grow. */
reallocate = TRUE; reallocate = TRUE;
@@ -143,7 +140,7 @@ __realloc(void *ptr,size_t size,const char * file,int line)
allocation. We also take into account that the allocation. We also take into account that the
actual size of the allocation is affected by a actual size of the allocation is affected by a
certain operating system imposed granularity. */ certain operating system imposed granularity. */
reallocate = (rounded_allocation_size < mn->mn_Size && rounded_allocation_size <= mn->mn_Size / 2); reallocate = (size < old_size && size <= old_size / 2);
} }
} }
#endif /* __MEM_DEBUG */ #endif /* __MEM_DEBUG */
@@ -152,7 +149,7 @@ __realloc(void *ptr,size_t size,const char * file,int line)
{ {
void * new_ptr; void * new_ptr;
D(("realloc() size has changed; old=%ld, new=%ld",mn->mn_Size,size)); D(("realloc() size has changed; old=%ld, new=%ld",old_size,size));
/* We allocate the new memory chunk before we /* We allocate the new memory chunk before we
attempt to replace the old. */ attempt to replace the old. */
@@ -164,8 +161,8 @@ __realloc(void *ptr,size_t size,const char * file,int line)
} }
/* Copy the contents of the old allocation to the new buffer. */ /* Copy the contents of the old allocation to the new buffer. */
if(size > mn->mn_Size) if(size > old_size)
size = mn->mn_Size; size = old_size;
memmove(new_ptr,ptr,size); memmove(new_ptr,ptr,size);
@@ -177,7 +174,7 @@ __realloc(void *ptr,size_t size,const char * file,int line)
} }
else else
{ {
D(("size didn't actually change that much (%ld -> %ld); returning memory block as is.",mn->mn_Size,size)); D(("size didn't actually change that much (%ld -> %ld); returning memory block as is.",old_size,size));
/* No change in size. */ /* No change in size. */
result = ptr; result = ptr;

View File

@@ -85,6 +85,8 @@ __slab_allocate(size_t allocation_size)
/* No integer overflow? */ /* No integer overflow? */
if(allocation_size < total_single_allocation_size) if(allocation_size < total_single_allocation_size)
{ {
PROFILE_OFF();
#if defined(__amigaos4__) #if defined(__amigaos4__)
{ {
ssa = AllocMem(total_single_allocation_size,MEMF_PRIVATE); ssa = AllocMem(total_single_allocation_size,MEMF_PRIVATE);
@@ -94,6 +96,8 @@ __slab_allocate(size_t allocation_size)
ssa = AllocMem(total_single_allocation_size,MEMF_ANY); ssa = AllocMem(total_single_allocation_size,MEMF_ANY);
} }
#endif /* __amigaos4__ */ #endif /* __amigaos4__ */
PROFILE_ON();
} }
/* Integer overflow has occured. */ /* Integer overflow has occured. */
else else
@@ -125,6 +129,7 @@ __slab_allocate(size_t allocation_size)
else else
{ {
struct MinList * slab_list = NULL; struct MinList * slab_list = NULL;
BOOL slab_reused = FALSE;
ULONG entry_size; ULONG entry_size;
ULONG chunk_size; ULONG chunk_size;
int slab_index; int slab_index;
@@ -202,6 +207,9 @@ __slab_allocate(size_t allocation_size)
/* Pull it out of the list of slabs available for reuse. */ /* Pull it out of the list of slabs available for reuse. */
Remove((struct Node *)&sn->sn_EmptyLink); Remove((struct Node *)&sn->sn_EmptyLink);
sn->sn_EmptyDecay = 0;
sn->sn_NumReused++;
} }
sn->sn_UseCount++; sn->sn_UseCount++;
@@ -252,14 +260,27 @@ __slab_allocate(size_t allocation_size)
/* Unlink from list of empty slabs. */ /* Unlink from list of empty slabs. */
Remove((struct Node *)free_node); Remove((struct Node *)free_node);
/* If the chunk size of the reused slab matches
* exactly what we need then we won't have to
* completely reinitialize it again.
*/
if(sn->sn_ChunkSize == chunk_size)
{
slab_reused = TRUE;
}
else
{
/* Unlink from list of slabs which keep chunks /* Unlink from list of slabs which keep chunks
* of the same size. It will be added there * of the same size. It will be added there
* again, at a different position. * again, at a different position.
*/ */
Remove((struct Node *)sn); Remove((struct Node *)sn);
}
D(("reusing a slab")); D(("reusing a slab"));
sn->sn_NumReused++;
new_sn = sn; new_sn = sn;
break; break;
} }
@@ -272,6 +293,8 @@ __slab_allocate(size_t allocation_size)
{ {
D(("no slab is available for reuse; allocating a new slab (%lu bytes)",sizeof(*new_sn) + __slab_data.sd_StandardSlabSize)); D(("no slab is available for reuse; allocating a new slab (%lu bytes)",sizeof(*new_sn) + __slab_data.sd_StandardSlabSize));
PROFILE_OFF();
#if defined(__amigaos4__) #if defined(__amigaos4__)
{ {
new_sn = (struct SlabNode *)AllocVec(sizeof(*new_sn) + __slab_data.sd_StandardSlabSize,MEMF_PRIVATE); new_sn = (struct SlabNode *)AllocVec(sizeof(*new_sn) + __slab_data.sd_StandardSlabSize,MEMF_PRIVATE);
@@ -282,6 +305,8 @@ __slab_allocate(size_t allocation_size)
} }
#endif /* __amigaos4__ */ #endif /* __amigaos4__ */
PROFILE_ON();
if(new_sn == NULL) if(new_sn == NULL)
D(("slab allocation failed")); D(("slab allocation failed"));
@@ -295,19 +320,28 @@ __slab_allocate(size_t allocation_size)
if(new_sn != NULL) if(new_sn != NULL)
{ {
struct MinNode * free_chunk;
ULONG num_free_chunks = 0;
BYTE * first_byte;
BYTE * last_byte;
D(("setting up slab 0x%08lx", new_sn)); D(("setting up slab 0x%08lx", new_sn));
assert( chunk_size <= __slab_data.sd_StandardSlabSize ); assert( chunk_size <= __slab_data.sd_StandardSlabSize );
/* Do we have to completely initialize this slab from scratch? */
if(NOT slab_reused)
{
struct SlabChunk * free_chunk;
ULONG num_free_chunks = 0;
BYTE * first_byte;
BYTE * last_byte;
memset(new_sn,0,sizeof(*new_sn)); memset(new_sn,0,sizeof(*new_sn));
NewList((struct List *)&new_sn->sn_FreeList); NewList((struct List *)&new_sn->sn_FreeList);
/* This slab has room for new allocations, so make sure that
* it goes to the front of the slab list. It will be used
* by the next allocation request of this size.
*/
AddHead((struct List *)slab_list,(struct Node *)new_sn);
/* Split up the slab memory into individual chunks /* Split up the slab memory into individual chunks
* of the same size and keep track of them * of the same size and keep track of them
* in the free list. The memory managed by * in the free list. The memory managed by
@@ -317,37 +351,42 @@ __slab_allocate(size_t allocation_size)
first_byte = (BYTE *)&new_sn[1]; first_byte = (BYTE *)&new_sn[1];
last_byte = &first_byte[__slab_data.sd_StandardSlabSize - chunk_size]; last_byte = &first_byte[__slab_data.sd_StandardSlabSize - chunk_size];
for(free_chunk = (struct MinNode *)first_byte ; for(free_chunk = (struct SlabChunk *)first_byte ;
free_chunk <= (struct MinNode *)last_byte; free_chunk <= (struct SlabChunk *)last_byte;
free_chunk = (struct MinNode *)(((BYTE *)free_chunk) + chunk_size)) free_chunk = (struct SlabChunk *)(((BYTE *)free_chunk) + chunk_size))
{ {
AddTail((struct List *)&new_sn->sn_FreeList, (struct Node *)free_chunk); AddTail((struct List *)&new_sn->sn_FreeList, (struct Node *)free_chunk);
num_free_chunks++; num_free_chunks++;
} }
D(("slab contains %lu chunks, %lu bytes each",num_free_chunks,chunk_size)); new_sn->sn_Count = num_free_chunks;
new_sn->sn_ChunkSize = chunk_size;
D(("new slab contains %lu chunks, %lu bytes each",num_free_chunks,chunk_size));
}
/* This slab was reused and need not be reinitialized from scratch. */
else
{
assert( new_sn->sn_FreeList.mlh_Head != NULL );
assert( new_sn->sn_ChunkSize == chunk_size );
assert( new_sn->sn_Count == 0 );
}
/* Grab the first free chunk (there has to be one). */ /* Grab the first free chunk (there has to be one). */
chunk = (struct SlabChunk *)RemHead((struct List *)&new_sn->sn_FreeList); chunk = (struct SlabChunk *)RemHead((struct List *)&new_sn->sn_FreeList);
assert( chunk != NULL );
/* Keep track of this chunk's parent slab. */ /* Keep track of this chunk's parent slab. */
chunk->sc_Parent = new_sn; chunk->sc_Parent = new_sn;
assert( chunk != NULL );
assert( chunk->sc_Parent == new_sn );
allocation = &chunk[1]; allocation = &chunk[1];
D(("allocation succeeded at 0x%08lx in slab 0x%08lx (slab use count = %lu)",allocation,new_sn,new_sn->sn_UseCount+1)); /* This slab is now in use. */
/* Set up the new slab and put it where it belongs. */
new_sn->sn_EmptyDecay = 0;
new_sn->sn_UseCount = 1; new_sn->sn_UseCount = 1;
new_sn->sn_Count = num_free_chunks;
new_sn->sn_ChunkSize = chunk_size;
SHOWVALUE(new_sn->sn_ChunkSize); D(("allocation succeeded at 0x%08lx in slab 0x%08lx (slab use count = %lu)",allocation,new_sn,new_sn->sn_UseCount));
AddHead((struct List *)slab_list,(struct Node *)new_sn);
} }
/* Mark unused slabs for purging, and purge those which /* Mark unused slabs for purging, and purge those which
@@ -355,6 +394,8 @@ __slab_allocate(size_t allocation_size)
*/ */
if(purge) if(purge)
{ {
size_t total_purged = 0;
D(("purging empty slabs")); D(("purging empty slabs"));
for(free_node = (struct MinNode *)__slab_data.sd_EmptySlabs.mlh_Head ; for(free_node = (struct MinNode *)__slab_data.sd_EmptySlabs.mlh_Head ;
@@ -379,7 +420,18 @@ __slab_allocate(size_t allocation_size)
/* Unlink from list of slabs of the same size. */ /* Unlink from list of slabs of the same size. */
Remove((struct Node *)sn); Remove((struct Node *)sn);
PROFILE_OFF();
FreeVec(sn); FreeVec(sn);
PROFILE_ON();
total_purged += sizeof(*sn) + __slab_data.sd_StandardSlabSize;
/* Stop releasing memory if we reach the threshold. If no
* threshold has been set, we will free as much memory
* as possible.
*/
if(__slab_purge_threshold > 0 && total_purged >= __slab_purge_threshold)
break;
} }
/* Give it another chance. */ /* Give it another chance. */
else else
@@ -470,7 +522,10 @@ __slab_free(void * address,size_t allocation_size)
assert( size <= __slab_data.sd_TotalSingleAllocationSize ); assert( size <= __slab_data.sd_TotalSingleAllocationSize );
Remove((struct Node *)ssa); Remove((struct Node *)ssa);
PROFILE_OFF();
FreeMem(ssa, size); FreeMem(ssa, size);
PROFILE_ON();
__slab_data.sd_NumSingleAllocations--; __slab_data.sd_NumSingleAllocations--;
__slab_data.sd_TotalSingleAllocationSize -= size; __slab_data.sd_TotalSingleAllocationSize -= size;
@@ -700,6 +755,21 @@ __slab_init(size_t slab_size)
/****************************************************************************/ /****************************************************************************/
#if DEBUG
static int print_json(void * ignore,const char * buffer,size_t len)
{
extern void kputs(const char * str);
kputs(buffer);
return(0);
}
#endif /* DEBUG */
/****************************************************************************/
void void
__slab_exit(void) __slab_exit(void)
{ {
@@ -712,7 +782,19 @@ __slab_exit(void)
struct SlabNode * sn_next; struct SlabNode * sn_next;
struct MinNode * mn; struct MinNode * mn;
struct MinNode * mn_next; struct MinNode * mn_next;
int i; size_t slab_count = 0, total_slab_size = 0;
size_t single_allocation_count = 0, total_single_allocation_size = 0;
int i, j;
#if DEBUG
{
kprintf("---BEGIN JSON DATA ---\n");
__get_slab_stats(NULL, print_json);
kprintf("---END JSON DATA ---\n\n");
}
#endif /* DEBUG */
D(("freeing slabs")); D(("freeing slabs"));
@@ -720,25 +802,40 @@ __slab_exit(void)
for(i = 0 ; i < (int)NUM_ENTRIES(__slab_data.sd_Slabs) ; i++) for(i = 0 ; i < (int)NUM_ENTRIES(__slab_data.sd_Slabs) ; i++)
{ {
if(__slab_data.sd_Slabs[i].mlh_Head->mln_Succ != NULL) if(__slab_data.sd_Slabs[i].mlh_Head->mln_Succ != NULL)
D(("freeing slab #%ld (%lu bytes per chunk)", i, (1UL << i))); D(("freeing slab slot #%ld (%lu bytes per chunk)", i, (1UL << i)));
for(sn = (struct SlabNode *)__slab_data.sd_Slabs[i].mlh_Head ; for(sn = (struct SlabNode *)__slab_data.sd_Slabs[i].mlh_Head, j = 0 ;
sn->sn_MinNode.mln_Succ != NULL ; sn->sn_MinNode.mln_Succ != NULL ;
sn = sn_next) sn = sn_next)
{ {
sn_next = (struct SlabNode *)sn->sn_MinNode.mln_Succ; sn_next = (struct SlabNode *)sn->sn_MinNode.mln_Succ;
D((" slab #%ld.%ld at 0x%08lx",i, ++j, sn));
D((" fragmentation = %ld%%",100 * (__slab_data.sd_StandardSlabSize - sn->sn_Count * sn->sn_ChunkSize) / __slab_data.sd_StandardSlabSize));
D((" total space used = %ld (%ld%%)",sn->sn_UseCount * sn->sn_ChunkSize, 100 * sn->sn_UseCount / sn->sn_Count));
D((" number of chunks total = %ld",sn->sn_Count));
D((" number of chunks used = %ld%s",sn->sn_UseCount,sn->sn_UseCount == 0 ? " (empty)" : (sn->sn_UseCount == sn->sn_Count) ? " (full)" : ""));
D((" how often reused = %ld",sn->sn_NumReused));
total_slab_size += sizeof(*sn) + __slab_data.sd_StandardSlabSize;
slab_count++;
PROFILE_OFF();
FreeVec(sn); FreeVec(sn);
PROFILE_ON();
} }
} }
if(slab_count > 0)
D(("number of slabs = %ld, total slab size = %ld bytes",slab_count, total_slab_size));
if(__slab_data.sd_SingleAllocations.mlh_Head->mln_Succ != NULL) if(__slab_data.sd_SingleAllocations.mlh_Head->mln_Succ != NULL)
D(("freeing single allocations")); D(("freeing single allocations"));
/* Free the memory allocated for each allocation which did not /* Free the memory allocated for each allocation which did not
* go into a slab. * go into a slab.
*/ */
for(mn = __slab_data.sd_SingleAllocations.mlh_Head ; for(mn = __slab_data.sd_SingleAllocations.mlh_Head, j = 0 ;
mn->mln_Succ != NULL ; mn->mln_Succ != NULL ;
mn = mn_next) mn = mn_next)
{ {
@@ -746,9 +843,19 @@ __slab_exit(void)
ssa = (struct SlabSingleAllocation *)mn; ssa = (struct SlabSingleAllocation *)mn;
D((" allocation #%ld at 0x%08lx, %lu bytes", ++j, ssa, ssa->ssa_Size));
total_single_allocation_size += ssa->ssa_Size;
single_allocation_count++;
PROFILE_OFF();
FreeMem(ssa, ssa->ssa_Size); FreeMem(ssa, ssa->ssa_Size);
PROFILE_ON();
} }
if(single_allocation_count > 0)
D(("number of single allocations = %ld, total single allocation size = %ld", single_allocation_count, total_single_allocation_size));
__slab_data.sd_InUse = FALSE; __slab_data.sd_InUse = FALSE;
} }

View File

@@ -0,0 +1,38 @@
/*
* :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_HEADERS_H
#include "stdlib_headers.h"
#endif /* _STDLIB_HEADERS_H */
/****************************************************************************/
unsigned long __slab_purge_threshold;

View File

@@ -184,7 +184,6 @@ __getcwd(char * buffer,size_t buffer_size,const char *file,int line)
if(__unix_path_semantics) if(__unix_path_semantics)
{ {
const char * path_name = buffer; const char * path_name = buffer;
size_t len;
if(__translate_amiga_to_unix_path_name(&path_name,&buffer_nti) != 0) if(__translate_amiga_to_unix_path_name(&path_name,&buffer_nti) != 0)
goto out; goto out;

View File

@@ -1,6 +1,6 @@
#define VERSION 1 #define VERSION 1
#define REVISION 211 #define REVISION 213
#define DATE "23.11.2016" #define DATE "4.12.2016"
#define VERS "unix.lib 1.211" #define VERS "unix.lib 1.213"
#define VSTRING "unix.lib 1.211 (23.11.2016)\r\n" #define VSTRING "unix.lib 1.213 (4.12.2016)\r\n"
#define VERSTAG "\0$VER: unix.lib 1.211 (23.11.2016)" #define VERSTAG "\0$VER: unix.lib 1.213 (4.12.2016)"

View File

@@ -1 +1 @@
211 213

View File

@@ -13,7 +13,7 @@ DELETE = delete all quiet
.c.o: .c.o:
@echo "Compiling $<" @echo "Compiling $<"
@$(CC) -c $(CFLAGS) $< $(CC) -c $(CFLAGS) $<
############################################################################## ##############################################################################
@@ -31,7 +31,7 @@ WARNINGS = \
INCLUDE = -I../library/include INCLUDE = -I../library/include
LIB = -L../library/lib LIB = -L../library/lib
OPTIONS = -DNDEBUG -fno-builtin -fwritable-strings -DNO_INLINE_STDARG -DIEEE_FLOATING_POINT_SUPPORT -DVERBOSE OPTIONS = -DNDEBUG -fno-builtin -fwritable-strings -DNO_INLINE_STDARG -DIEEE_FLOATING_POINT_SUPPORT -DVERBOSE=1
#OPTIONS = -D__MEM_DEBUG -fno-builtin #OPTIONS = -D__MEM_DEBUG -fno-builtin
#OPTIONS = -DDEBUG -D__MEM_DEBUG -DNO_INLINE_STDARG -fno-builtin #OPTIONS = -DDEBUG -D__MEM_DEBUG -DNO_INLINE_STDARG -fno-builtin
OPTIMIZE = -O OPTIMIZE = -O
@@ -50,14 +50,14 @@ all: test fgets_test iotest sscanf_test printf_test sprintf_test \
stack_size_test translate_test strtok_test uname simple \ stack_size_test translate_test strtok_test uname simple \
fstat_stdout_test simple_sprintf date_test sscanf_64 factorial \ fstat_stdout_test simple_sprintf date_test sscanf_64 factorial \
execvp_test setlocale rand fstat_test base_dir_nametest \ execvp_test setlocale rand fstat_test base_dir_nametest \
malloc-test malloc-test slab-test
clean: clean:
$(DELETE) #?.o #?.map test fgets_test iotest sscanf_test printf_test \ $(DELETE) #?.o #?.map test fgets_test iotest sscanf_test printf_test \
sprintf_test stack_size_test translate_test strtok_test uname \ sprintf_test stack_size_test translate_test strtok_test uname \
simple fstat_stdout_test fstat_test simple_sprintf date_test sscanf_64 \ simple fstat_stdout_test fstat_test simple_sprintf date_test sscanf_64 \
factorial execvp_test setlocale rand base_dir_nametest \ factorial execvp_test setlocale rand base_dir_nametest \
malloc-test malloc-test slab-test
############################################################################## ##############################################################################
@@ -147,7 +147,11 @@ rand : rand.o
malloc-test : malloc-test.o malloc-test : malloc-test.o
@echo "Linking $@" @echo "Linking $@"
$(CC) $(CFLAGS) -o $@ rand.o $(LIBS) -Wl,--cref,-M,-Map=$@.map $(CC) $(CFLAGS) -o $@ malloc-test.o $(LIBS) -Wl,--cref,-M,-Map=$@.map
slab-test : slab-test.o
@echo "Linking $@"
$(CC) $(CFLAGS) -o $@ slab-test.o $(LIBS) -Wl,--cref,-M,-Map=$@.map
############################################################################## ##############################################################################

View File

@@ -21,104 +21,120 @@ WARNINGS = \
-Wall -W -Wshadow -Wpointer-arith -Wsign-compare -Wmissing-prototypes \ -Wall -W -Wshadow -Wpointer-arith -Wsign-compare -Wmissing-prototypes \
-Wundef -Wbad-function-cast -Wmissing-declarations -Wconversion -Wundef -Wbad-function-cast -Wmissing-declarations -Wconversion
V = /V
INCLUDE = -I$(V)/include -I../library/include INCLUDE = -I$(V)/include -I../library/include
LIB = -L../library/lib #OPTIONS = -D__MEM_DEBUG
#OPTIONS = -DNDEBUG -fno-builtin -DNO_INLINE_STDARG -DIEEE_FLOATING_POINT_SUPPORT #OPTIONS = -DDEBUG
#OPTIONS = -D__MEM_DEBUG -fno-builtin OPTIONS = -DNDEBUG
OPTIONS = -DDEBUG -DNO_INLINE_STDARG -fno-builtin #OPTIMIZE = -O3
#OPTIMIZE = -O #DEBUG = -ggdb
#OPTIMIZE = -O2 -fomit-frame-pointer
DEBUG = -ggdb
CFLAGS = $(WARNINGS) $(OPTIMIZE) $(DEBUG) $(OPTIONS) $(CODE_TYPE) $(INCLUDE) $(LIB) # Note: Because the matching startup code needs to be used for
# correctly linking the test programs, you need to make sure
# that the current development version of clib2 is visible
# where the linker expects it (soft link). Some more tuning would be
# required here because you really should not need to tinker
# with the location of library and the options "-L. -L../library/lib"
# should be sufficient.
CFLAGS = -mcrt=clib2 -fno-builtin $(WARNINGS) $(OPTIMIZE) $(DEBUG) $(OPTIONS) $(INCLUDE)
LFLAGS = -Wl,-d
############################################################################## ##############################################################################
LIBS = -lm -lc -ldebug -lgcc LIBS = -ldebug -lm -lc
############################################################################## ##############################################################################
all: test fgets_test iotest sscanf_test printf_test sprintf_test \ all: test fgets_test iotest sscanf_test printf_test sprintf_test \
stack_size_test translate_test strtok_test uname simple \ stack_size_test translate_test strtok_test uname simple \
fstat_stdout_test simple_sprintf date_test sscanf_64 \ fstat_stdout_test simple_sprintf date_test sscanf_64 \
factorial setlocale factorial setlocale rand malloc-test slab-test
clean: clean:
$(DELETE) *.o *.map test fgets_test iotest sscanf_test printf_test \ $(DELETE) *.o *.map test fgets_test iotest sscanf_test printf_test \
sprintf_test stack_size_test translate_test strtok_test \ sprintf_test stack_size_test translate_test strtok_test \
uname simple fstat_stdout_test simple_sprintf date_test \ uname simple fstat_stdout_test simple_sprintf date_test \
sscanf_64 factorial setlocale rand sscanf_64 factorial setlocale rand malloc-test slab-test
############################################################################## ##############################################################################
setlocale : setlocale.o setlocale : setlocale.o
@echo "Linking $@" @echo "Linking $@"
$(CC) $(CFLAGS) -o $@ setlocale.o $(LIBS) -Wl,--cref,-M,-Map=$@.map $(CC) $(CFLAGS) -o $@ setlocale.o $(LIBS) $(LFLAGS) -Wl,--cref,-M,-Map=$@.map
test : test.o test : test.o
@echo "Linking $@" @echo "Linking $@"
$(CC) $(CFLAGS) -o $@ test.o $(LIBS) -Wl,--cref,-M,-Map=$@.map $(CC) $(CFLAGS) -o $@ test.o $(LIBS) $(LFLAGS) -Wl,--cref,-M,-Map=$@.map
date_test : date_test.o date_test : date_test.o
@echo "Linking $@" @echo "Linking $@"
$(CC) $(CFLAGS) -o $@ test.o $(LIBS) -Wl,--cref,-M,-Map=$@.map $(CC) $(CFLAGS) -o $@ test.o $(LIBS) $(LFLAGS) -Wl,--cref,-M,-Map=$@.map
fgets_test : fgets_test.o fgets_test : fgets_test.o
@echo "Linking $@" @echo "Linking $@"
$(CC) $(CFLAGS) -o $@ fgets_test.o $(LIBS) -Wl,--cref,-M,-Map=$@.map $(CC) $(CFLAGS) -o $@ fgets_test.o $(LIBS) $(LFLAGS) -Wl,--cref,-M,-Map=$@.map
strtok_test : strtok_test.o strtok_test : strtok_test.o
@echo "Linking $@" @echo "Linking $@"
$(CC) $(CFLAGS) -o $@ strtok_test.o $(LIBS) -Wl,--cref,-M,-Map=$@.map $(CC) $(CFLAGS) -o $@ strtok_test.o $(LIBS) $(LFLAGS) -Wl,--cref,-M,-Map=$@.map
iotest : iotest.o iotest : iotest.o
@echo "Linking $@" @echo "Linking $@"
$(CC) $(CFLAGS) -o $@ iotest.o $(LIBS) -Wl,--cref,-M,-Map=$@.map $(CC) $(CFLAGS) -o $@ iotest.o $(LIBS) $(LFLAGS) -Wl,--cref,-M,-Map=$@.map
sscanf_test : sscanf_test.o sscanf_test : sscanf_test.o
@echo "Linking $@" @echo "Linking $@"
$(CC) $(CFLAGS) -o $@ sscanf_test.o $(LIBS) -Wl,--cref,-M,-Map=$@.map $(CC) $(CFLAGS) -o $@ sscanf_test.o $(LIBS) $(LFLAGS) -Wl,--cref,-M,-Map=$@.map
printf_test : printf_test.o printf_test : printf_test.o
@echo "Linking $@" @echo "Linking $@"
$(CC) $(CFLAGS) -o $@ printf_test.o $(LIBS) -Wl,--cref,-M,-Map=$@.map $(CC) $(CFLAGS) -o $@ printf_test.o $(LIBS) $(LFLAGS) -Wl,--cref,-M,-Map=$@.map
sprintf_test : sprintf_test.o sprintf_test : sprintf_test.o
@echo "Linking $@" @echo "Linking $@"
$(CC) $(CFLAGS) -o $@ sprintf_test.o $(LIBS) -Wl,--cref,-M,-Map=$@.map $(CC) $(CFLAGS) -o $@ sprintf_test.o $(LIBS) $(LFLAGS) -Wl,--cref,-M,-Map=$@.map
stack_size_test : stack_size_test.o stack_size_test : stack_size_test.o
@echo "Linking $@" @echo "Linking $@"
$(CC) $(CFLAGS) -o $@ stack_size_test.o $(LIBS) -Wl,--cref,-M,-Map=$@.map $(CC) $(CFLAGS) -o $@ stack_size_test.o $(LIBS) $(LFLAGS) -Wl,--cref,-M,-Map=$@.map
translate_test : translate_test.o translate_test : translate_test.o
@echo "Linking $@" @echo "Linking $@"
$(CC) $(CFLAGS) -o $@ translate_test.o -lunix $(LIBS) -Wl,--cref,-M,-Map=$@.map $(CC) $(CFLAGS) -o $@ translate_test.o -lunix $(LIBS) $(LFLAGS) -Wl,--cref,-M,-Map=$@.map
uname : uname.o uname : uname.o
@echo "Linking $@" @echo "Linking $@"
$(CC) $(CFLAGS) -o $@ uname.o -lunix $(LIBS) -Wl,--cref,-M,-Map=$@.map $(CC) $(CFLAGS) -o $@ uname.o -lunix $(LIBS) $(LFLAGS) -Wl,--cref,-M,-Map=$@.map
simple : simple.o simple : simple.o
@echo "Linking $@" @echo "Linking $@"
$(CC) $(CFLAGS) -o $@ simple.o $(LIBS) -Wl,--cref,-M,-Map=$@.map $(CC) $(CFLAGS) -o $@ simple.o $(LIBS) $(LFLAGS) -Wl,--cref,-M,-Map=$@.map
fstat_stdout_test : fstat_stdout_test.o fstat_stdout_test : fstat_stdout_test.o
@echo "Linking $@" @echo "Linking $@"
$(CC) $(CFLAGS) -o $@ fstat_stdout_test.o $(LIBS) -Wl,--cref,-M,-Map=$@.map $(CC) $(CFLAGS) -o $@ fstat_stdout_test.o $(LIBS) $(LFLAGS) -Wl,--cref,-M,-Map=$@.map
simple_sprintf : simple_sprintf.o simple_sprintf : simple_sprintf.o
@echo "Linking $@" @echo "Linking $@"
$(CC) -nostdlib $(CFLAGS) -o $@ simple_sprintf.o -lc -lgcc -Wl,--cref,-M,-Map=$@.map $(CC) $(CFLAGS) -nostdlib -o $@ simple_sprintf.o -lc -lgcc $(LFLAGS) -Wl,--cref,-M,-Map=$@.map
sscanf_64 : sscanf_64.o sscanf_64 : sscanf_64.o
@echo "Linking $@" @echo "Linking $@"
$(CC) -nostdlib $(CFLAGS) -o $@ sscanf_64.o -lc -lgcc -Wl,--cref,-M,-Map=$@.map $(CC) $(CFLAGS) -o $@ sscanf_64.o $(LIBS) $(LFLAGS) -Wl,--cref,-M,-Map=$@.map
factorial : factorial.o factorial : factorial.o
@echo "Linking $@" @echo "Linking $@"
$(CC) -nostdlib $(CFLAGS) -o $@ factorial.o -lc -lgcc -Wl,--cref,-M,-Map=$@.map $(CC) $(CFLAGS) -o $@ factorial.o $(LIBS) $(LFLAGS) -Wl,--cref,-M,-Map=$@.map
rand : rand.o rand : rand.o
@echo "Linking $@" @echo "Linking $@"
$(CC) -nostdlib $(CFLAGS) -o $@ rand.o -lc -lgcc -Wl,--cref,-M,-Map=$@.map $(CC) $(CFLAGS) -o $@ rand.o $(LIBS) $(LFLAGS) -Wl,--cref,-M,-Map=$@.map
malloc-test : malloc-test.o
@echo "Linking $@"
$(CC) $(CFLAGS) -o $@ malloc-test.o $(LIBS) $(LFLAGS) -Wl,--cref,-M,-Map=$@.map
slab-test : slab-test.o
@echo "Linking $@"
$(CC) $(CFLAGS) -o $@ slab-test.o $(LIBS) $(LFLAGS) -Wl,--cref,-M,-Map=$@.map

90
test_programs/slab-test.c Normal file
View File

@@ -0,0 +1,90 @@
#include <stdlib.h>
#include <stdio.h>
unsigned long __slab_max_size = 4096;
static int print_json(void * ignore,const char * buffer,size_t len)
{
fputs(buffer, stdout);
return(0);
}
int
main(int argc,char ** argv)
{
int num_allocations = 1000;
int max_allocation_size = 8192;
int random_free_percentage = 33;
char ** allocation_table;
char * allocation;
int i;
allocation_table = malloc(sizeof(*allocation_table) * num_allocations);
if(allocation_table == NULL)
exit(EXIT_FAILURE);
srand(1);
printf("/* Allocating %d random length fragments of memory (maximum size = %ld bytes). */\n", num_allocations, max_allocation_size);
for(i = 0 ; i < num_allocations ; i++)
{
allocation = malloc(1 + (rand() % max_allocation_size));
if(allocation == NULL)
exit(EXIT_FAILURE);
allocation_table[i] = allocation;
}
__get_slab_stats(NULL, print_json);
printf("\n/* Changing all allocations to different random lengths. */\n");
for(i = 0 ; i < num_allocations ; i++)
{
allocation = realloc(allocation_table[i], 1 + (rand() % max_allocation_size));
if(allocation == NULL)
exit(EXIT_FAILURE);
allocation_table[i] = allocation;
}
__get_slab_stats(NULL, print_json);
printf("\n/* Freeing %d%% of all allocations. */\n", random_free_percentage);
for(i = 0 ; i < num_allocations ; i++)
{
if((rand() % 100) < 33)
{
free(allocation_table[i]);
allocation_table[i] = NULL;
}
}
__get_slab_stats(NULL, print_json);
printf("\n/* Marking unused slabs for reuse; reallocating memory/changing allocation lengths. */\n");
__decay_unused_slabs();
for(i = 0 ; i < num_allocations ; i++)
{
allocation = realloc(allocation_table[i], 1 + (rand() % max_allocation_size));
if(allocation == NULL)
exit(EXIT_FAILURE);
allocation_table[i] = allocation;
}
__get_slab_stats(NULL, print_json);
printf("\n/* Freeing all unused slabs. */\n");
__free_unused_slabs();
__get_slab_stats(NULL, print_json);
return(EXIT_SUCCESS);
}

View File

@@ -1,6 +1,4 @@
# #
# $Id: smakefile,v 1.9 2006-01-02 13:11:39 obarthel Exp $
#
# :ts=8 # :ts=8
# #
@@ -10,6 +8,10 @@
@echo "Compiling $<" @echo "Compiling $<"
@sc nover $(CFLAGS) $< @sc nover $(CFLAGS) $<
.c.mo:
@echo "Compiling $<"
@sc nover $(CFLAGS) objname=$*.mo math=IEEE $<
.asm.o: .asm.o:
@echo "Assembling $<" @echo "Assembling $<"
@asm $(AFLAGS) $< @asm $(AFLAGS) $<
@@ -19,30 +21,31 @@
# You might want to change this to the directory where your operating system # You might want to change this to the directory where your operating system
# header files are stored. On my system, that's "V:include", but you might # header files are stored. On my system, that's "V:include", but you might
# get lucky with "sc:include" instead, which is the default for SAS/C. # get lucky with "sc:include" instead, which is the default for SAS/C.
INCLUDE_DIR = V:include #INCLUDE_DIR = V:include
#INCLUDE_DIR = sc:include INCLUDE_DIR = sc:include
############################################################################## ##############################################################################
# This is where the header files, the startup code and the c.lib files are # This is where the header files, the startup code and the c.lib files are
# stored; see below how this prefix is used. # stored; see below how this prefix is used.
LIB = /library/ LIB = /library
STARTUP = $(LIB)/startup.o
LIBS = $(LIB)/amiga.lib $(LIB)/c.lib $(LIB)/debug.lib
MATH_LIBS = $(LIB)/math.lib $(LIBS)
############################################################################## ##############################################################################
OPTIMIZE = optimize opttime optschedule optinline OPTIMIZE = optimize opttime optschedule optinline
#DEBUG = debug=line noopt define=CHECK_FOR_NULL_POINTERS #DEBUG = debug=line noopt define=CHECK_FOR_NULL_POINTERS
#DEBUG = debug=line #DEBUG = debug=line
#DEBUG = debug=line define=NDEBUG DEBUG = debug=line define=NDEBUG
DEBUG = debug=sf noopt #DEBUG = debug=sf noopt
#DEBUG = debug=sf noopt define=CHECK_FOR_NULL_POINTERS #DEBUG = debug=sf noopt define=CHECK_FOR_NULL_POINTERS
#PROFILE = profile #PROFILE = profile
DATA = data=faronly DATA = data=faronly
#CODE = code=far #CODE = code=far
CPU = cpu=060 CPU = cpu=030
MATH = define=IEEE_FLOATING_POINT_SUPPORT math=IEEE
SUPPORT = define=UNIX_PATH_SEMANTICS define=SOCKET_SUPPORT define=USERGROUP_SUPPORT \
define=__C_MACROS__
############################################################################## ##############################################################################
@@ -55,8 +58,9 @@ CFLAGS = \
nostackcheck \ nostackcheck \
stringmerge \ stringmerge \
errorrexx \ errorrexx \
$(PROFILE) $(OPTIMIZE) $(CODE) $(DATA) $(CPU) $(MATH) \ $(PROFILE) $(OPTIMIZE) $(CODE) $(DATA) $(CPU) $(DEBUG) \
$(SUPPORT) $(DEBUG) math=ieee \
define=VERBOSE
AFLAGS = \ AFLAGS = \
-d -m2 -d -m2
@@ -68,19 +72,23 @@ all: \
test fgets_test iotest sscanf_test printf_test sprintf_test \ test fgets_test iotest sscanf_test printf_test sprintf_test \
stack_size_test translate_test strtok_test uname simple \ stack_size_test translate_test strtok_test uname simple \
fstat_stdout_test simple_sprintf date_test factorial \ fstat_stdout_test simple_sprintf date_test factorial \
execvp_test setlocale rand fstat_test base_dir_nametest \
malloc-test slab-test \
cleanup cleanup
clean: clean:
-delete \#?.o \#?.map \ -delete \#?.o \#?.mo \#?.map \
test fgets_test iotest sscanf_test printf_test sprintf_test \ test fgets_test iotest sscanf_test printf_test sprintf_test \
stack_size_test translate_test strtok_test uname simple \ stack_size_test translate_test strtok_test uname simple \
simple_sprintf date_test factorial fstat_stdout_test simple_sprintf date_test factorial \
execvp_test setlocale rand fstat_test base_dir_nametest \
malloc-test slab-test
############################################################################## ##############################################################################
setup: setup:
@echo "Setting up include: assignment" @echo "Setting up include: assignment"
@assign include: $(LIB)include V:include @assign include: $(LIB)/include $(INCLUDE_DIR)
cleanup: cleanup:
@echo "Cleaning up include: assignment" @echo "Cleaning up include: assignment"
@@ -90,77 +98,112 @@ cleanup:
test: test.o test: test.o
@echo "Linking $@" @echo "Linking $@"
@slink $(LIB)startup.o test.o to $@ lib $(LIB)c.lib addsym \ @slink $(STARTUP) $*.o to $@ lib $(LIBS) addsym \
map $@.map,fhx fwidth 32 pwidth 32 swidth 32 map $@.map,fhx fwidth 32 pwidth 32 swidth 32
date_test: date_test.o date_test: date_test.o
@echo "Linking $@" @echo "Linking $@"
@slink $(LIB)startup.o date_test.o to $@ lib $(LIB)c.lib addsym \ @slink $(STARTUP) $*.o to $@ lib $(LIBS) addsym \
map $@.map,fhx fwidth 32 pwidth 32 swidth 32 map $@.map,fhx fwidth 32 pwidth 32 swidth 32
fgets_test: fgets_test.o fgets_test: fgets_test.o
@echo "Linking $@" @echo "Linking $@"
@slink $(LIB)startup.o fgets_test.o to $@ lib $(LIB)c.lib addsym \ @slink $(STARTUP) $*.o to $@ lib $(LIBS) addsym \
map $@.map,fhx fwidth 32 pwidth 32 swidth 32 map $@.map,fhx fwidth 32 pwidth 32 swidth 32
strtok_test: strtok_test.o strtok_test: strtok_test.o
@echo "Linking $@" @echo "Linking $@"
@slink $(LIB)startup.o strtok_test.o to $@ lib $(LIB)c.lib addsym \ @slink $(STARTUP) $*.o to $@ lib $(LIBS) addsym \
map $@.map,fhx fwidth 32 pwidth 32 swidth 32 map $@.map,fhx fwidth 32 pwidth 32 swidth 32
iotest: iotest.o iotest: iotest.o
@echo "Linking $@" @echo "Linking $@"
@slink $(LIB)startup.o iotest.o to $@ lib $(LIB)c.lib addsym \ @slink $(STARTUP) $*.o to $@ lib $(LIBS) addsym \
map $@.map,fhx fwidth 32 pwidth 32 swidth 32 map $@.map,fhx fwidth 32 pwidth 32 swidth 32
sscanf_test: sscanf_test.o sscanf_test: sscanf_test.mo
@echo "Linking $@" @echo "Linking $@"
@slink $(LIB)startup.o sscanf_test.o to $@ lib $(LIB)c.lib addsym \ @slink $(STARTUP) $*.mo to $@ lib $(MATH_LIBS) addsym \
map $@.map,fhx fwidth 32 pwidth 32 swidth 32 map $@.map,fhx fwidth 32 pwidth 32 swidth 32
printf_test: printf_test.o printf_test: printf_test.mo
@echo "Linking $@" @echo "Linking $@"
@slink $(LIB)startup.o printf_test.o to $@ lib $(LIB)c.lib addsym \ @slink $(STARTUP) $*.mo to $@ lib $(MATH_LIBS) addsym \
map $@.map,fhx fwidth 32 pwidth 32 swidth 32 map $@.map,fhx fwidth 32 pwidth 32 swidth 32
sprintf_test: sprintf_test.o sprintf_test: sprintf_test.o
@echo "Linking $@" @echo "Linking $@"
@slink $(LIB)startup.o sprintf_test.o to $@ lib $(LIB)c.lib addsym \ @slink $(STARTUP) $*.o to $@ lib $(LIBS) addsym \
map $@.map,fhx fwidth 32 pwidth 32 swidth 32 map $@.map,fhx fwidth 32 pwidth 32 swidth 32
stack_size_test: stack_size_test.o stack_size_test: stack_size_test.o
@echo "Linking $@" @echo "Linking $@"
@slink $(LIB)startup.o stack_size_test.o to $@ lib $(LIB)c.lib addsym \ @slink $(STARTUP) $*.o to $@ lib $(LIBS) addsym \
map $@.map,fhx fwidth 32 pwidth 32 swidth 32 map $@.map,fhx fwidth 32 pwidth 32 swidth 32
translate_test: translate_test.o translate_test: translate_test.o
@echo "Linking $@" @echo "Linking $@"
@slink $(LIB)startup.o translate_test.o to $@ lib $(LIB)c.lib addsym \ @slink $(STARTUP) $*.o to $@ lib $(LIB)/unix.lib $(LIBS) addsym \
map $@.map,fhx fwidth 32 pwidth 32 swidth 32 map $@.map,fhx fwidth 32 pwidth 32 swidth 32
uname: uname.o uname: uname.o
@echo "Linking $@" @echo "Linking $@"
@slink $(LIB)startup.o uname.o to $@ lib $(LIB)c.lib addsym \ @slink $(STARTUP) $*.o to $@ lib $(LIBS) addsym \
map $@.map,fhx fwidth 32 pwidth 32 swidth 32 map $@.map,fhx fwidth 32 pwidth 32 swidth 32
simple: simple.o simple: simple.o
@echo "Linking $@" @echo "Linking $@"
@slink $(LIB)startup.o simple.o to $@ lib $(LIB)c.lib addsym \ @slink $(STARTUP) $*.o to $@ lib $(LIBS) addsym \
map $@.map,fhx fwidth 32 pwidth 32 swidth 32 map $@.map,fhx fwidth 32 pwidth 32 swidth 32
fstat_stdout_test: fstat_stdout_test.o fstat_stdout_test: fstat_stdout_test.o
@echo "Linking $@" @echo "Linking $@"
@slink $(LIB)startup.o fstat_stdout_test.o to $@ lib $(LIB)c.lib addsym \ @slink $(STARTUP) $*.o to $@ lib $(LIBS) addsym \
map $@.map,fhx fwidth 32 pwidth 32 swidth 32 map $@.map,fhx fwidth 32 pwidth 32 swidth 32
simple_sprintf: simple_sprintf.o simple_sprintf: simple_sprintf.o
@echo "Linking $@" @echo "Linking $@"
@slink simple_sprintf.o to $@ lib $(LIB)c.lib addsym \ @slink $*.o to $@ lib $(LIBS) addsym \
map $@.map,fhx fwidth 32 pwidth 32 swidth 32 map $@.map,fhx fwidth 32 pwidth 32 swidth 32
factorial: factorial.o factorial: factorial.mo
@echo "Linking $@" @echo "Linking $@"
@slink $(LIB)startup.o factorial.o to $@ lib $(LIB)c.lib addsym \ @slink $(STARTUP) $*.mo to $@ lib $(MATH_LIBS) addsym \
map $@.map,fhx fwidth 32 pwidth 32 swidth 32
execvp_test: execvp_test.o
@echo "Linking $@"
@slink $(STARTUP) $*.o to $@ lib $(LIB)/unix.lib $(LIBS) addsym \
map $@.map,fhx fwidth 32 pwidth 32 swidth 32
setlocale: setlocale.o
@echo "Linking $@"
@slink $(STARTUP) $*.o to $@ lib $(LIBS) addsym \
map $@.map,fhx fwidth 32 pwidth 32 swidth 32
rand: rand.o
@echo "Linking $@"
@slink $(STARTUP) $*.o to $@ lib $(LIBS) addsym \
map $@.map,fhx fwidth 32 pwidth 32 swidth 32
fstat_test: fstat_test.o
@echo "Linking $@"
@slink $(STARTUP) $*.o to $@ lib $(LIBS) addsym \
map $@.map,fhx fwidth 32 pwidth 32 swidth 32
base_dir_nametest: base_dir_nametest.o
@echo "Linking $@"
@slink $(STARTUP) $*.o to $@ lib $(LIBS) addsym \
map $@.map,fhx fwidth 32 pwidth 32 swidth 32
malloc-test: malloc-test.mo
@echo "Linking $@"
@slink $(STARTUP) $*.mo to $@ lib $(MATH_LIBS) addsym \
map $@.map,fhx fwidth 32 pwidth 32 swidth 32
slab-test: slab-test.o
@echo "Linking $@"
@slink $(STARTUP) $*.o to $@ lib $(LIBS) addsym \
map $@.map,fhx fwidth 32 pwidth 32 swidth 32 map $@.map,fhx fwidth 32 pwidth 32 swidth 32
############################################################################## ##############################################################################

View File

@@ -77,6 +77,8 @@ main(int argc,char ** argv)
long n,r; long n,r;
char time_buffer[100]; char time_buffer[100];
free(malloc(4));
for(i = 0 ; i < argc ; i++) for(i = 0 ; i < argc ; i++)
printf("%2d) \"%s\"\n",i,argv[i]); printf("%2d) \"%s\"\n",i,argv[i]);