mirror of
				https://gitlab.com/freepascal.org/fpc/source.git
				synced 2025-10-26 11:51:37 +01:00 
			
		
		
		
	 380bc56e32
			
		
	
	
		380bc56e32
		
	
	
	
	
		
			
			* sysutils for the embedded target depends on a working heap manager git-svn-id: trunk@23853 -
		
			
				
	
	
		
			254 lines
		
	
	
		
			6.4 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
			
		
		
	
	
			254 lines
		
	
	
		
			6.4 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
| {
 | |
|     This file is part of the Free Pascal run time library.
 | |
|     Copyright (c) 2011 by the Free Pascal development team.
 | |
| 
 | |
|     Tiny heap manager for the FPC embedded target
 | |
| 
 | |
|     See the file COPYING.FPC, included in this distribution,
 | |
|     for details about the copyright.
 | |
| 
 | |
|     This program 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.
 | |
| 
 | |
|  **********************************************************************}
 | |
| {$mode objfpc}
 | |
| Unit heapmgr;
 | |
| 
 | |
|   interface
 | |
| 
 | |
|     procedure RegisterHeapBlock(AAddress: pointer; ASize: ptruint);
 | |
| 
 | |
|   implementation
 | |
| 
 | |
|     const
 | |
|       MinBlock = 16;
 | |
| 
 | |
|     type
 | |
|       PHeapBlock = ^THeapBlock;
 | |
|       THeapBlock = record
 | |
|         Size: ptruint;
 | |
|         Next: PHeapBlock;
 | |
|         EndAddr: pointer;
 | |
|       end;
 | |
| 
 | |
|     var
 | |
|       Blocks: PHeapBlock = nil;
 | |
| 
 | |
|     procedure InternalFreeMem(Addr: Pointer; Size: ptruint); forward;
 | |
| 
 | |
|     function FindSize(p: pointer): ptruint;
 | |
|       begin
 | |
|         FindSize := PPtrUInt(p)[-1];
 | |
|       end;
 | |
| 
 | |
|     function SysGetMem(Size: ptruint): pointer;
 | |
|       var
 | |
|         p, prev: PHeapBlock;
 | |
|         AllocSize, RestSize: ptruint;
 | |
|       begin
 | |
|         AllocSize := align(size+sizeof(ptruint), sizeof(pointer));
 | |
| 
 | |
|         p := Blocks;
 | |
|         prev := nil;
 | |
|         while assigned(p) and (p^.Size < AllocSize) do
 | |
|           begin
 | |
|             prev := p;
 | |
|             p := p^.Next;
 | |
|           end;
 | |
| 
 | |
|         if assigned(p) then
 | |
|           begin
 | |
|             result := @pptruint(p)[1];
 | |
| 
 | |
|             if p^.Size-AllocSize >= MinBlock then
 | |
|               RestSize := p^.Size-AllocSize
 | |
|             else
 | |
|               begin
 | |
|                 AllocSize := p^.Size;
 | |
|                 RestSize := 0;
 | |
|               end;
 | |
| 
 | |
|             if prev = nil then
 | |
|               Blocks := p^.Next
 | |
|             else
 | |
|               prev^.next := p^.next;
 | |
| 
 | |
|             pptruint(p)^ := size;
 | |
| 
 | |
|             InternalFreemem(pointer(ptruint(p)+AllocSize), RestSize);
 | |
|           end
 | |
|         else
 | |
|           Result := nil;
 | |
|       end;
 | |
| 
 | |
|     function GetAlignedMem(Size, Alignment: ptruint): pointer;
 | |
|       var
 | |
|         mem: Pointer;
 | |
|         memp: ptruint;
 | |
|       begin
 | |
|         if alignment <= sizeof(pointer) then
 | |
|           result := GetMem(size)
 | |
|         else
 | |
|           begin
 | |
|             mem := GetMem(Size+Alignment-1);
 | |
|             memp := align(ptruint(mem), Alignment);
 | |
|             InternalFreemem(mem, ptruint(memp)-ptruint(mem));
 | |
|             result := pointer(memp);
 | |
|           end;
 | |
|       end;
 | |
| 
 | |
|     procedure InternalFreeMem(Addr: Pointer; Size: ptruint);
 | |
|       var
 | |
|         b, p, prev: PHeapBlock;
 | |
|         concatenated: boolean;
 | |
|       begin
 | |
|         concatenated := true;
 | |
|         while concatenated do
 | |
|           begin
 | |
|             concatenated := false;
 | |
|             b := addr;
 | |
| 
 | |
|             b^.Next := Blocks;
 | |
|             b^.Size := Size;
 | |
|             b^.EndAddr := pointer(ptruint(addr)+size);
 | |
| 
 | |
|             if Blocks = nil then
 | |
|               Blocks := b
 | |
|             else
 | |
|               begin
 | |
|                 p := Blocks;
 | |
|                 prev := nil;
 | |
| 
 | |
|                 while assigned(p) do
 | |
|                   begin
 | |
|                     if p^.EndAddr = addr then
 | |
|                       begin
 | |
|                         addr:=p;
 | |
|                         size:=p^.size+size;
 | |
|                         if prev = nil then
 | |
|                           blocks:=p^.next
 | |
|                         else
 | |
|                           prev^.next:=p^.next;
 | |
|                         concatenated:=true;
 | |
|                         break;
 | |
|                       end
 | |
|                     else if p = b^.EndAddr then
 | |
|                       begin
 | |
|                         size:=p^.size+size;
 | |
|                         if prev = nil then
 | |
|                           blocks:=p^.next
 | |
|                         else
 | |
|                           prev^.next:=p^.next;
 | |
|                         concatenated:=true;
 | |
|                         break;
 | |
|                       end;
 | |
| 
 | |
|                     prev := p;
 | |
|                     p := p^.next;
 | |
|                   end;
 | |
| 
 | |
|                 if not concatenated then
 | |
|                   begin
 | |
|                     p := Blocks;
 | |
|                     prev := nil;
 | |
| 
 | |
|                     while assigned(p) and (p^.Size < size) do
 | |
|                       begin
 | |
|                         prev := p;
 | |
|                         p := p^.Next;
 | |
|                       end;
 | |
| 
 | |
|                     if assigned(prev) then
 | |
|                       begin
 | |
|                         b^.Next := p;
 | |
|                         prev^.Next := b;
 | |
|                       end
 | |
|                     else
 | |
|                       Blocks := b;
 | |
|                   end;
 | |
|               end;
 | |
|           end;
 | |
|       end;
 | |
| 
 | |
|     function SysFreeMem(Addr: Pointer): ptruint;
 | |
|       var
 | |
|         sz: ptruint;
 | |
|       begin
 | |
|         sz := Align(FindSize(addr)+SizeOf(ptruint), sizeof(pointer));
 | |
| 
 | |
|         InternalFreeMem(@pptruint(addr)[-1], sz);
 | |
| 
 | |
|         result := sz;
 | |
|       end;
 | |
| 
 | |
|     function SysFreeMemSize(Addr: Pointer; Size: Ptruint): ptruint;
 | |
|       begin
 | |
|         result := SysFreeMem(addr);
 | |
|       end;
 | |
| 
 | |
|     function SysMemSize(p: pointer): ptruint;
 | |
|       begin
 | |
|         result := findsize(p);
 | |
|       end;
 | |
| 
 | |
|     function SysAllocMem(size: ptruint): pointer;
 | |
|       begin
 | |
|         result := SysGetMem(size);
 | |
|         if result<>nil then
 | |
|           FillChar(result^,SysMemSize(result),0);
 | |
|       end;
 | |
| 
 | |
|     function SysReAllocMem(var p: pointer; size: ptruint):pointer;
 | |
|       var
 | |
|         sz: ptruint;
 | |
|       begin
 | |
|         result := AllocMem(size);
 | |
|         if result <> nil then
 | |
|           begin
 | |
|             if p <> nil then
 | |
|               begin
 | |
|                 sz := FindSize(p);
 | |
|                 if sz > size then
 | |
|                   sz := size;
 | |
|                 move(pbyte(p)^, pbyte(result)^, sz);
 | |
|               end;
 | |
|           end;
 | |
|         SysFreeMem(p);
 | |
|         p := result;
 | |
|       end;
 | |
| 
 | |
|     procedure RegisterHeapBlock(AAddress: pointer; ASize: ptruint);
 | |
|       begin
 | |
|         FreeMem(AAddress, ASize);
 | |
|       end;
 | |
| 
 | |
|     const
 | |
|       MyMemoryManager: TMemoryManager = (
 | |
|         NeedLock: false;  // Obsolete
 | |
|         GetMem: @SysGetMem;
 | |
|         FreeMem: @SysFreeMem;
 | |
|         FreeMemSize: @SysFreeMemSize;
 | |
|         AllocMem: @SysAllocMem;
 | |
|         ReAllocMem: @SysReAllocMem;
 | |
|         MemSize: @SysMemSize;
 | |
|         InitThread: nil;
 | |
|         DoneThread: nil;
 | |
|         RelocateHeap: nil;
 | |
|         GetHeapStatus: nil;
 | |
|         GetFPCHeapStatus: nil;
 | |
|       );
 | |
| 
 | |
| var
 | |
|   initialheap : record end; external name '__fpc_initialheap';
 | |
|   heapsize : PtrInt; external name '__heapsize';
 | |
| 
 | |
| 
 | |
| initialization
 | |
|   SetMemoryManager(MyMemoryManager);
 | |
|   RegisterHeapBlock(@initialheap,heapsize);
 | |
| finalization
 | |
|   //FinalizeHeap;
 | |
| end.
 | |
| 
 |