Example #1
0
        /// <summary>
        /// Prepares the loaded form of the binary and the disassembly project.
        /// </summary>
        public bool Prepare()
        {
            if (!CreateMap())
            {
                mSegmentMap = null;
                return(false);
            }

            Debug.WriteLine("Segment map:");
            for (int i = 0; i < mSegmentMap.Count; i++)
            {
                SegmentMapEntry ent = mSegmentMap[i];
                if (ent == null)
                {
                    Debug.Assert(i == 0 || i == 1);     // initial hole and optional ~ExpressLoad
                    continue;
                }
                OmfSegment omfSeg = ent.Segment;
                Debug.WriteLine(i + " " + ent.Address.ToString("x6") + " SegNum=" + omfSeg.SegNum +
                                " '" + omfSeg.SegName + "'");

                Debug.Assert(i == ent.Segment.SegNum);
            }

            if (!GenerateDataAndProject())
            {
                mSegmentMap = null;
                return(false);
            }

            return(true);
        }
Example #2
0
        /// <summary>
        /// Adds one or more entries to the address map for the specified segment.
        /// </summary>
        private static void AddAddressEntries(DisasmProject proj, SegmentMapEntry ent,
                                              int bufOffset, ChangeSet cs)
        {
            int addr   = ent.Address;
            int segRem = ent.Segment.Length;

            while (true)
            {
                // Generate an ORG directive.
                int            origAddr = proj.AddrMap.Get(bufOffset);
                UndoableChange uc       = UndoableChange.CreateAddressChange(bufOffset,
                                                                             origAddr, addr);
                cs.Add(uc);

                // Compare amount of space in this bank to amount left in segment.
                int bankRem = 0x00010000 - (addr & 0xffff);
                if (bankRem > segRem)
                {
                    // All done, bail.
                    break;
                }

                // Advance to start of next bank.
                addr += bankRem;
                Debug.Assert((addr & 0x0000ffff) == 0);
                bufOffset += bankRem;
                segRem    -= bankRem;
                Debug.WriteLine("Adding additional ORG at " + addr);
            }
        }
Example #3
0
        /// <summary>
        /// Edits the data file, changing values based on the relocation dictionary.
        /// </summary>
        private bool RelocSegment(SegmentMapEntry ent, byte[] data, int bufOffset)
        {
            const int INVALID_ADDR = 0x00ffffff;

            byte[] srcData = ent.Segment.GetConstData();
            Array.Copy(srcData, 0, data, bufOffset, srcData.Length);

            foreach (OmfReloc omfRel in ent.Segment.Relocs)
            {
                int relocAddr = omfRel.RelOffset;
                if (omfRel.FileNum != -1 && omfRel.FileNum != 1)
                {
                    // Some other file; not much we can do with this.  Drop in an obviously
                    // invalid address and keep going.
                    Debug.WriteLine("Unable to process reloc with FileNum=" + omfRel.FileNum);
                    relocAddr = INVALID_ADDR;
                }
                else if (omfRel.SegNum == -1)
                {
                    // Within this segment.
                    relocAddr += ent.Address;
                }
                else
                {
                    // Find other segment.  This may fail if the file is damaged.
                    if (omfRel.SegNum < 0 || omfRel.SegNum >= mSegmentMap.Count ||
                        mSegmentMap[omfRel.SegNum] == null)
                    {
                        // Can't find the segment.  Unlike the file case, this was expected to
                        // be something we could resolve with what we were given, so this is
                        // a hard failure.
                        Debug.WriteLine("Reloc SegNum=" + omfRel.SegNum + " not in map");
                        return(false);
                    }
                    else
                    {
                        relocAddr += mSegmentMap[omfRel.SegNum].Address;
                    }
                }

                if (omfRel.Shift < -32 || omfRel.Shift > 32)
                {
                    Debug.WriteLine("Invalid reloc shift " + omfRel.Shift);
                    return(false);
                }
                int adjRelocAddr = relocAddr;
                if (omfRel.Shift < 0)
                {
                    adjRelocAddr >>= -omfRel.Shift;
                }
                else if (omfRel.Shift > 0)
                {
                    adjRelocAddr <<= omfRel.Shift;
                }

                switch (omfRel.Width)
                {
                case 1:
                    data[bufOffset + omfRel.Offset] = (byte)(adjRelocAddr);
                    break;

                case 2:
                    data[bufOffset + omfRel.Offset]     = (byte)(adjRelocAddr);
                    data[bufOffset + omfRel.Offset + 1] = (byte)(adjRelocAddr >> 8);
                    break;

                case 3:
                    data[bufOffset + omfRel.Offset]     = (byte)(adjRelocAddr);
                    data[bufOffset + omfRel.Offset + 1] = (byte)(adjRelocAddr >> 8);
                    data[bufOffset + omfRel.Offset + 2] = (byte)(adjRelocAddr >> 16);
                    break;

                case 4:
                    data[bufOffset + omfRel.Offset]     = (byte)(adjRelocAddr);
                    data[bufOffset + omfRel.Offset + 1] = (byte)(adjRelocAddr >> 8);
                    data[bufOffset + omfRel.Offset + 2] = (byte)(adjRelocAddr >> 16);
                    data[bufOffset + omfRel.Offset + 3] = (byte)(adjRelocAddr >> 24);
                    break;

                default:
                    Debug.WriteLine("Invalid reloc width " + omfRel.Width);
                    return(false);
                }

                mRelocData.Add(bufOffset + omfRel.Offset, new DisasmProject.RelocData(
                                   (byte)omfRel.Width, (sbyte)omfRel.Shift, relocAddr));
            }

            return(true);
        }
