/*****************************************************************************/ /* */ /* data.c */ /* */ /* Data output routines */ /* */ /* */ /* */ /* (C) 2000 Ullrich von Bassewitz */ /* Wacholderweg 14 */ /* D-70597 Stuttgart */ /* EMail: uz@musoftware.de */ /* */ /* */ /* This software is provided 'as-is', without any expressed or implied */ /* warranty. In no event will the authors be held liable for any damages */ /* arising from the use of this software. */ /* */ /* Permission is granted to anyone to use this software for any purpose, */ /* including commercial applications, and to alter it and redistribute it */ /* freely, subject to the following restrictions: */ /* */ /* 1. The origin of this software must not be misrepresented; you must not */ /* claim that you wrote the original software. If you use this software */ /* in a product, an acknowledgment in the product documentation would be */ /* appreciated but is not required. */ /* 2. Altered source versions must be plainly marked as such, and must not */ /* be misrepresented as being the original software. */ /* 3. This notice may not be removed or altered from any source */ /* distribution. */ /* */ /*****************************************************************************/ /* da65 */ #include "attrtab.h" #include "code.h" #include "error.h" #include "global.h" #include "output.h" #include "data.h" /*****************************************************************************/ /* Code */ /*****************************************************************************/ static unsigned GetSpan (attr_t Style) /* Get the number of bytes for a given style */ { /* Get the number of bytes still available */ unsigned RemainingBytes = GetRemainingBytes (); /* Count how many bytes are available. This number is limited by the * number of remaining bytes, a label, or the end of the given Style * attribute. */ unsigned Count = 1; while (Count < RemainingBytes) { if (HaveLabel(PC+Count) || GetStyle (PC+Count) != Style) { break; } ++Count; } /* Return the number of bytes */ return Count; } static unsigned DoTable (attr_t Style, unsigned MemberSize, void (*TableFunc) (unsigned)) /* Output a table of bytes */ { unsigned BytesLeft; /* Count how many bytes may be output. */ unsigned Count = GetSpan (Style); /* Make Count an even number of multiples of MemberSize */ Count &= ~(MemberSize-1); /* Output as many data bytes lines as needed */ BytesLeft = Count; while (BytesLeft > 0) { /* Calculate the number of bytes for the next line */ unsigned Chunk = (BytesLeft > BytesPerLine)? BytesPerLine : BytesLeft; /* Output a line with these bytes */ TableFunc (Chunk); /* Next line */ BytesLeft -= Chunk; PC += Chunk; } /* If the next line is not the same style, add a separator */ if (CodeLeft() && GetStyle (PC) != Style) { SeparatorLine (); } /* Return the number of bytes output */ return Count; } unsigned ByteTable (void) /* Output a table of bytes */ { /* Call the low level function */ return DoTable (atByteTab, 1, DataByteLine); } unsigned WordTable (void) /* Output a table of words */ { /* Call the low level function */ return DoTable (atWordTab, 2, DataWordLine); } unsigned DWordTable (void) /* Output a table of double words */ { /* Call the low level function */ return DoTable (atDWordTab, 4, DataDWordLine); } unsigned AddrTable (void) /* Output a table of addresses */ { unsigned BytesLeft; /* Count how many bytes may be output. */ unsigned Count = GetSpan (atAddrTab); /* Make the given number even */ Count &= ~1U; /* Output as many data bytes lines as needed. For addresses, each line * will hold just one address. */ BytesLeft = Count; while (BytesLeft > 0) { /* Get the address */ unsigned Addr = GetCodeWord (PC); /* In pass 1, define a label, in pass 2 output the line */ if (Pass == 1) { if (!HaveLabel (Addr)) { AddLabel (Addr, MakeLabelName (Addr)); } } else { const char* Label = GetLabel (Addr); if (Label == 0) { /* OOPS! Should not happen */ Internal ("OOPS - Label for address %04X disappeard!", Addr); } Indent (MIndent); Output (".word"); Indent (AIndent); Output ("%s", Label); LineComment (PC, 2); LineFeed (); } /* Next line */ PC += 2; BytesLeft -= 2; } /* If the next line is not a byte table line, add a separator */ if (CodeLeft() && GetStyle (PC) != atAddrTab) { SeparatorLine (); } /* Return the number of bytes output */ return Count; }