AngelScript
Pre-compiled byte code

Sometimes it may be useful to use pre-compiled bytecode instead of re-building the scripts every time a script is loaded. This can for example reduce load times, or if scripts are transferred over the net, it can reduce the transfer time for large scripts.

To save already compiled code you should call the SaveByteCode method on the asIScriptModule, passing in a pointer to the object that implements the asIBinaryStream interface. The script module will then push all the bytecode and related data onto this object with the Write method.

When loading the pre-compiled bytecode you should first get an instance of the asIScriptModule, just as if compiling a new script. Instead of calling AddScriptSection followed by Build you will call the LoadByteCode method, passing it a pointer to the binary stream object. The script module will then pull all the bytecode and related data from that object with the Read method to reconstruct the script module.

Here's a very simple example of how to implement the asIBinaryStream interface:

class CBytecodeStream : public asIBinaryStream
{
public:
CBytecodeStream(FILE *fp) : f(fp) {}
void Write(const void *ptr, asUINT size)
{
if( size == 0 ) return;
fwrite(ptr, size, 1, f);
}
void Read(void *ptr, asUINT size)
{
if( size == 0 ) return;
fread(ptr, size, 1, f);
}
protected:
FILE *f;
};
See also
Generic compiler

Things to remember

  • All the objects, methods, properties, etc, used when compiling the bytecode must also be registered when loading the precompiled bytecode, otherwise the load will fail when not finding the correct entity.
  • The script engine doesn't perform validation of the pre-compiled bytecode, so the application is responsible for authenticating the source. If this authentication isn't performed by the application you'll have potential security risks in your application as the bytecode may have been manually manipulated to perform otherwise illegal tasks.
  • If the application that compiles the script code is separate from the application that will execute them, then you may register the functions and methods with null pointers so you don't need to actually implement all of them. Object properties must be registered with different offsets so that they can be differentiated by the bytecode loader. You should also set the engine property asEP_INIT_GLOBAL_VARS_AFTER_BUILD to false with asIScriptEngine::SetEngineProperty, so the script engine doesn't attempt to initialize the global variables after the script has been built. The helper add-on has a couple of functions to facilitate building an offline compiler.
  • If the application is not going to compile scripts from source code, but only load pre-compiled scripts, then it may be beneficial to compile the library with the AS_NO_COMPILER define so as to reduce the size of the executable.
  • To reduce the size of the saved bytecode you can strip the debug information from the code. This will skip saving information that is not necessary for executing the scripts, such as the name of script sections, line numbers, and the name and type of local variables.
  • If the scripts use shared entities then the size of the saved bytecode can be further reduced if the entities are declared as external. To allow the offline compiler to compile scripts with external shared entities, the offline compiler must first build another module with all the shared entities that should be allowed to be declared as external.
  • The saved bytecode is platform independent with only a couple of exceptions that is unlikely to affect application developers. The exceptions are 1) difference in size of primitive types (e.g. bool on older Mac PPC platforms) and 2) CPU floating point representation (i.e. IEEE 754 versus non-IEEE 754). Differences in pointer size (32bit vs 64bit), CPU endianess, different size of registered types and offsets of registered properties are all handled to provide platform independence.
  • The saved bytecode is not guaranteed to be compatible for different versions of the AngelScript library so make sure you use the same library version when saving the bytecode that will be used when loading it.
asIBinaryStream::Read
virtual int Read(void *ptr, asUINT size)=0
Read size bytes from the stream into the memory pointed to by ptr.
asUINT
unsigned int asUINT
32 bit unsigned integer
Definition: angelscript.h:600
asIBinaryStream
A binary stream interface.
Definition: angelscript.h:4051
asIBinaryStream::Write
virtual int Write(const void *ptr, asUINT size)=0
Write size bytes to the stream from the memory pointed to by ptr.