提交 f1b35267 authored 作者: Andrei Tokar's avatar Andrei Tokar

MVTable.lastModificationId race condition

上级 e9b9edb6
...@@ -344,6 +344,10 @@ public abstract class Query extends Prepared { ...@@ -344,6 +344,10 @@ public abstract class Query extends Prepared {
if (!cacheableChecked) { if (!cacheableChecked) {
long max = getMaxDataModificationId(); long max = getMaxDataModificationId();
noCache = max == Long.MAX_VALUE; noCache = max == Long.MAX_VALUE;
if (!isEverything(ExpressionVisitor.DETERMINISTIC_VISITOR) ||
!isEverything(ExpressionVisitor.INDEPENDENT_VISITOR)) {
noCache = true;
}
cacheableChecked = true; cacheableChecked = true;
} }
if (noCache) { if (noCache) {
...@@ -356,18 +360,10 @@ public abstract class Query extends Prepared { ...@@ -356,18 +360,10 @@ public abstract class Query extends Prepared {
return false; return false;
} }
} }
if (!isEverything(ExpressionVisitor.DETERMINISTIC_VISITOR) || return getMaxDataModificationId() <= lastEval;
!isEverything(ExpressionVisitor.INDEPENDENT_VISITOR)) {
return false;
}
if (db.getModificationDataId() > lastEval &&
getMaxDataModificationId() > lastEval) {
return false;
}
return true;
} }
public final Value[] getParameterValues() { private Value[] getParameterValues() {
ArrayList<Parameter> list = getParameters(); ArrayList<Parameter> list = getParameters();
if (list == null) { if (list == null) {
return new Value[0]; return new Value[0];
......
...@@ -14,6 +14,7 @@ import java.util.Set; ...@@ -14,6 +14,7 @@ import java.util.Set;
import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicInteger; import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.AtomicLong;
import org.h2.api.DatabaseEventListener; import org.h2.api.DatabaseEventListener;
import org.h2.api.ErrorCode; import org.h2.api.ErrorCode;
...@@ -105,7 +106,7 @@ public class MVTable extends TableBase { ...@@ -105,7 +106,7 @@ public class MVTable extends TableBase {
private MVPrimaryIndex primaryIndex; private MVPrimaryIndex primaryIndex;
private final ArrayList<Index> indexes = Utils.newSmallArrayList(); private final ArrayList<Index> indexes = Utils.newSmallArrayList();
private volatile long lastModificationId; private final AtomicLong lastModificationId = new AtomicLong();
private volatile Session lockExclusiveSession; private volatile Session lockExclusiveSession;
// using a ConcurrentHashMap as a set // using a ConcurrentHashMap as a set
...@@ -661,7 +662,7 @@ public class MVTable extends TableBase { ...@@ -661,7 +662,7 @@ public class MVTable extends TableBase {
@Override @Override
public void removeRow(Session session, Row row) { public void removeRow(Session session, Row row) {
lastModificationId = database.getNextModificationDataId(); syncLastModificationIdWithDatabase();
Transaction t = session.getTransaction(); Transaction t = session.getTransaction();
long savepoint = t.setSavepoint(); long savepoint = t.setSavepoint();
try { try {
...@@ -682,7 +683,7 @@ public class MVTable extends TableBase { ...@@ -682,7 +683,7 @@ public class MVTable extends TableBase {
@Override @Override
public void truncate(Session session) { public void truncate(Session session) {
lastModificationId = database.getNextModificationDataId(); syncLastModificationIdWithDatabase();
for (int i = indexes.size() - 1; i >= 0; i--) { for (int i = indexes.size() - 1; i >= 0; i--) {
Index index = indexes.get(i); Index index = indexes.get(i);
index.truncate(session); index.truncate(session);
...@@ -694,7 +695,7 @@ public class MVTable extends TableBase { ...@@ -694,7 +695,7 @@ public class MVTable extends TableBase {
@Override @Override
public void addRow(Session session, Row row) { public void addRow(Session session, Row row) {
lastModificationId = database.getNextModificationDataId(); syncLastModificationIdWithDatabase();
Transaction t = session.getTransaction(); Transaction t = session.getTransaction();
long savepoint = t.setSavepoint(); long savepoint = t.setSavepoint();
try { try {
...@@ -715,7 +716,7 @@ public class MVTable extends TableBase { ...@@ -715,7 +716,7 @@ public class MVTable extends TableBase {
@Override @Override
public void updateRow(Session session, Row oldRow, Row newRow) { public void updateRow(Session session, Row oldRow, Row newRow) {
newRow.setKey(oldRow.getKey()); newRow.setKey(oldRow.getKey());
lastModificationId = database.getNextModificationDataId(); syncLastModificationIdWithDatabase();
Transaction t = session.getTransaction(); Transaction t = session.getTransaction();
long savepoint = t.setSavepoint(); long savepoint = t.setSavepoint();
try { try {
...@@ -782,7 +783,7 @@ public class MVTable extends TableBase { ...@@ -782,7 +783,7 @@ public class MVTable extends TableBase {
@Override @Override
public long getMaxDataModificationId() { public long getMaxDataModificationId() {
return lastModificationId; return lastModificationId.get();
} }
public boolean getContainsLargeObject() { public boolean getContainsLargeObject() {
...@@ -895,8 +896,23 @@ public class MVTable extends TableBase { ...@@ -895,8 +896,23 @@ public class MVTable extends TableBase {
*/ */
public void commit() { public void commit() {
if (database != null) { if (database != null) {
lastModificationId = database.getNextModificationDataId(); syncLastModificationIdWithDatabase();
} }
}
// Field lastModificationId can not be just a volatile, because window of opportunity
// between reading database's modification id and storing this value in the field
// could be exploited by another thread.
// Second thread may do the same with possibly bigger (already advanced) modification id,
// and when first thread finally updates the field, it will result in lastModificationId jumping back.
// This is, of course, unacceptable.
private void syncLastModificationIdWithDatabase() {
long nextModificationDataId = database.getNextModificationDataId();
long currentId;
do {
currentId = lastModificationId.get();
} while (nextModificationDataId > currentId &&
!lastModificationId.compareAndSet(currentId, nextModificationDataId));
} }
/** /**
......
Markdown 格式
0%
您添加了 0 到此讨论。请谨慎行事。
请先完成此评论的编辑!
注册 或者 后发表评论