提交 d5e0bdb4 authored 作者: Thomas Mueller's avatar Thomas Mueller

Experimental feature to support very large transactions.

上级 df2e2a0d
...@@ -294,6 +294,11 @@ public class Constants { ...@@ -294,6 +294,11 @@ public class Constants {
* The delay that is to be used if throttle has been enabled. * The delay that is to be used if throttle has been enabled.
*/ */
public static final int THROTTLE_DELAY = 50; public static final int THROTTLE_DELAY = 50;
/**
* The maximum size of an undo log block.
*/
public static final int UNDO_BLOCK_SIZE = 1024 * 1024;
/** /**
* The database URL format in simplified Backus-Naur form. * The database URL format in simplified Backus-Naur form.
......
...@@ -76,23 +76,25 @@ public class UndoLog { ...@@ -76,23 +76,25 @@ public class UndoLog {
public UndoLogRecord getLast() { public UndoLogRecord getLast() {
int i = records.size() - 1; int i = records.size() - 1;
if (SysProperties.LARGE_TRANSACTIONS) { if (SysProperties.LARGE_TRANSACTIONS) {
int test; if (i < 0 && storedEntries > 0) {
if (i <= 0 && storedEntries > 0) {
int last = storedEntriesPos.size() - 1; int last = storedEntriesPos.size() - 1;
long pos = storedEntriesPos.get(last); long pos = storedEntriesPos.get(last);
storedEntriesPos.remove(last); storedEntriesPos.remove(last);
long end = file.length(); long end = file.length();
int bufferLength = (int) (end - pos); int bufferLength = (int) (end - pos);
Data buff = Data.create(database, bufferLength); Data buff = Data.create(database, bufferLength);
seek(pos); file.seek(pos);
file.readFully(buff.getBytes(), 0, bufferLength); file.readFully(buff.getBytes(), 0, bufferLength);
while (buff.length() < bufferLength) { while (buff.length() < bufferLength) {
UndoLogRecord e = UndoLogRecord.loadFromBuffer(buff, this); UndoLogRecord e = UndoLogRecord.loadFromBuffer(buff, this);
records.add(e); records.add(e);
memoryUndo++; memoryUndo++;
} }
storedEntries -= records.size();
file.setLength(pos); file.setLength(pos);
file.seek(pos);
} }
i = records.size() - 1;
} }
UndoLogRecord entry = records.get(i); UndoLogRecord entry = records.get(i);
if (entry.isStored()) { if (entry.isStored()) {
...@@ -149,12 +151,9 @@ public class UndoLog { ...@@ -149,12 +151,9 @@ public class UndoLog {
*/ */
public void add(UndoLogRecord entry) { public void add(UndoLogRecord entry) {
records.add(entry); records.add(entry);
if (!entry.isStored()) { if (SysProperties.LARGE_TRANSACTIONS) {
memoryUndo++; memoryUndo++;
} if (memoryUndo > database.getMaxMemoryUndo() && database.isPersistent() && !database.isMultiVersion()) {
if (memoryUndo > database.getMaxMemoryUndo() && database.isPersistent() && !database.isMultiVersion()) {
if (SysProperties.LARGE_TRANSACTIONS) {
int todo;
if (file == null) { if (file == null) {
String fileName = database.createTempFile(); String fileName = database.createTempFile();
file = database.openFile(fileName, "rw", false); file = database.openFile(fileName, "rw", false);
...@@ -165,28 +164,38 @@ public class UndoLog { ...@@ -165,28 +164,38 @@ public class UndoLog {
UndoLogRecord r = records.get(i); UndoLogRecord r = records.get(i);
buff.checkCapacity(SysProperties.PAGE_SIZE); buff.checkCapacity(SysProperties.PAGE_SIZE);
r.append(buff, this); r.append(buff, this);
if (i == records.size() - 1 || buff.length() > Constants.UNDO_BLOCK_SIZE) {
storedEntriesPos.add(file.getFilePointer());
file.write(buff.getBytes(), 0, buff.length());
buff.reset();
}
} }
storedEntriesPos.add(file.getFilePointer()); storedEntries += records.size();
file.write(buff.getBytes(), 0, buff.length());
memoryUndo = 0; memoryUndo = 0;
records.clear(); records.clear();
file.autoDelete(); file.autoDelete();
return; return;
} }
if (file == null) { } else {
String fileName = database.createTempFile(); if (!entry.isStored()) {
file = database.openFile(fileName, "rw", false); memoryUndo++;
file.seek(FileStore.HEADER_LENGTH); }
rowBuff = Data.create(database, SysProperties.PAGE_SIZE); if (memoryUndo > database.getMaxMemoryUndo() && database.isPersistent() && !database.isMultiVersion()) {
Data buff = rowBuff; if (file == null) {
for (int i = 0; i < records.size(); i++) { String fileName = database.createTempFile();
UndoLogRecord r = records.get(i); file = database.openFile(fileName, "rw", false);
saveIfPossible(r, buff); file.seek(FileStore.HEADER_LENGTH);
rowBuff = Data.create(database, SysProperties.PAGE_SIZE);
Data buff = rowBuff;
for (int i = 0; i < records.size(); i++) {
UndoLogRecord r = records.get(i);
saveIfPossible(r, buff);
}
} else {
saveIfPossible(entry, rowBuff);
} }
} else { file.autoDelete();
saveIfPossible(entry, rowBuff);
} }
file.autoDelete();
} }
} }
...@@ -202,9 +211,9 @@ public class UndoLog { ...@@ -202,9 +211,9 @@ public class UndoLog {
if (tables == null) { if (tables == null) {
tables = New.hashMap(); tables = New.hashMap();
} }
if (tables.get(id) == null) { // need to overwrite the old entry, because the old object
tables.put(id, table); // might be deleted in the meantime
} tables.put(id, table);
return id; return id;
} }
......
...@@ -156,7 +156,7 @@ public class UndoLogRecord { ...@@ -156,7 +156,7 @@ public class UndoLogRecord {
buff.writeValue(v); buff.writeValue(v);
} }
buff.fillAligned(); buff.fillAligned();
buff.setInt(p, buff.length() / Constants.FILE_BLOCK_SIZE); buff.setInt(p, (buff.length() - p) / Constants.FILE_BLOCK_SIZE);
} }
/** /**
...@@ -201,16 +201,17 @@ public class UndoLogRecord { ...@@ -201,16 +201,17 @@ public class UndoLogRecord {
if (len - min > 0) { if (len - min > 0) {
file.readFully(buff.getBytes(), min, len - min); file.readFully(buff.getBytes(), min, len - min);
} }
int oldOp = operation;
load(buff, log); load(buff, log);
}
private void load(Data buff, UndoLog log) {
int op = buff.readInt();
if (SysProperties.CHECK) { if (SysProperties.CHECK) {
if (operation != op) { if (operation != oldOp) {
DbException.throwInternalError("operation=" + operation + " op=" + op); DbException.throwInternalError("operation=" + operation + " op=" + oldOp);
} }
} }
}
private void load(Data buff, UndoLog log) {
operation = (short) buff.readInt();
boolean deleted = buff.readByte() == 1; boolean deleted = buff.readByte() == 1;
if (SysProperties.LARGE_TRANSACTIONS) { if (SysProperties.LARGE_TRANSACTIONS) {
table = log.getTable(buff.readInt()); table = log.getTable(buff.readInt());
......
Markdown 格式
0%
您添加了 0 到此讨论。请谨慎行事。
请先完成此评论的编辑!
注册 或者 后发表评论