ruby-x.github.io/rubyx/optimisations.md

85 lines
5.0 KiB
Markdown
Raw Normal View History

2014-06-16 15:04:46 +02:00
---
2017-01-02 01:45:44 +02:00
layout: rubyx
2014-06-16 15:04:46 +02:00
title: Optimisation ideas
---
2017-01-02 01:45:44 +02:00
I won't manage to implement all of these idea in the beginning, so i just jot them down.
2014-06-16 15:04:46 +02:00
### Avoid dynamic lookup
This off course is a broad topic, which may be seen under the topic of caching. Slightly wrongly though in my view, as avoiding them is really the aim. Especially for variables.
#### I - Instance Variables
Ruby has dynamic instance variables, meaning you can add a new one at any time. This is as it should be.
But this can easily lead to a dictionary/hash type of implementation. As variable "lookup" is probably *the* most
2017-01-02 01:45:44 +02:00
common thing an OO system does, that leads to bad performance (unneccessarily).
2014-06-16 15:04:46 +02:00
2017-01-02 01:45:44 +02:00
So instead we keep variables layed out c++ style, continous, array style, at the address of the object. Then we have
to manage that in a dynamic manner. This (as i mentioned [here](memory.html)) is done by the indirection of the Type. A Type is
2014-06-16 15:04:46 +02:00
a dynamic structure mapping names to indexes (actually implemented as an array too, but the api is hash-like).
2016-02-25 19:10:56 -08:00
When a new variable is added, we create a *new* Type and change the Type of the object. We can do this as the Type will
2014-06-16 15:04:46 +02:00
determine the Class of the object, which stays the same. The memory page mentions how this works with constant sized objects.
2014-07-11 21:02:13 +03:00
So, Problem one fixed: instance variable access at O(1)
2014-06-16 15:04:46 +02:00
#### II - Method lookup
Off course that helps with Method access. All Methods are at the end variables on some (class) object. But as we can't very well have the same (continuous) index for a given method name on all classes, it has to be looked up. Or does it?
2017-01-02 01:45:44 +02:00
Well, yes it does, but maybe not more than once: We can conceivably store the result, except off course not in a dynamic
2014-06-16 15:04:46 +02:00
structure as that would defeat the purpose.
In fact there could be several caching strategies, possibly for different use cases, possibly determined by actual run-time
2014-06-16 19:43:42 +02:00
measurements, but for now I just destribe a simeple one using Data-Blocks, Plocks.
2014-06-16 15:04:46 +02:00
2017-01-02 01:45:44 +02:00
So at a call-site, we know the name of the function we want to call, and the object we want to call it on, and so have to
find the actual function object, and by that the actual call address. In abstract terms we want to create a switch with
2014-06-16 15:04:46 +02:00
3 cases and a default.
2017-01-02 01:45:44 +02:00
So the code is something like, if first cache hit, call first cache , .. times three and if not do the dynamic lookup.
2014-06-16 19:43:42 +02:00
The Plock can store those cache hits inside the code. So then we "just" need to get the cache loaded.
2014-06-16 15:04:46 +02:00
2017-01-02 01:45:44 +02:00
Initializing the cached values is by normal lazy initialization. Ie we check for nil and if so we do the dynamic lookup, and store the result.
2014-06-16 15:04:46 +02:00
2017-01-02 01:45:44 +02:00
Remember, we cache Type against function address. Since Types never change, we're done. We could (as hinted above)
2014-06-16 15:04:46 +02:00
do things with counters or robins, but that is for later.
2017-01-02 01:45:44 +02:00
Alas: While Types are constant, darn the ruby, method implementations can actually change! And while it is tempting to
2016-02-25 19:10:56 -08:00
just create a new Type for that too, that would mean going through existing objects and changing the Type, nischt gut.
2014-06-16 15:04:46 +02:00
So we need change notifications, so when we cache, we must register a change listener and update the generated function,
or at least nullify it.
### Inlining
2017-01-02 01:45:44 +02:00
Ok, this may not need too much explanation. Just work. It may be intersting to experiment how much this saves, and how much
inlining is useful. I could imagine at some point it's the register shuffling that determines the effort, not the
2014-06-16 15:04:46 +02:00
actual call.
2017-01-02 01:45:44 +02:00
Again the key is the update notifications when some of the inlined functions have changed.
2014-06-16 15:04:46 +02:00
2017-01-02 01:45:44 +02:00
And it is important to code the functions so that they have a single exit point, otherwise it gets messy. Up to now this
2014-06-16 15:04:46 +02:00
was quite simple, but then blocks and exceptions are undone.
### Register negotiation
This is a little less baked, but it comes from the same idea as inlining. As calling functions is a lot of register
shuffling, we could try to avoid some of that.
More precisely, usually calling conventions have registers in which arguments are passed. And to call an "unknown", ie any function, some kind of convention is neccessary.
2017-01-02 01:45:44 +02:00
But on "cached" functions, where the function is know, it is possible to do something else. And since we have the source
2014-06-16 15:04:46 +02:00
(ast) of the function around, we can do things previouly imposible.
One such thing may be to recompile the function to acccept arguments exactly where they are in the calling function. Well, now that it's written down. it does sound a lot like inlining, except without the inlining:-)
2017-01-02 01:45:44 +02:00
An expansion if this idea would be to have a Negotiator on every function call. Meaning that the calling function would not
2014-06-16 15:04:46 +02:00
do any shuffling, but instead call a Negotiator, and the Negotiator does the shuffling and calling of the function.
This only really makes sense if the register shuffling information is encoded in the Negotiator object (and does not have
to be passed).
2017-01-02 01:45:44 +02:00
Negotiators could do some counting and do the recompiling when it seems worth it. The Negotiator would remove itself from
2014-07-17 00:54:36 +03:00
the chain and connect called and new receiver directly. How much is in this i couldn't say though.