shithub: rgbds

ref: c31b87b16a9e96c2156ebc6c89e07bac86c10094
dir: /src/link/assign.c/

View raw version
#include <stdio.h>
#include <stdlib.h>
#include "mylink.h"
#include "main.h"
#include "symbol.h"
#include "assign.h"

struct	sFreeArea
{
	SLONG	nOrg;
	SLONG	nSize;
	struct	sFreeArea	*pPrev, *pNext;
};

struct	sFreeArea	*BankFree[MAXBANKS];
SLONG	MaxAvail[MAXBANKS];
SLONG	MaxBankUsed;

#define	DOMAXBANK(x)	{if( (x)>MaxBankUsed ) MaxBankUsed=(x);}

SLONG	area_Avail( SLONG bank )
{
	SLONG	r;
	struct	sFreeArea	*pArea;

	r=0;
	pArea=BankFree[bank];

	while( pArea )
	{
		r+=pArea->nSize;
		pArea=pArea->pNext;
	}

	return( r );
}

SLONG	area_AllocAbs( struct sFreeArea **ppArea, SLONG org, SLONG size )
{
	struct	sFreeArea	*pArea;

	pArea=*ppArea;
	while( pArea )
	{
		if( org>=pArea->nOrg && (org+size-1)<=(pArea->nOrg+pArea->nSize-1) )
		{
			if( org==pArea->nOrg )
			{
				pArea->nOrg+=size;
				pArea->nSize-=size;
				return( org );
			}
			else
			{
				if( (org+size-1)==(pArea->nOrg+pArea->nSize-1) )
				{
					pArea->nSize-=size;
					return( org );
				}
				else
				{
					struct	sFreeArea	*pNewArea;

					if( (pNewArea=(struct sFreeArea *)malloc(sizeof(struct sFreeArea)))!=NULL )
					{
						*pNewArea=*pArea;
						pNewArea->pPrev=pArea;
						pArea->pNext=pNewArea;
						pArea->nSize=org-pArea->nOrg;
						pNewArea->nOrg=org+size;
						pNewArea->nSize-=size+pArea->nSize;

						return( org );
					}
					else
						fatalerror( "Out of memory!" );
				}
			}
		}
		ppArea=&(pArea->pNext);
		pArea=*ppArea;
	}

	return( -1 );
}

SLONG	area_AllocAbsCODEAnyBank( SLONG org, SLONG size )
{
	SLONG	i;

	for( i=1; i<=255; i+=1 )
	{
		if( area_AllocAbs( &BankFree[i], org, size )==org )
			return( i );
	}

	return( -1 );
}

SLONG	area_Alloc( struct sFreeArea **ppArea, SLONG size )
{
	struct	sFreeArea	*pArea;

	pArea=*ppArea;
	while( pArea )
	{
		if( size<=pArea->nSize )
		{
			SLONG	r;

			r=pArea->nOrg;
			pArea->nOrg+=size;
			pArea->nSize-=size;

			return( r );
		}
		ppArea=&(pArea->pNext);
		pArea=*ppArea;
	}

	return( -1 );
}

SLONG	area_AllocCODEAnyBank( SLONG size )
{
	SLONG	i, org;

	for( i=1; i<=255; i+=1 )
	{
		if( (org=area_Alloc(&BankFree[i],size))!=-1 )
			return( (i<<16)|org );
	}

	return( -1 );
}

struct	sSection	*FindLargestCode( void )
{
	struct sSection *pSection, *r=NULL;
	SLONG	nLargest=0;

	pSection=pSections;
	while( pSection )
	{
		if( pSection->oAssigned==0 && pSection->Type==SECT_CODE )
		{
			if( pSection->nByteSize > nLargest )
			{
				nLargest=pSection->nByteSize;
				r=pSection;
			}
		}
		pSection=pSection->pNext;
	}
	return( r );
}


void AssignCodeSections( void )
{
	struct sSection *pSection;

	while( pSection=FindLargestCode() )
	{
		SLONG	org;

		if( (org=area_AllocCODEAnyBank( pSection->nByteSize ))!=-1 )
		{
			pSection->nOrg=org&0xFFFF;
			pSection->nBank=org>>16;
			pSection->oAssigned=1;
			DOMAXBANK(pSection->nBank);
		}
		else
			fatalerror( "Unable to place CODE section anywhere" );
	}
}

void GBROM_AssignSections( void )
{
	SLONG	i;
	struct sSection *pSection;

	MaxBankUsed=0;

	/*
	 * Initialize the memory areas
	 *
	 */

	for( i=0; i<MAXBANKS; i+=1 )
	{
		if( BankFree[i]=(struct sFreeArea *)malloc(sizeof(struct sFreeArea)) )
		{
			if( i==0 )
			{
				BankFree[i]->nOrg=0x0000;
				if( options&OPT_SMALL )
				{
					BankFree[i]->nSize=0x8000;
					MaxAvail[i]=0x8000;
				}
				else
				{
					BankFree[i]->nSize=0x4000;
					MaxAvail[i]=0x4000;
				}
			}
			else if( i>=1 && i<=255 )
			{
				BankFree[i]->nOrg=0x4000;
				/*
					* Now, this shouldn't really be necessary... but for good
					* measure we'll do it anyway
					*
					*/
				if( options&OPT_SMALL )
				{
					BankFree[i]->nSize=0;
					MaxAvail[i]=0;
				}
				else
				{
					BankFree[i]->nSize=0x4000;
					MaxAvail[i]=0x4000;
				}
			}
			else if( i==BANK_BSS )
			{
				BankFree[i]->nOrg =0xC000;
				BankFree[i]->nSize=0x2000;
				MaxAvail[i]=0x2000;
			}
			else if( i==BANK_VRAM )
			{
				BankFree[i]->nOrg =0x8000;
				BankFree[i]->nSize=0x2000;
				MaxAvail[i]=0x2000;
			}
			else if( i==BANK_HRAM )
			{
				BankFree[i]->nOrg =0xFF80;
				BankFree[i]->nSize=0x007F;
				MaxAvail[i]=0x007F;
			}
			BankFree[i]->pPrev=NULL;
			BankFree[i]->pNext=NULL;
		}
		else
			fatalerror( "Out of memory!" );
	}

	/*
	 * First, let's assign all the fixed sections...
	 * And all because of that Jens Restemeier character ;)
	 *
	 */

	pSection=pSections;
	while( pSection )
	{
		if( (pSection->nOrg!=-1 || pSection->nBank!=-1) && pSection->oAssigned==0 )
		{
			/* User wants to have a say... */

			switch( pSection->Type )
			{
				case SECT_BSS:
					if( area_AllocAbs(&BankFree[BANK_BSS],pSection->nOrg,pSection->nByteSize)!=pSection->nOrg )
					{
						sprintf( temptext, "Unable to load fixed BSS section at $%X", pSection->nOrg );
						fatalerror( temptext );
					}
					pSection->oAssigned=1;
					pSection->nBank=BANK_BSS;
					break;
				case SECT_HRAM:
					if( area_AllocAbs(&BankFree[BANK_HRAM],pSection->nOrg,pSection->nByteSize)!=pSection->nOrg )
					{
						sprintf( temptext, "Unable to load fixed HRAM section at $%X", pSection->nOrg );
						fatalerror( temptext );
					}
					pSection->oAssigned=1;
					pSection->nBank=BANK_HRAM;
					break;
				case SECT_VRAM:
					if( area_AllocAbs(&BankFree[BANK_VRAM],pSection->nOrg,pSection->nByteSize)!=pSection->nOrg )
					{
						sprintf( temptext, "Unable to load fixed VRAM section at $%X", pSection->nOrg );
						fatalerror( temptext );
					}
					pSection->oAssigned=1;
					pSection->nBank=BANK_VRAM;
					break;
				case SECT_HOME:
					if( area_AllocAbs(&BankFree[BANK_HOME],pSection->nOrg,pSection->nByteSize)!=pSection->nOrg )
					{
						sprintf( temptext, "Unable to load fixed HOME section at $%X", pSection->nOrg );
						fatalerror( temptext );
					}
					pSection->oAssigned=1;
					pSection->nBank=BANK_HOME;
					break;
				case SECT_CODE:
					if( pSection->nBank==-1 )
					{
						/*
						 * User doesn't care which bank, so he must want to
						 * decide which position within that bank.
						 * We'll do that at a later stage when the really
						 * hardcoded things are allocated
						 *
						 */
					}
					else
					{
						/*
						 * User wants to decide which bank we use
						 * Does he care about the position as well?
						 *
						 */

						if( pSection->nOrg==-1 )
						{
							/*
							 * Nope, any position will do
							 * Again, we'll do that later
							 *
							 */
						}
						else
						{
							/*
							 * How hardcore can you possibly get? Why does
							 * he even USE this package? Yeah let's just
							 * direct address everything, shall we?
							 * Oh well, the customer is always right
							 *
							 */

							if( pSection->nBank>=1 && pSection->nBank<=255 )
							{
								if( area_AllocAbs(&BankFree[pSection->nBank],pSection->nOrg,pSection->nByteSize)!=pSection->nOrg )
								{
									sprintf( temptext, "Unable to load fixed CODE/DATA section at $%X in bank $%02X", pSection->nOrg, pSection->nBank );
									fatalerror( temptext );
								}
								DOMAXBANK(pSection->nBank);
								pSection->oAssigned=1;
							}
							else
							{
								sprintf( temptext, "Unable to load fixed CODE/DATA section at $%X in bank $%02X", pSection->nOrg, pSection->nBank );
								fatalerror( temptext );
							}
						}

					}
					break;
			}
		}
		pSection=pSection->pNext;
	}

	/*
	 * Next, let's assign all the bankfixed ONLY CODE sections...
	 *
	 */

	pSection=pSections;
	while( pSection )
	{
		if( pSection->oAssigned==0
		&&	pSection->Type==SECT_CODE
		&&	pSection->nOrg==-1
		&&	pSection->nBank!=-1 )
		{
			/* User wants to have a say... and he's pissed */
			if( pSection->nBank>=1 && pSection->nBank<=255 )
			{
				if( (pSection->nOrg=area_Alloc(&BankFree[pSection->nBank],pSection->nByteSize))==-1 )
				{
					sprintf( temptext, "Unable to load fixed CODE/DATA section into bank $%02X", pSection->nBank );
					fatalerror( temptext );
				}
				pSection->oAssigned=1;
				DOMAXBANK(pSection->nBank);
			}
			else
			{
				sprintf( temptext, "Unable to load fixed CODE/DATA section into bank $%02X", pSection->nBank );
				fatalerror( temptext );
			}
		}
		pSection=pSection->pNext;
	}

	/*
	 * Now, let's assign all the floating bank but fixed CODE sections...
	 *
	 */

	pSection=pSections;
	while( pSection )
	{
		if( pSection->oAssigned==0
		&&	pSection->Type==SECT_CODE
		&&	pSection->nOrg!=-1
		&&	pSection->nBank==-1 )
		{
			/* User wants to have a say... and he's back with a vengeance */
			if( (pSection->nBank=area_AllocAbsCODEAnyBank(pSection->nOrg,pSection->nByteSize))==-1 )
			{
				sprintf( temptext, "Unable to load fixed CODE/DATA section at $%X into any bank", pSection->nOrg );
				fatalerror( temptext );
			}
			pSection->oAssigned=1;
			DOMAXBANK(pSection->nBank);
		}
		pSection=pSection->pNext;
	}

	/*
	 * OK, all that nasty stuff is done so let's assign all the other
	 * sections
	 *
	 */

	pSection=pSections;
	while( pSection )
	{
		if( pSection->oAssigned==0 )
		{
			switch( pSection->Type )
			{
				case SECT_BSS:
					if( (pSection->nOrg=area_Alloc(&BankFree[BANK_BSS],pSection->nByteSize))==-1 )
					{
						fatalerror( "BSS section too large\n" );
					}
					pSection->nBank=BANK_BSS;
					pSection->oAssigned=1;
					break;
				case SECT_HRAM:
					if( (pSection->nOrg=area_Alloc(&BankFree[BANK_HRAM],pSection->nByteSize))==-1 )
					{
						fatalerror( "HRAM section too large" );
					}
					pSection->nBank=BANK_HRAM;
					pSection->oAssigned=1;
					break;
				case SECT_VRAM:
					if( (pSection->nOrg=area_Alloc(&BankFree[BANK_VRAM],pSection->nByteSize))==-1 )
					{
						fatalerror( "VRAM section too large" );
					}
					pSection->nBank=BANK_VRAM;
					pSection->oAssigned=1;
					break;
				case SECT_HOME:
					if( (pSection->nOrg=area_Alloc(&BankFree[BANK_HOME],pSection->nByteSize))==-1 )
					{
						fatalerror( "HOME section too large" );
					}
					pSection->nBank=BANK_HOME;
					pSection->oAssigned=1;
					break;
				case SECT_CODE:
					break;
				default:
					fatalerror( "(INTERNAL) Unknown section type!" );
					break;
			}
		}
		pSection=pSection->pNext;
	}

	AssignCodeSections();
}