Example #4
0
        /// <summary>
        /// Edits the data file, essentially putting the jump table entries into the
        /// "loaded" state.
        /// </summary>
        /// <remarks>
        /// We don't use ent.Segment.Relocs, as that is expected to be empty.
        /// </remarks>
        private bool RelocJumpTable(SegmentMapEntry ent, byte[] data, int bufOffset,
                                    ChangeSet cs)
        {
            const int ENTRY_LEN = 14;

            if (ent.Segment.Relocs.Count != 0)
            {
                Debug.WriteLine("WEIRD: jump table has reloc data?");
            }

            byte[] srcData = ent.Segment.GetConstData();
            Array.Copy(srcData, 0, data, bufOffset, srcData.Length);

            // For no documented reason, jump tables start with 8 zero bytes.
            for (int i = 0; i < 8; i++)
            {
                if (data[bufOffset + i] != 0)
                {
                    Debug.WriteLine("JumpTab: missing 8-byte header");
                    return(false);
                }
            }

            TypedRangeSet newSet  = new TypedRangeSet();
            TypedRangeSet undoSet = new TypedRangeSet();

            for (int i = 8; i + 4 <= ent.Segment.Length; i += ENTRY_LEN)
            {
                //int userId = RawData.GetWord(data, bufOffset + i, 2, false);
                int fileNum = RawData.GetWord(data, bufOffset + i + 2, 2, false);

                if (fileNum == 0)
                {
                    // A zero file number indicates end of table.
                    Debug.WriteLine("JumpTab: found fileNum=0 at offset " + i + ", len=" +
                                    ent.Segment.Length);
                    break;
                }
                else if (fileNum != 1)
                {
                    // External file, ignore entry.
                    Debug.WriteLine("JumpTab: ignoring entry with FileNum=" + fileNum);
                    continue;
                }
                else if (i + ENTRY_LEN > ent.Segment.Length)
                {
                    // Make sure the rest fits.
                    Debug.WriteLine("JumpTab: overran buffer");
                    return(false);
                }

                // Note: segment might end right after FileNum, so don't try to read further
                // until we've confirmed that FileNum != 0.

                int segNum = RawData.GetWord(data, bufOffset + i + 4, 2, false);
                int segOff = RawData.GetWord(data, bufOffset + i + 6, 4, false);

                if (segNum < 0 || segNum >= mSegmentMap.Count || mSegmentMap[segNum] == null)
                {
                    Debug.WriteLine("JumpTab: invalid SegNum=" + segNum);
                    return(false);
                }
                if (data[bufOffset + i + 10] != 0x22)
                {
                    Debug.WriteLine("JumpTab: did not find expected JSL at off=" + i);
                    return(false);
                }

                int addr      = mSegmentMap[segNum].Address + segOff;
                int jmlOffset = bufOffset + i + 10;
                data[jmlOffset]     = 0x5c; // JML
                data[jmlOffset + 1] = (byte)addr;
                data[jmlOffset + 2] = (byte)(addr >> 8);
                data[jmlOffset + 3] = (byte)(addr >> 16);
                //Debug.WriteLine("JumpTab: off=" + i + " -> " +
                //    mFormatter.FormatAddress(addr, true));

                // It seems to be fairly common for jump table entries to not be referenced
                // from the program, which can leave whole dynamic segments unreferenced.  Set
                // a code start tag on the JML instruction.
                undoSet.Add(jmlOffset, (int)CodeAnalysis.AnalyzerTag.None);
                newSet.Add(jmlOffset, (int)CodeAnalysis.AnalyzerTag.Code);
            }

            UndoableChange uc = UndoableChange.CreateAnalyzerTagChange(undoSet, newSet);

            cs.Add(uc);

            return(true);
        }
