ref: 4d06e6de796e9e96f03f419b976bbc3dc19d2afe
dir: /src/link/assign.c/
#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; } }