void PSION2_AssignSections( void )
{
	struct sSection *pSection;

	if( BankFree[0]=(struct sFreeArea *)malloc(sizeof(struct sFreeArea)) )
	{
		BankFree[0]->nOrg=0x0000;
		BankFree[0]->nSize=0x10000;
		MaxAvail[0]=0x10000;
		BankFree[0]->pPrev=NULL;
		BankFree[0]->pNext=NULL;

		pSection=pSections;
		while( pSection )
		{
			if( pSection->oAssigned==0 && pSection->Type==SECT_CODE )
			{
				pSection->oAssigned=1;
				pSection->nBank=0;
				pSection->nOrg=BankFree[0]->nOrg;
				BankFree[0]->nOrg+=pSection->nByteSize;
				BankFree[0]->nSize-=pSection->nByteSize;
			}
			pSection=pSection->pNext;
		}

		pSection=pSections;
		while( pSection )
		{
			if( pSection->oAssigned==0 && pSection->Type==SECT_BSS )
			{
				pSection->oAssigned=1;
				pSection->nBank=0;
				pSection->nOrg=BankFree[0]->nOrg;
				BankFree[0]->nOrg+=pSection->nByteSize;
				BankFree[0]->nSize-=pSection->nByteSize;
			}
			pSection=pSection->pNext;
		}
	}
	else
		fatalerror( "Out of memory!" );
}

void	AssignSections( void )
{
	switch( outputtype )
	{
		case OUTPUT_GBROM:
			GBROM_AssignSections();
			break;
		case OUTPUT_PSION2:
			PSION2_AssignSections();
			break;
	}
}

void CreateSymbolTable( void )
{
	struct sSection *pSect;

	sym_Init();

	pSect=pSections;

	while( pSect )
	{
		SLONG	i;

		i=pSect->nNumberOfSymbols;

		while( i-- )
		{
			if( (pSect->tSymbols[i]->Type==SYM_EXPORT) &&
				 ( (pSect->tSymbols[i]->pSection==pSect) ||
					(pSect->tSymbols[i]->pSection==NULL)) )
			{
				if( pSect->tSymbols[i]->pSection==NULL )
					sym_CreateSymbol( pSect->tSymbols[i]->pzName,
											pSect->tSymbols[i]->nOffset,
											-1 );
				else
					sym_CreateSymbol( pSect->tSymbols[i]->pzName,
											pSect->nOrg+pSect->tSymbols[i]->nOffset,
											pSect->nBank );
			}
		}
		pSect=pSect->pNext;
	}
}