Commit 7a61cb48 authored by Kevin Modzelewski's avatar Kevin Modzelewski

Merge pull request #268 from undingen/RuntimeICCache

Add a simple cache for runtime ICs
parents 636aca17 bc373da9
...@@ -311,11 +311,12 @@ extern "C" Box* sum(Box* container, Box* initial) { ...@@ -311,11 +311,12 @@ extern "C" Box* sum(Box* container, Box* initial) {
if (initial->cls == str_cls) if (initial->cls == str_cls)
raiseExcHelper(TypeError, "sum() can't sum strings [use ''.join(seq) instead]"); raiseExcHelper(TypeError, "sum() can't sum strings [use ''.join(seq) instead]");
BinopIC pp; static RuntimeICCache<BinopIC, 3> runtime_ic_cache;
std::shared_ptr<BinopIC> pp = runtime_ic_cache.getIC(__builtin_return_address(0));
Box* cur = initial; Box* cur = initial;
for (Box* e : container->pyElements()) { for (Box* e : container->pyElements()) {
cur = pp.call(cur, e, AST_TYPE::Add); cur = pp->call(cur, e, AST_TYPE::Add);
} }
return cur; return cur;
} }
......
...@@ -65,7 +65,7 @@ public: ...@@ -65,7 +65,7 @@ public:
class BinopIC : public RuntimeIC { class BinopIC : public RuntimeIC {
public: public:
BinopIC() : RuntimeIC((void*)binop, 1, 160) {} BinopIC() : RuntimeIC((void*)binop, 2, 160) {}
Box* call(Box* lhs, Box* rhs, int op_type) { return (Box*)call_ptr(lhs, rhs, op_type); } Box* call(Box* lhs, Box* rhs, int op_type) { return (Box*)call_ptr(lhs, rhs, op_type); }
}; };
...@@ -77,6 +77,56 @@ public: ...@@ -77,6 +77,56 @@ public:
bool call(Box* obj) { return call_bool(obj); } bool call(Box* obj) { return call_bool(obj); }
}; };
template <class ICType, unsigned cache_size> class RuntimeICCache {
private:
struct PerCallerIC {
void* caller_addr;
std::shared_ptr<ICType> ic;
};
PerCallerIC ics[cache_size];
unsigned next_to_replace;
RuntimeICCache(const RuntimeICCache&) = delete;
void operator=(const RuntimeICCache&) = delete;
PerCallerIC* findBestSlotToReplace() {
// search for an unassigned slot
for (unsigned i = 0; i < cache_size; ++i) {
if (!ics[i].caller_addr)
return &ics[i];
}
PerCallerIC* ic = &ics[next_to_replace];
++next_to_replace;
if (next_to_replace >= cache_size)
next_to_replace = 0;
return ic;
}
public:
RuntimeICCache() : next_to_replace(0) {
for (unsigned i = 0; i < cache_size; ++i)
ics[i].caller_addr = 0;
}
std::shared_ptr<ICType> getIC(void* caller_addr) {
assert(caller_addr);
// try to find a cached IC for the caller
for (unsigned i = 0; i < cache_size; ++i) {
if (ics[i].caller_addr == caller_addr)
return ics[i].ic;
}
// could not find a cached runtime IC, create new one and save it
PerCallerIC* slot_to_replace = findBestSlotToReplace();
std::shared_ptr<ICType> ic = std::make_shared<ICType>();
slot_to_replace->caller_addr = caller_addr;
slot_to_replace->ic = ic;
return ic;
}
};
} // namespace pyston } // namespace pyston
#endif #endif
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment