Blame view

morfeusz/fsa/simplefsa_impl.hpp 2.12 KB
Michał Lenart authored
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
/* 
 * File:   simplefsa_impl.hpp
 * Author: mlenart
 *
 * Created on November 4, 2013, 1:05 PM
 */

#ifndef SIMPLEFSA_IMPL_HPP
#define	SIMPLEFSA_IMPL_HPP

#pragma pack(push)  /* push current alignment to stack */
#pragma pack(1)     /* set alignment to 1 byte boundary */

struct StateData {
    unsigned transitionsNum : 7;
    unsigned accepting : 1;
};

#pragma pack(pop)   /* restore original alignment from stack */

template <class T>
SimpleFSA<T>::SimpleFSA(const unsigned char* ptr, const Deserializer<T>& deserializer)
: FSA<T>(ptr, deserializer) {
}

template <class T>
SimpleFSA<T>::~SimpleFSA() {

}

static unsigned int decodeOffset(const unsigned char* ptr) {
    unsigned int res = 0;
    res = ptr[0] << 16 | ptr[1] << 8 | ptr[2];
    return res;
}

template <class T>
void SimpleFSA<T>::proceedToNext(const char c, State<T>& state) const {
    const unsigned char* fromPointer = this->initialStatePtr + state.getOffset();
Michał Lenart authored
40
    long transitionsTableOffset = sizeof (StateData);
Michał Lenart authored
41
42
43
    if (state.isAccepting()) {
        transitionsTableOffset += state.getValueSize();
    }
Michał Lenart authored
44
    StateData stateData = *reinterpret_cast<const StateData*>(fromPointer);
Michał Lenart authored
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
    const unsigned char* foundTransition = fromPointer + transitionsTableOffset;
    bool found = false;
    for (int i = 0; i < stateData.transitionsNum; i++, foundTransition += 4) {
        if ((char) *foundTransition == c) {
            found = true;
            break;
        }
    }
    //    const_cast<Counter*>(&counter)->increment(foundTransition - transitionsStart + 1);
    if (!found) {
        state.setNextAsSink();
    }
    else {
        unsigned int offset = decodeOffset(foundTransition + 1);
        const unsigned char* nextStatePointer = this->initialStatePtr + offset;
Michał Lenart authored
60
        const StateData* nextStateData = reinterpret_cast<const StateData*>(nextStatePointer);
Michał Lenart authored
61
62
        if (nextStateData->accepting) {
            T object;
Michał Lenart authored
63
            long size = this->deserializer.deserialize(nextStatePointer + sizeof (StateData), object);
Michał Lenart authored
64
65
66
67
68
69
70
71
72
            state.setNext(offset, object, size);
        } else {
            state.setNext(offset);
        }
    }
}

#endif	/* SIMPLEFSA_IMPL_HPP */