| Index: third_party/android_prediction/suggest/policyimpl/dictionary/utils/sparse_table.cpp
|
| diff --git a/third_party/android_prediction/suggest/policyimpl/dictionary/utils/sparse_table.cpp b/third_party/android_prediction/suggest/policyimpl/dictionary/utils/sparse_table.cpp
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..c97e3ff7f8c956972576fa080548b575f23999f5
|
| --- /dev/null
|
| +++ b/third_party/android_prediction/suggest/policyimpl/dictionary/utils/sparse_table.cpp
|
| @@ -0,0 +1,101 @@
|
| +/*
|
| + * Copyright (C) 2013, The Android Open Source Project
|
| + *
|
| + * Licensed under the Apache License, Version 2.0 (the "License");
|
| + * you may not use this file except in compliance with the License.
|
| + * You may obtain a copy of the License at
|
| + *
|
| + * http://www.apache.org/licenses/LICENSE-2.0
|
| + *
|
| + * Unless required by applicable law or agreed to in writing, software
|
| + * distributed under the License is distributed on an "AS IS" BASIS,
|
| + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
| + * See the License for the specific language governing permissions and
|
| + * limitations under the License.
|
| + */
|
| +
|
| +#include "third_party/android_prediction/suggest/policyimpl/dictionary/utils/sparse_table.h"
|
| +
|
| +namespace latinime {
|
| +
|
| +const int SparseTable::NOT_EXIST = -1;
|
| +const int SparseTable::INDEX_SIZE = 4;
|
| +
|
| +bool SparseTable::contains(const int id) const {
|
| + const int readingPos = getPosInIndexTable(id);
|
| + if (id < 0 || mIndexTableBuffer->getTailPosition() <= readingPos) {
|
| + return false;
|
| + }
|
| + const int index = mIndexTableBuffer->readUint(INDEX_SIZE, readingPos);
|
| + return index != NOT_EXIST;
|
| +}
|
| +
|
| +uint32_t SparseTable::get(const int id) const {
|
| + const int indexTableReadingPos = getPosInIndexTable(id);
|
| + const int index = mIndexTableBuffer->readUint(INDEX_SIZE, indexTableReadingPos);
|
| + const int contentTableReadingPos = getPosInContentTable(id, index);
|
| + if (contentTableReadingPos < 0
|
| + || contentTableReadingPos >= mContentTableBuffer->getTailPosition()) {
|
| + AKLOGE("contentTableReadingPos(%d) is invalid. id: %d, index: %d",
|
| + contentTableReadingPos, id, index);
|
| + return NOT_A_DICT_POS;
|
| + }
|
| + const int contentValue = mContentTableBuffer->readUint(mDataSize, contentTableReadingPos);
|
| + return contentValue == NOT_EXIST ? NOT_A_DICT_POS : contentValue;
|
| +}
|
| +
|
| +bool SparseTable::set(const int id, const uint32_t value) {
|
| + const int posInIndexTable = getPosInIndexTable(id);
|
| + // Extends the index table if needed.
|
| + int tailPos = mIndexTableBuffer->getTailPosition();
|
| + while (tailPos <= posInIndexTable) {
|
| + if (!mIndexTableBuffer->writeUintAndAdvancePosition(NOT_EXIST, INDEX_SIZE, &tailPos)) {
|
| + AKLOGE("cannot extend index table. tailPos: %d to: %d", tailPos, posInIndexTable);
|
| + return false;
|
| + }
|
| + }
|
| + if (contains(id)) {
|
| + // The entry is already in the content table.
|
| + const int index = mIndexTableBuffer->readUint(INDEX_SIZE, posInIndexTable);
|
| + if (!mContentTableBuffer->writeUint(value, mDataSize, getPosInContentTable(id, index))) {
|
| + AKLOGE("cannot update value %d. pos: %d, tailPos: %d, mDataSize: %d", value,
|
| + getPosInContentTable(id, index), mContentTableBuffer->getTailPosition(),
|
| + mDataSize);
|
| + return false;
|
| + }
|
| + return true;
|
| + }
|
| + // The entry is not in the content table.
|
| + // Create new entry in the content table.
|
| + const int index = getIndexFromContentTablePos(mContentTableBuffer->getTailPosition());
|
| + if (!mIndexTableBuffer->writeUint(index, INDEX_SIZE, posInIndexTable)) {
|
| + AKLOGE("cannot write index %d. pos %d", index, posInIndexTable);
|
| + return false;
|
| + }
|
| + // Write a new block that containing the entry to be set.
|
| + int writingPos = getPosInContentTable(0 /* id */, index);
|
| + for (int i = 0; i < mBlockSize; ++i) {
|
| + if (!mContentTableBuffer->writeUintAndAdvancePosition(NOT_EXIST, mDataSize,
|
| + &writingPos)) {
|
| + AKLOGE("cannot write content table to extend. writingPos: %d, tailPos: %d, "
|
| + "mDataSize: %d", writingPos, mContentTableBuffer->getTailPosition(), mDataSize);
|
| + return false;
|
| + }
|
| + }
|
| + return mContentTableBuffer->writeUint(value, mDataSize, getPosInContentTable(id, index));
|
| +}
|
| +
|
| +int SparseTable::getIndexFromContentTablePos(const int contentTablePos) const {
|
| + return contentTablePos / mDataSize / mBlockSize;
|
| +}
|
| +
|
| +int SparseTable::getPosInIndexTable(const int id) const {
|
| + return (id / mBlockSize) * INDEX_SIZE;
|
| +}
|
| +
|
| +int SparseTable::getPosInContentTable(const int id, const int index) const {
|
| + const int offset = id % mBlockSize;
|
| + return (index * mBlockSize + offset) * mDataSize;
|
| +}
|
| +
|
| +} // namespace latinime
|
|
|