Example #5
0
        /// <summary>
        /// Creates a map of file segments.  The position of each segment in the list will
        /// match the segment's position in the file, i.e. the segment in Map[5] will have
        /// SEGNUM==5.
        /// </summary>
        /// <remarks>
        /// I'm assuming that the SEGNUM in the file matches the position.  This seems to be
        /// the case everywhere.  ExpressLoad goes to some lengths to ensure this is still the
        /// case after a file is "expressed", including a remap table so that loader calls made
        /// by the application go to the right place (instead of, say, giving ~ExpressLoad a
        /// SEGNUM of 255.)
        /// </remarks>
        /// <returns>True on success.</returns>
        private bool CreateMap()
        {
            // Segments are numbered 1-N, so create a map with N+1 entries and leave first blank.
            mSegmentMap = new List <SegmentMapEntry>(mOmfFile.SegmentList.Count + 1);
            mSegmentMap.Add(null);

            // Create a bank in-use map.
            bool[] inUse = new bool[256];

            // Flag special memory as in-use.
            inUse[0x00] = inUse[0x01] = inUse[0xe0] = inUse[0xe1] = true;

            // Find segments that require specific addresses, and mark those banks as in use.
            foreach (OmfSegment omfSeg in mOmfFile.SegmentList)
            {
                if (omfSeg.Kind == OmfSegment.SegmentKind.DpStack)
                {
                    // This just allocates space in bank 0.
                    continue;
                }
                if (omfSeg.Length == 0)
                {
                    // Nothing to do here.
                    continue;
                }

                int addr;

                if (omfSeg.Org == 0)
                {
                    // The docs say that a value of zero always means relocatable, but that
                    // would mean you can't set the "absolute bank" flag to position code or
                    // data in bank 0.  I'm going to assume that's intentional, since people
                    // (a) shouldn't be doing that, and (b) can use DP/Stack instead (?).
                    //
                    // It also means that "bank relative" can't be used to set the position
                    // to zero, which is probably fine since you can do that with ALIGN=$10000.
                    continue;
                }

                addr = omfSeg.Org;
                if ((omfSeg.Attrs & OmfSegment.SegmentAttribute.AbsBank) != 0)
                {
                    // Bank is specified, rest of address is not.
                    addr &= 0x00ff0000;
                }

                // Mark the banks as being in use.  It's okay if multiple segments want the
                // same space.
                MarkBanks(addr, omfSeg.Length, inUse);
            }

            //
            // Assign segments to banks.  Note we always start at offset $0000 within a bank.
            //

            int nextBank = 0;
            int dpAddr   = 0x1000;  // somewhat arbitrary

            foreach (OmfSegment omfSeg in mOmfFile.SegmentList)
            {
                if (omfSeg.Kind == OmfSegment.SegmentKind.DpStack || omfSeg.Length == 0)
                {
                    mSegmentMap.Add(new SegmentMapEntry(omfSeg, dpAddr));
                    dpAddr += omfSeg.Length;
                    if (dpAddr > 0x00010000)
                    {
                        Debug.WriteLine("Stack/DP overflow");
                        return(false);
                    }
                    continue;
                }
                if (omfSeg.IsExpressLoad)
                {
                    // We totally ignore these.  Add a null ref as a placeholder.
                    mSegmentMap.Add(null);
                    continue;
                }

                int addr;

                // We want to put the segment at a specific offset in an arbitrary bank
                // if ORG is nonzero, the BankRel flag is set, and the AbsBank flag is clear.
                bool bankRel = omfSeg.Org != 0 &&
                               (omfSeg.Attrs & OmfSegment.SegmentAttribute.BankRel) != 0 &&
                               (omfSeg.Attrs & OmfSegment.SegmentAttribute.AbsBank) == 0;
                // We want to put the segment at an arbitrary offset in a specific bank
                // if ORG is nonzero, the BankRel flag is clear, and the AbsBank flag is set.
                bool fixedBank = omfSeg.Org != 0 &&
                                 (omfSeg.Attrs & OmfSegment.SegmentAttribute.BankRel) == 0 &&
                                 (omfSeg.Attrs & OmfSegment.SegmentAttribute.AbsBank) != 0;
                // We want to put the segment at a specific offset and bank
                // if ORG is nonzero, and BankRel and FixedBank are either both set or
                // both clear.
                bool fixedAddr = omfSeg.Org != 0 && (bankRel ^ fixedBank) == false;

                if (fixedAddr || fixedBank)
                {
                    // Specific bank requested.
                    addr = omfSeg.Org;
                    if ((omfSeg.Attrs & OmfSegment.SegmentAttribute.AbsBank) != 0)
                    {
                        // just keep the bank
                        addr &= 0x00ff0000;
                    }
                }
                else
                {
                    // Find next available bank with enough space.
                    while (true)
                    {
                        while (nextBank < 256 && inUse[nextBank])
                        {
                            nextBank++;
                        }
                        if (nextBank == 256)
                        {
                            // Should be impossible on any sane Apple IIgs Load file.
                            Debug.Assert(false);
                            return(false);
                        }
                        if (!CheckBanks(nextBank << 16, omfSeg.Length, inUse))
                        {
                            // Didn't fit in the space.
                            nextBank++;
                            continue;
                        }

                        // We only go forward, so no need to mark them.

                        break;
                    }

                    addr = nextBank << 16;
                    if (bankRel)
                    {
                        // TODO(maybe): reject if incompatible with BANKSIZE
                        addr |= omfSeg.Org & 0x0000ffff;
                    }

                    // Advance nextBank.  We do this by identifying the last address touched,
                    // then incrementing the bank number.
                    int lastAddr = addr + omfSeg.Length - 1;
                    nextBank = (lastAddr >> 16) + 1;
                    if (nextBank >= 0x0100)
                    {
                        // Overflowed the 65816 address space.
                        Debug.WriteLine("Bank exceeded $ff");
                        return(false);
                    }
                }

                // If possible, shift the address to xx/$0100.  This is useful because it means
                // we won't have to put width disambiguators on any data accesses to $00xx
                // locations.  We can't do this if the address is fixed, aligned to a 64K
                // boundary, or is too large.
                if ((mFlags & Flags.OffsetSegmentStart) != 0 && !fixedAddr && !bankRel &&
                    omfSeg.Align <= 0x0100 && omfSeg.Length <= (65536 - 256))
                {
                    if ((addr & 0x0000ffff) == 0x0000)
                    {
                        addr |= 0x0100;
                    }
                    else
                    {
                        Debug.Assert(false, "Unexpected nonzero bank address found");
                    }
                }

                SegmentMapEntry ent = new SegmentMapEntry(omfSeg, addr);
                mSegmentMap.Add(ent);
            }

            return(true);
        }