LVMs: Improve chapter

This commit is contained in:
Ezhh 2018-10-07 18:36:51 +01:00 committed by rubenwardy
parent 1daff5ad57
commit b854f52310

View File

@ -10,10 +10,11 @@ redirect_from: /en/chapters/lvm.html
## Introduction ## Introduction
The functions outlined in the [Basic Map Operations](environment.html) chapter The functions outlined in the [Basic Map Operations](environment.html) chapter
are easy to use and convenient, but for large areas they are not efficient. are convenient and easy to use, but for large areas they are inefficient.
Every time you call `set_node` or `get_node` your mod needs to communicate with Every time you call `set_node` or `get_node`, your mod needs to communicate with
the engine, which results in copying. Copying is slow, and will quickly kill the the engine. This results in constant individual copying operations between the
performance of your game. engine and your mod, which is slow, and will quickly decrease the performance of
your game. Using a Lua Voxel Manipulator (LVM) can be a better alternative.
* [Concepts](#concepts) * [Concepts](#concepts)
* [Reading into the LVM](#reading-into-the-lvm) * [Reading into the LVM](#reading-into-the-lvm)
@ -24,24 +25,24 @@ performance of your game.
## Concepts ## Concepts
Creating a Lua Voxel Manipulator allows you to load large areas of the map into An LVM allows you to load large areas of the map into your mod's memory.
your mod's memory at once. You can then read and write to this data without You can then read and write this data without further interaction with the
interacting with the engine at all or running any callbacks, which means that engine and without running any callbacks, which means that these
these operations are very fast. Once done, you can then write the area back into operations are very fast. Once done, you can then write the area back into
the engine and run any lighting calculations. the engine and run any lighting calculations.
## Reading into the LVM ## Reading into the LVM
You can only load a cubic area into an LVM, so you need to work out the minimum You can only load a cubic area into an LVM, so you need to work out the minimum
and maximum positions that you need to modify. Then you can create and read into and maximum positions that you need to modify. Then you can create and read into
an LVM like so: an LVM. For example:
```lua ```lua
local vm = minetest.get_voxel_manip() local vm = minetest.get_voxel_manip()
local emin, emax = vm:read_from_map(pos1, pos2) local emin, emax = vm:read_from_map(pos1, pos2)
``` ```
An LVM may not read exactly the area you tell it to, for performance reasons. For performance reasons, an LVM may not read the exact area you tell it to.
Instead, it may read a larger area. The larger area is given by `emin` and `emax`, Instead, it may read a larger area. The larger area is given by `emin` and `emax`,
which stand for *emerged min pos* and *emerged max pos*. An LVM will load the area which stand for *emerged min pos* and *emerged max pos*. An LVM will load the area
it contains for you - whether that involves loading from memory, from disk, or it contains for you - whether that involves loading from memory, from disk, or
@ -50,7 +51,7 @@ calling the map generator.
## Reading Nodes ## Reading Nodes
To read the types of nodes at particular positions, you'll need to use `get_data()`. To read the types of nodes at particular positions, you'll need to use `get_data()`.
`get_data()` returns a flat array where each entry represents the type of a This returns a flat array where each entry represents the type of a
particular node. particular node.
```lua ```lua
@ -61,7 +62,7 @@ You can get param2 and lighting data using the methods `get_light_data()` and `g
You'll need to use `emin` and `emax` to work out where a node is in the flat arrays You'll need to use `emin` and `emax` to work out where a node is in the flat arrays
given by the above methods. There's a helper class called `VoxelArea` which handles given by the above methods. There's a helper class called `VoxelArea` which handles
the calculation for you: the calculation for you.
```lua ```lua
local a = VoxelArea:new{ local a = VoxelArea:new{
@ -76,16 +77,16 @@ local idx = a:index(x, y, z)
print(data[idx]) print(data[idx])
``` ```
If you run that, you'll notice that `data[vi]` is an integer. This is because When you run this, you'll notice that `data[vi]` is an integer. This is because
the engine doesn't store nodes using their name string, as string comparision the engine doesn't store nodes using their name string, as string comparison
is slow. Instead, the engine uses a content ID. You can find out the content is slow. Instead, the engine uses a content ID. You can find out the content
ID for a particular type of node like so: ID for a particular type of node with `get_content_id()`. For example:
```lua ```lua
local c_stone = minetest.get_content_id("default:stone") local c_stone = minetest.get_content_id("default:stone")
``` ```
and then you can check whether a node is stone like so: You can then check whether the node is stone:
```lua ```lua
local idx = a:index(x, y, z) local idx = a:index(x, y, z)
@ -94,12 +95,12 @@ if data[idx] == c_stone then
end end
``` ```
It is recommended that you find out and store the content IDs of nodes types It is recommended that you find and store the content IDs of nodes types
using load time, as the IDs of a node type will never change. Make sure to store at load time, because the IDs of a node type will never change. Make sure to store
the IDs in a local for performance reasons. the IDs in a local variable for performance reasons.
Nodes in an LVM data are stored in reverse co-ordinate order, so you should Nodes in an LVM data array are stored in reverse co-ordinate order, so you should
always iterate in the order of `z, y, x` like so: always iterate in the order `z, y, x`. For example:
```lua ```lua
for z = min.z, max.z do for z = min.z, max.z do
@ -115,14 +116,14 @@ for z = min.z, max.z do
end end
``` ```
The reason for this touches computer architecture. Reading from RAM is rather The reason for this touches on the topic of computer architecture. Reading from RAM is rather
costly, so CPUs have multiple levels of caching. If the data a process requests costly, so CPUs have multiple levels of caching. If the data a process requests
is in the cache, it can very quickly retrieve it. If the data is not in the cache, is in the cache, it can very quickly retrieve it. If the data is not in the cache,
then a cache miss occurs so it'll fetch the data it needs from RAM. Any data then a cache miss occurs and it will fetch the data it needs from RAM. Any data
surrounding the requested data is also fetched and then replaces the data in the cache as surrounding the requested data is also fetched and then replaces the data in the cache. This is
it's quite likely that the process will ask for data near there again. So a because it's quite likely that the process will ask for data near that location again. This means
good rule of optimisation is to iterate in a way that you read data one after a good rule of optimisation is to iterate in a way that you read data one after
another, and avoid *memory thrashing*. another, and avoid memory thrashing.
## Writing Nodes ## Writing Nodes
@ -141,7 +142,7 @@ for z = min.z, max.z do
end end
``` ```
When you finished setting nodes in the LVM, you then need to upload the data When you finish setting nodes in the LVM, you then need to upload the data
array to the engine: array to the engine:
```lua ```lua
@ -149,12 +150,12 @@ vm:set_data(data)
vm:write_to_map(true) vm:write_to_map(true)
``` ```
For setting lighting and param2 data, there are the appropriately named For setting lighting and param2 data, use the appropriately named
`set_light_data()` and `set_param2_data()` methods. `set_light_data()` and `set_param2_data()` methods.
`write_to_map()` takes a Boolean which is true if you want lighting to be `write_to_map()` takes a Boolean which is true if you want lighting to be
calculated. If you pass false, you need to recalculate lighting at some future calculated. If you pass false, you need to recalculate lighting at a future
date using `minetest.fix_light`. time using `minetest.fix_light`.
## Example ## Example
@ -194,9 +195,9 @@ end
## Your Turn ## Your Turn
* Create `replace_in_area(from, to, pos1, pos2)` which replaces all instances of * Create `replace_in_area(from, to, pos1, pos2)` which replaces all instances of
`from` with `to` in the area given, where from and to are node names. `from` with `to` in the area given, where `from` and `to` are node names.
* Make a function which rotates all chest nodes by 90°. * Make a function which rotates all chest nodes by 90°.
* Make a function which uses an LVM to cause mossy cobble to spread to nearby * Make a function which uses an LVM to cause mossy cobble to spread to nearby
stone and cobble nodes. stone and cobble nodes.
Does your implementation cause mossy cobble to spread more than a distance of one each Does your implementation cause mossy cobble to spread more than a distance of one each
time? How could you stop this? time? If so, how could you stop this?