summaryrefslogtreecommitdiff
path: root/src/mongo/db/query/index_tag.cpp
blob: f51bf8d9015804352dcefa0b4df49ba27e84082e (plain)
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
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
/**
 *    Copyright (C) 2013 10gen Inc.
 *
 *    This program is free software: you can redistribute it and/or  modify
 *    it under the terms of the GNU Affero General Public License, version 3,
 *    as published by the Free Software Foundation.
 *
 *    This program is distributed in the hope that it will be useful,
 *    but WITHOUT ANY WARRANTY; without even the implied warranty of
 *    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *    GNU Affero General Public License for more details.
 *
 *    You should have received a copy of the GNU Affero General Public License
 *    along with this program.  If not, see <http://www.gnu.org/licenses/>.
 */

#include "mongo/db/query/index_tag.h"

#include "mongo/db/query/indexability.h"

#include <algorithm>
#include <limits>

namespace mongo {

    const size_t IndexTag::kNoIndex = std::numeric_limits<size_t>::max();

    void tagForSort(MatchExpression* tree) {
        if (!Indexability::nodeCanUseIndexOnOwnField(tree)) {
            size_t myTagValue = IndexTag::kNoIndex;
            for (size_t i = 0; i < tree->numChildren(); ++i) {
                MatchExpression* child = tree->getChild(i);
                tagForSort(child);
                IndexTag* childTag = static_cast<IndexTag*>(child->getTag());
                if (NULL != childTag) {
                    myTagValue = std::min(myTagValue, childTag->index);
                }
            }
            if (myTagValue != IndexTag::kNoIndex) {
                tree->setTag(new IndexTag(myTagValue));
            }
        }
    }

    bool TagComparison(const MatchExpression* lhs, const MatchExpression* rhs) {
        IndexTag* lhsTag = static_cast<IndexTag*>(lhs->getTag());
        size_t lhsValue = (NULL == lhsTag) ? IndexTag::kNoIndex : lhsTag->index;
        size_t lhsPos = (NULL == lhsTag) ? IndexTag::kNoIndex : lhsTag->pos;

        IndexTag* rhsTag = static_cast<IndexTag*>(rhs->getTag());
        size_t rhsValue = (NULL == rhsTag) ? IndexTag::kNoIndex : rhsTag->index;
        size_t rhsPos = (NULL == rhsTag) ? IndexTag::kNoIndex : rhsTag->pos;

        // First, order on indices.
        if (lhsValue != rhsValue) {
            // This relies on kNoIndex being larger than every other possible index.
            return lhsValue < rhsValue;
        }

        // Next, order so that the first field of a compound index appears first.
        if (lhsPos != rhsPos) {
            return lhsPos < rhsPos;
        }

        // Next, order on fields.
        int cmp = lhs->path().compare(rhs->path());
        if (0 != cmp) {
            return 0;
        }

        // Finally, order on expression type.
        return lhs->matchType() < rhs->matchType();
    }

    void sortUsingTags(MatchExpression* tree) {
        for (size_t i = 0; i < tree->numChildren(); ++i) {
            sortUsingTags(tree->getChild(i));
        }
        std::vector<MatchExpression*>* children = tree->getChildVector();
        if (NULL != children) {
            std::sort(children->begin(), children->end(), TagComparison);
        }
    }

} // namespace mongo