Home | History | Annotate | Download | only in utils
      1 /*
      2  * Copyright (C) 2013, The Android Open Source Project
      3  *
      4  * Licensed under the Apache License, Version 2.0 (the "License");
      5  * you may not use this file except in compliance with the License.
      6  * You may obtain a copy of the License at
      7  *
      8  *     http://www.apache.org/licenses/LICENSE-2.0
      9  *
     10  * Unless required by applicable law or agreed to in writing, software
     11  * distributed under the License is distributed on an "AS IS" BASIS,
     12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     13  * See the License for the specific language governing permissions and
     14  * limitations under the License.
     15  */
     16 
     17 #include "dictionary/utils/sparse_table.h"
     18 
     19 namespace latinime {
     20 
     21 const int SparseTable::NOT_EXIST = -1;
     22 const int SparseTable::INDEX_SIZE = 4;
     23 
     24 bool SparseTable::contains(const int id) const {
     25     const int readingPos = getPosInIndexTable(id);
     26     if (id < 0 || mIndexTableBuffer->getTailPosition() <= readingPos) {
     27         return false;
     28     }
     29     const int index = mIndexTableBuffer->readUint(INDEX_SIZE, readingPos);
     30     return index != NOT_EXIST;
     31 }
     32 
     33 uint32_t SparseTable::get(const int id) const {
     34     const int indexTableReadingPos = getPosInIndexTable(id);
     35     const int index = mIndexTableBuffer->readUint(INDEX_SIZE, indexTableReadingPos);
     36     const int contentTableReadingPos = getPosInContentTable(id, index);
     37     if (contentTableReadingPos < 0
     38             || contentTableReadingPos >= mContentTableBuffer->getTailPosition()) {
     39         AKLOGE("contentTableReadingPos(%d) is invalid. id: %d, index: %d",
     40                 contentTableReadingPos, id, index);
     41         return NOT_A_DICT_POS;
     42     }
     43     const int contentValue = mContentTableBuffer->readUint(mDataSize, contentTableReadingPos);
     44     return contentValue == NOT_EXIST ? NOT_A_DICT_POS : contentValue;
     45 }
     46 
     47 bool SparseTable::set(const int id, const uint32_t value) {
     48     const int posInIndexTable = getPosInIndexTable(id);
     49     // Extends the index table if needed.
     50     int tailPos = mIndexTableBuffer->getTailPosition();
     51     while (tailPos <= posInIndexTable) {
     52         if (!mIndexTableBuffer->writeUintAndAdvancePosition(NOT_EXIST, INDEX_SIZE, &tailPos)) {
     53             AKLOGE("cannot extend index table. tailPos: %d to: %d", tailPos, posInIndexTable);
     54             return false;
     55         }
     56     }
     57     if (contains(id)) {
     58         // The entry is already in the content table.
     59         const int index = mIndexTableBuffer->readUint(INDEX_SIZE, posInIndexTable);
     60         if (!mContentTableBuffer->writeUint(value, mDataSize, getPosInContentTable(id, index))) {
     61             AKLOGE("cannot update value %d. pos: %d, tailPos: %d, mDataSize: %d", value,
     62                     getPosInContentTable(id, index), mContentTableBuffer->getTailPosition(),
     63                     mDataSize);
     64             return false;
     65         }
     66         return true;
     67     }
     68     // The entry is not in the content table.
     69     // Create new entry in the content table.
     70     const int index = getIndexFromContentTablePos(mContentTableBuffer->getTailPosition());
     71     if (!mIndexTableBuffer->writeUint(index, INDEX_SIZE, posInIndexTable)) {
     72         AKLOGE("cannot write index %d. pos %d", index, posInIndexTable);
     73         return false;
     74     }
     75     // Write a new block that containing the entry to be set.
     76     int writingPos = getPosInContentTable(0 /* id */, index);
     77     for (int i = 0; i < mBlockSize; ++i) {
     78         if (!mContentTableBuffer->writeUintAndAdvancePosition(NOT_EXIST, mDataSize,
     79                 &writingPos)) {
     80             AKLOGE("cannot write content table to extend. writingPos: %d, tailPos: %d, "
     81                     "mDataSize: %d", writingPos, mContentTableBuffer->getTailPosition(), mDataSize);
     82             return false;
     83         }
     84     }
     85     return mContentTableBuffer->writeUint(value, mDataSize, getPosInContentTable(id, index));
     86 }
     87 
     88 int SparseTable::getIndexFromContentTablePos(const int contentTablePos) const {
     89     return contentTablePos / mDataSize / mBlockSize;
     90 }
     91 
     92 int SparseTable::getPosInIndexTable(const int id) const {
     93     return (id / mBlockSize) * INDEX_SIZE;
     94 }
     95 
     96 int SparseTable::getPosInContentTable(const int id, const int index) const {
     97     const int offset = id % mBlockSize;
     98     return (index * mBlockSize + offset) * mDataSize;
     99 }
    100 
    101 } // namespace latinime
    102