diff options
-rw-r--r-- | src/changes/changes.xml | 5 | ||||
-rw-r--r-- | src/main/java/com/healthmarketscience/jackcess/impl/UsageMap.java | 165 |
2 files changed, 92 insertions, 78 deletions
diff --git a/src/changes/changes.xml b/src/changes/changes.xml index 87cb653..f65e501 100644 --- a/src/changes/changes.xml +++ b/src/changes/changes.xml @@ -4,6 +4,11 @@ <author email="javajedi@users.sf.net">Tim McCune</author> </properties> <body> + <release version="4.0.3" date="TBD"> + <action dev="jahlborn" type="fix" system="SourceForge2" issue="156"> + Fix edge case which can cause table to be considered corrupt. + </action> + </release> <release version="4.0.2" date="2022-08-26"> <action dev="jahlborn" type="update"> Add Table methods to get the creation and last modified dates. diff --git a/src/main/java/com/healthmarketscience/jackcess/impl/UsageMap.java b/src/main/java/com/healthmarketscience/jackcess/impl/UsageMap.java index c67bcba..6928cfa 100644 --- a/src/main/java/com/healthmarketscience/jackcess/impl/UsageMap.java +++ b/src/main/java/com/healthmarketscience/jackcess/impl/UsageMap.java @@ -36,7 +36,7 @@ public class UsageMap /** bit index value for an invalid page number */ private static final int INVALID_BIT_INDEX = -1; - + /** owning database */ private final DatabaseImpl _database; /** Page number of the map table declaration */ @@ -84,7 +84,7 @@ public class UsageMap public DatabaseImpl getDatabase() { return _database; } - + public JetFormat getFormat() { return getDatabase().getFormat(); } @@ -106,7 +106,7 @@ public class UsageMap int umapPageNum = ByteUtil.get3ByteInt(buf); return read(database, umapPageNum, umapRowNum, false); } - + /** * @param database database that contains this usage map * @param pageNum Page number that this usage map is contained in @@ -130,7 +130,7 @@ public class UsageMap pageChannel.readPage(tableBuffer, pageNum); short rowStart = TableImpl.findRowStart(tableBuffer, rowNum, format); int rowEnd = TableImpl.findRowEnd(tableBuffer, rowNum, format); - tableBuffer.limit(rowEnd); + tableBuffer.limit(rowEnd); byte mapType = tableBuffer.get(rowStart); UsageMap rtn = new UsageMap(database, tableBuffer, pageNum, rowStart); rtn.initHandler(mapType, isGlobal); @@ -141,16 +141,16 @@ public class UsageMap throws IOException { if (mapType == MAP_TYPE_INLINE) { - _handler = (isGlobal ? new GlobalInlineHandler() : + _handler = (isGlobal ? new GlobalInlineHandler() : new InlineHandler()); } else if (mapType == MAP_TYPE_REFERENCE) { - _handler = (isGlobal ? new GlobalReferenceHandler() : + _handler = (isGlobal ? new GlobalReferenceHandler() : new ReferenceHandler()); } else { throw new IOException(MSG_PREFIX_UNRECOGNIZED_MAP + mapType); } } - + public PageCursor cursor() { return new PageCursor(); } @@ -158,7 +158,7 @@ public class UsageMap public int getPageCount() { return _pageNumbers.cardinality(); } - + protected short getRowStart() { return _rowStart; } @@ -170,27 +170,27 @@ public class UsageMap protected void setStartOffset(int startOffset) { _startOffset = startOffset; } - + protected int getStartOffset() { return _startOffset; } - + protected ByteBuffer getTableBuffer() { return _tableBuffer; } - + protected int getTablePageNumber() { return _tablePageNum; } - + protected int getStartPage() { return _startPage; } - + protected int getEndPage() { return _endPage; } - + protected BitSet getPageNumbers() { return _pageNumbers; } @@ -206,7 +206,7 @@ public class UsageMap } protected int getFirstPageNumber() { - return bitIndexToPageNumber(getNextBitIndex(-1), + return bitIndexToPageNumber(getNextBitIndex(-1), RowIdImpl.LAST_PAGE_NUMBER); } @@ -214,12 +214,12 @@ public class UsageMap return bitIndexToPageNumber( getNextBitIndex(pageNumberToBitIndex(curPage)), RowIdImpl.LAST_PAGE_NUMBER); - } - + } + protected int getNextBitIndex(int curIndex) { return _pageNumbers.nextSetBit(curIndex + 1); - } - + } + protected int getLastPageNumber() { return bitIndexToPageNumber(getPrevBitIndex(_pageNumbers.length()), RowIdImpl.FIRST_PAGE_NUMBER); @@ -229,16 +229,16 @@ public class UsageMap return bitIndexToPageNumber( getPrevBitIndex(pageNumberToBitIndex(curPage)), RowIdImpl.FIRST_PAGE_NUMBER); - } - + } + protected int getPrevBitIndex(int curIndex) { --curIndex; while((curIndex >= 0) && !_pageNumbers.get(curIndex)) { --curIndex; } return curIndex; - } - + } + protected int bitIndexToPageNumber(int bitIndex, int invalidPageNumber) { return((bitIndex >= 0) ? (_startPage + bitIndex) : invalidPageNumber); @@ -256,20 +256,20 @@ public class UsageMap _startPage = 0; _endPage = 0; ++_modCount; - + // clear out the table data (everything except map type) int tableStart = getRowStart() + 1; int tableEnd = getRowEnd(); ByteUtil.clearRange(_tableBuffer, tableStart, tableEnd); } - + protected void writeTable() throws IOException { // note, we only want to write the row data with which we are working getPageChannel().writePage(_tableBuffer, _tablePageNum, _rowStart); } - + /** * Read in the page numbers in this inline map */ @@ -305,7 +305,7 @@ public class UsageMap public boolean containsPageNumber(int pageNumber) { return _handler.containsPageNumber(pageNumber); } - + /** * Add a page number to this usage map */ @@ -313,23 +313,23 @@ public class UsageMap ++_modCount; _handler.addOrRemovePageNumber(pageNumber, true, false); } - + /** * Remove a page number from this usage map */ - public void removePageNumber(int pageNumber) - throws IOException + public void removePageNumber(int pageNumber) + throws IOException { removePageNumber(pageNumber, true); } - - private void removePageNumber(int pageNumber, boolean force) - throws IOException + + private void removePageNumber(int pageNumber, boolean force) + throws IOException { ++_modCount; _handler.addOrRemovePageNumber(pageNumber, false, force); } - + protected void updateMap(int absolutePageNumber, int bufferRelativePageNumber, ByteBuffer buffer, boolean add, boolean force) @@ -349,7 +349,7 @@ public class UsageMap " usage map, expected range " + _startPage + " to " + _endPage); } - + //Apply the bitmask if (add) { b |= bitmask; @@ -373,13 +373,13 @@ public class UsageMap // clear out the main table (inline usage map data and start page) clearTableAndPages(); - + // set the new map type _tableBuffer.put(getRowStart(), MAP_TYPE_REFERENCE); // write the new table data writeTable(); - + // set new handler _handler = new ReferenceHandler(); @@ -444,7 +444,12 @@ public class UsageMap ranges.add(String.valueOf(rangeStart)); } } - + + private static int toValidStartPage(int startPage) { + // start page must be a multiple of 8 + return ((startPage / 8) * 8); + } + private abstract class Handler { protected Handler() { @@ -454,7 +459,7 @@ public class UsageMap return(isPageWithinRange(pageNumber) && getPageNumbers().get(pageNumberToBitIndex(pageNumber))); } - + /** * @param pageNumber Page number to add or remove from this map * @param add True to add it, false to remove it @@ -476,7 +481,7 @@ public class UsageMap private class InlineHandler extends Handler { private final int _maxInlinePages; - + protected InlineHandler() throws IOException { _maxInlinePages = (getInlineDataEnd() - getInlineDataStart()) * 8; @@ -496,15 +501,15 @@ public class UsageMap protected final int getInlineDataEnd() { return getRowEnd(); } - + /** * Sets the page range for an inline usage map starting from the given * page. */ private void setInlinePageRange(int startPage) { setPageRange(startPage, startPage + getMaxInlinePages()); - } - + } + @Override public void addOrRemovePageNumber(int pageNumber, boolean add, boolean force) @@ -518,7 +523,7 @@ public class UsageMap force); // Write the updated map back to disk writeTable(); - + } else { // uh-oh, we've split our britches. what now? @@ -531,7 +536,7 @@ public class UsageMap throws IOException { // determine what our status is before taking action - + if(add) { int firstPage = getFirstPageNumber(); @@ -547,11 +552,14 @@ public class UsageMap } else { firstPage = pageNumber; } + + firstPage = toValidStartPage(firstPage); + if((lastPage - firstPage + 1) < getMaxInlinePages()) { // we can still fit within an inline map moveToNewStartPage(firstPage, pageNumber); - + } else { // not going to happen, need to promote the usage map to a // reference map @@ -635,7 +643,7 @@ public class UsageMap // since we assuming out-of-range bits are "on". Note, we are leaving // small holes in the database here (leaving behind some free pages), // but it's not the end of the world. - + if(!add) { int firstPage = getFirstPageNumber(); @@ -666,10 +674,11 @@ public class UsageMap throws IOException { int oldEndPage = getEndPage(); - int newStartPage = - ((firstPage <= PageChannel.INVALID_PAGE_NUMBER) ? newPageNumber : - // just shift a little and discard any initial unused pages. - (newPageNumber - (getMaxInlinePages() / 2))); + int newStartPage = + toValidStartPage( + ((firstPage <= PageChannel.INVALID_PAGE_NUMBER) ? newPageNumber : + // just shift a little and discard any initial unused pages. + (newPageNumber - (getMaxInlinePages() / 2)))); // move the current data moveToNewStartPage(newStartPage, PageChannel.INVALID_PAGE_NUMBER); @@ -708,19 +717,19 @@ public class UsageMap */ private class ReferenceHandler extends Handler { - /** Buffer that contains the current reference map page */ + /** Buffer that contains the current reference map page */ private final TempPageHolder _mapPageHolder = TempPageHolder.newHolder(TempBufferHolder.Type.SOFT); private final int _maxPagesPerUsageMapPage; - + private ReferenceHandler() throws IOException { - _maxPagesPerUsageMapPage = ((getFormat().PAGE_SIZE - + _maxPagesPerUsageMapPage = ((getFormat().PAGE_SIZE - getFormat().OFFSET_USAGE_MAP_PAGE_DATA) * 8); int numUsagePages = (getRowEnd() - getRowStart() - 1) / 4; setStartOffset(getFormat().OFFSET_USAGE_MAP_PAGE_DATA); setPageRange(0, (numUsagePages * _maxPagesPerUsageMapPage)); - + // there is no "start page" for a reference usage map, so we get an // extra page reference on top of the number of page references that fit // in the table @@ -745,7 +754,7 @@ public class UsageMap protected final int getMaxPagesPerUsagePage() { return _maxPagesPerUsageMapPage; } - + @Override public void addOrRemovePageNumber(int pageNumber, boolean add, boolean force) @@ -774,11 +783,11 @@ public class UsageMap mapPageBuffer, add, force); getPageChannel().writePage(mapPageBuffer, mapPageNum); } - + /** * Create a new usage map page and update the map declaration with a * pointer to it. - * @param pageIndex Index of the page reference within the map declaration + * @param pageIndex Index of the page reference within the map declaration */ private ByteBuffer createNewUsageMapPage(int pageIndex) throws IOException { @@ -789,14 +798,14 @@ public class UsageMap writeTable(); return mapPageBuffer; } - + private int calculateMapPagePointerOffset(int pageIndex) { return getRowStart() + getFormat().OFFSET_REFERENCE_MAP_PAGE_NUMBERS + (pageIndex * 4); } - protected ByteBuffer allocateNewUsageMapPage(int pageIndex) - throws IOException + protected ByteBuffer allocateNewUsageMapPage(int pageIndex) + throws IOException { ByteBuffer mapPageBuffer = _mapPageHolder.setNewPage(getPageChannel()); mapPageBuffer.put(PageTypes.USAGE_MAP); @@ -850,7 +859,7 @@ public class UsageMap super.addOrRemovePageNumber(pageNumber, add, force); while(_pendingPage != null) { - + // while updating our usage map, we needed to allocate a new page (and // thus mark a new page as used). we delayed that marking so that we // didn't get into an infinite loop. now that we completed the @@ -862,12 +871,12 @@ public class UsageMap _pendingPage = null; super.addOrRemovePageNumber(removedPageNumber, false, true); - } - } + } + } @Override protected ByteBuffer allocateNewUsageMapPage(int pageIndex) - throws IOException + throws IOException { try { // keep track of the fact that we are actively allocating a page for our @@ -882,7 +891,7 @@ public class UsageMap // load this map, which is fine because we should only add pages which // represent the size of the database currently in use). int dataStart = getFormat().OFFSET_USAGE_MAP_PAGE_DATA; - ByteUtil.fillRange(mapPageBuffer, dataStart, + ByteUtil.fillRange(mapPageBuffer, dataStart, getFormat().PAGE_SIZE - dataStart); int maxPagesPerUmapPage = getMaxPagesPerUsagePage(); @@ -924,7 +933,7 @@ public class UsageMap public UsageMap getUsageMap() { return UsageMap.this; } - + /** * Returns the DirHandler for the given direction */ @@ -938,7 +947,7 @@ public class UsageMap */ public boolean isUpToDate() { return(UsageMap.this._modCount == _lastModCount); - } + } /** * @return valid page number if there was another page to read, @@ -972,7 +981,7 @@ public class UsageMap } checkForModification(); - + _prevPageNumber = _curPageNumber; _curPageNumber = handler.getAnotherPageNumber(_curPageNumber); return _curPageNumber; @@ -1019,7 +1028,7 @@ public class UsageMap { restorePosition(curPageNumber, _curPageNumber); } - + /** * Restores a current and previous position for the cursor. */ @@ -1055,14 +1064,14 @@ public class UsageMap } return pageNumber; } - + @Override public String toString() { return getClass().getSimpleName() + " CurPosition " + _curPageNumber + ", PrevPosition " + _prevPageNumber; } - - + + /** * Handles moving the cursor in a given direction. Separates cursor * logic from value storage. @@ -1072,7 +1081,7 @@ public class UsageMap public abstract int getBeginningPageNumber(); public abstract int getEndPageNumber(); } - + /** * Handles moving the cursor forward. */ @@ -1093,7 +1102,7 @@ public class UsageMap return RowIdImpl.LAST_PAGE_NUMBER; } } - + /** * Handles moving the cursor backward. */ @@ -1114,7 +1123,7 @@ public class UsageMap return RowIdImpl.FIRST_PAGE_NUMBER; } } - - } - + + } + } |