2020-04-14 21:11:54 +02:00
mapgen_rivers = { }
2020-04-09 21:13:38 +02:00
2020-04-14 21:11:54 +02:00
local modpath = minetest.get_modpath ( minetest.get_current_modname ( ) ) .. ' / '
2021-06-03 19:21:45 +02:00
mapgen_rivers.modpath = modpath
2021-06-03 23:30:04 +02:00
mapgen_rivers.world_data_path = minetest.get_worldpath ( ) .. ' /river_data/ '
2020-04-09 21:13:38 +02:00
2021-09-07 11:59:33 +02:00
if minetest.get_mapgen_setting ( " mg_name " ) ~= " singlenode " then
minetest.set_mapgen_setting ( " mg_name " , " singlenode " , true )
2022-01-03 16:22:55 +01:00
minetest.log ( " warning " , " [mapgen_rivers] Mapgen set to singlenode " )
2021-09-07 11:59:33 +02:00
end
2020-04-14 21:11:54 +02:00
dofile ( modpath .. ' settings.lua ' )
2020-04-09 21:13:38 +02:00
2021-06-06 13:25:43 +02:00
local sea_level = mapgen_rivers.settings . sea_level
local elevation_chill = mapgen_rivers.settings . elevation_chill
local use_distort = mapgen_rivers.settings . distort
local use_biomes = mapgen_rivers.settings . biomes
2020-11-10 13:25:04 +01:00
local use_biomegen_mod = use_biomes and minetest.global_exists ( ' biomegen ' )
use_biomes = use_biomes and not use_biomegen_mod
2020-05-24 11:38:47 +02:00
2020-11-10 13:25:04 +01:00
if use_biomegen_mod then
biomegen.set_elevation_chill ( elevation_chill )
end
2020-05-24 11:38:47 +02:00
dofile ( modpath .. ' noises.lua ' )
2020-04-12 09:40:10 +02:00
2020-05-23 15:52:16 +02:00
local heightmaps = dofile ( modpath .. ' heightmap.lua ' )
2020-04-26 17:13:38 +02:00
-- Linear interpolation
2020-04-13 10:31:38 +02:00
local function interp ( v00 , v01 , v11 , v10 , xf , zf )
2020-04-12 09:40:10 +02:00
local v0 = v01 * xf + v00 * ( 1 - xf )
local v1 = v11 * xf + v10 * ( 1 - xf )
2020-04-09 21:13:38 +02:00
return v1 * zf + v0 * ( 1 - zf )
end
2022-01-03 11:56:16 +01:00
-- Localize for performance
local floor , min = math.floor , math.min
2020-04-09 21:13:38 +02:00
local data = { }
2020-05-23 18:13:00 +02:00
local noise_x_obj , noise_z_obj , noise_distort_obj , noise_heat_obj , noise_heat_blend_obj
2020-05-23 15:52:16 +02:00
local noise_x_map = { }
local noise_z_map = { }
local noise_distort_map = { }
2020-05-23 18:13:00 +02:00
local noise_heat_map = { }
local noise_heat_blend_map = { }
2020-05-23 15:52:16 +02:00
local mapsize
local init = false
2021-06-25 21:05:14 +02:00
local sumtime = 0
local sumtime2 = 0
local ngen = 0
2020-04-09 21:13:38 +02:00
local function generate ( minp , maxp , seed )
2022-01-03 16:22:55 +01:00
minetest.log ( " info " , ( " [mapgen_rivers] Generating from %s to %s " ) : format ( minetest.pos_to_string ( minp ) , minetest.pos_to_string ( maxp ) ) )
2021-06-25 21:05:14 +02:00
2020-05-23 15:52:16 +02:00
local chulens = {
x = maxp.x - minp.x + 1 ,
y = maxp.y - minp.y + 1 ,
z = maxp.z - minp.z + 1 ,
}
if not init then
mapsize = {
x = chulens.x ,
y = chulens.y + 1 ,
z = chulens.z ,
}
2020-07-21 14:01:29 +02:00
if use_distort then
noise_x_obj = minetest.get_perlin_map ( mapgen_rivers.noise_params . distort_x , mapsize )
noise_z_obj = minetest.get_perlin_map ( mapgen_rivers.noise_params . distort_z , mapsize )
noise_distort_obj = minetest.get_perlin_map ( mapgen_rivers.noise_params . distort_amplitude , chulens )
end
2020-11-10 13:25:04 +01:00
if use_biomes then
noise_heat_obj = minetest.get_perlin_map ( mapgen_rivers.noise_params . heat , chulens )
noise_heat_blend_obj = minetest.get_perlin_map ( mapgen_rivers.noise_params . heat_blend , chulens )
end
2020-05-23 15:52:16 +02:00
init = true
end
2021-06-25 21:05:14 +02:00
local t0 = os.clock ( )
2020-05-23 18:13:00 +02:00
local minp2d = { x = minp.x , y = minp.z }
2020-07-21 14:01:29 +02:00
if use_distort then
noise_x_obj : get_3d_map_flat ( minp , noise_x_map )
noise_z_obj : get_3d_map_flat ( minp , noise_z_map )
noise_distort_obj : get_2d_map_flat ( minp2d , noise_distort_map )
end
2020-11-10 13:25:04 +01:00
if use_biomes then
noise_heat_obj : get_2d_map_flat ( minp2d , noise_heat_map )
noise_heat_blend_obj : get_2d_map_flat ( minp2d , noise_heat_blend_map )
end
2020-05-23 15:52:16 +02:00
2020-07-21 14:01:29 +02:00
local terrain_map , lake_map , incr , i_origin
if use_distort then
local xmin , xmax , zmin , zmax = minp.x , maxp.x , minp.z , maxp.z
local i = 0
local i2d = 0
for z = minp.z , maxp.z do
for y = minp.y , maxp.y + 1 do
for x = minp.x , maxp.x do
i = i + 1
i2d = i2d + 1
local distort = noise_distort_map [ i2d ]
local xv = noise_x_map [ i ] * distort + x
if xv < xmin then xmin = xv end
if xv > xmax then xmax = xv end
noise_x_map [ i ] = xv
local zv = noise_z_map [ i ] * distort + z
if zv < zmin then zmin = zv end
if zv > zmax then zmax = zv end
noise_z_map [ i ] = zv
end
i2d = i2d - chulens.x
2020-05-23 15:52:16 +02:00
end
end
2022-01-03 11:56:16 +01:00
local pminp = { x = floor ( xmin ) , z = floor ( zmin ) }
local pmaxp = { x = floor ( xmax ) + 1 , z = floor ( zmax ) + 1 }
2020-07-21 14:01:29 +02:00
incr = pmaxp.x - pminp.x + 1
i_origin = 1 - pminp.z * incr - pminp.x
terrain_map , lake_map = heightmaps ( pminp , pmaxp )
else
terrain_map , lake_map = heightmaps ( minp , maxp )
end
2020-05-23 15:52:16 +02:00
2022-01-03 15:47:03 +01:00
-- Check that there is at least one position that reaches min y
if minp.y > sea_level then
local y0 = minp.y
local is_empty = true
for i = 1 , # terrain_map do
if terrain_map [ i ] >= y0 or lake_map [ i ] >= y0 then
is_empty = false
break
end
end
-- If not, skip chunk
if is_empty then
local t = os.clock ( ) - t0
ngen = ngen + 1
sumtime = sumtime + t
sumtime2 = sumtime2 + t * t
2022-01-03 16:22:55 +01:00
minetest.log ( " verbose " , " [mapgen_rivers] Skipping empty chunk (fully above ground level) " )
minetest.log ( " verbose " , ( " [mapgen_rivers] Done in %5.3f s " ) : format ( t ) )
2022-01-03 15:47:03 +01:00
return
end
end
2020-04-09 21:13:38 +02:00
local c_stone = minetest.get_content_id ( " default:stone " )
local c_dirt = minetest.get_content_id ( " default:dirt " )
local c_lawn = minetest.get_content_id ( " default:dirt_with_grass " )
2020-05-23 18:13:00 +02:00
local c_dirtsnow = minetest.get_content_id ( " default:dirt_with_snow " )
local c_snow = minetest.get_content_id ( " default:snowblock " )
2020-04-09 21:13:38 +02:00
local c_sand = minetest.get_content_id ( " default:sand " )
local c_water = minetest.get_content_id ( " default:water_source " )
local c_rwater = minetest.get_content_id ( " default:river_water_source " )
2020-05-23 18:13:00 +02:00
local c_ice = minetest.get_content_id ( " default:ice " )
2020-04-09 21:13:38 +02:00
local vm , emin , emax = minetest.get_mapgen_object ( " voxelmanip " )
vm : get_data ( data )
local a = VoxelArea : new ( { MinEdge = emin , MaxEdge = emax } )
local ystride = a.ystride -- Tip : the ystride of a VoxelArea is the number to add to the array index to get the index of the position above. It's faster because it avoids to completely recalculate the index.
2020-04-13 15:01:54 +02:00
2020-05-23 15:52:16 +02:00
local nid = mapsize.x * ( mapsize.y - 1 ) + 1
local incrY = - mapsize.x
local incrX = 1 - mapsize.y * incrY
local incrZ = mapsize.x * mapsize.y - mapsize.x * incrX - mapsize.x * mapsize.y * incrY
2020-05-23 18:13:00 +02:00
local i2d = 1
2021-06-25 21:05:14 +02:00
2020-05-23 15:52:16 +02:00
for z = minp.z , maxp.z do
for x = minp.x , maxp.x do
2022-01-03 11:56:16 +01:00
local ivm = a : index ( x , maxp.y + 1 , z )
2020-05-23 15:52:16 +02:00
local ground_above = false
2020-11-10 13:25:04 +01:00
local temperature
if use_biomes then
temperature = noise_heat_map [ i2d ] + noise_heat_blend_map [ i2d ]
end
2020-07-21 14:01:29 +02:00
local terrain , lake
if not use_distort then
terrain = terrain_map [ i2d ]
lake = lake_map [ i2d ]
end
2020-05-23 15:52:16 +02:00
2020-07-21 14:01:29 +02:00
for y = maxp.y + 1 , minp.y , - 1 do
if use_distort then
local xn = noise_x_map [ nid ]
local zn = noise_z_map [ nid ]
2022-01-03 11:56:16 +01:00
local x0 = floor ( xn )
local z0 = floor ( zn )
2020-07-21 14:01:29 +02:00
local i0 = i_origin + z0 * incr + x0
local i1 = i0 + 1
local i2 = i1 + incr
local i3 = i2 - 1
terrain = interp ( terrain_map [ i0 ] , terrain_map [ i1 ] , terrain_map [ i2 ] , terrain_map [ i3 ] , xn - x0 , zn - z0 )
2022-01-03 11:56:16 +01:00
lake = min ( lake_map [ i0 ] , lake_map [ i1 ] , lake_map [ i2 ] , lake_map [ i3 ] )
2020-07-21 14:01:29 +02:00
end
2020-05-23 15:52:16 +02:00
if y <= maxp.y then
local is_lake = lake > terrain
if y <= terrain then
2020-11-10 13:25:04 +01:00
if not use_biomes or y <= terrain - 1 or ground_above then
2020-04-09 21:13:38 +02:00
data [ ivm ] = c_stone
2020-05-23 18:13:00 +02:00
elseif is_lake or y < sea_level then
2020-05-23 15:52:16 +02:00
data [ ivm ] = c_sand
else
2020-05-23 18:13:00 +02:00
local temperature_y = temperature - y * elevation_chill
if temperature_y >= 15 then
data [ ivm ] = c_lawn
elseif temperature_y >= 0 then
data [ ivm ] = c_dirtsnow
else
data [ ivm ] = c_snow
end
2020-04-09 21:13:38 +02:00
end
2020-05-23 15:52:16 +02:00
elseif y <= lake and lake > sea_level then
2020-11-10 13:25:04 +01:00
if not use_biomes or temperature - y * elevation_chill >= 0 then
2020-05-23 18:13:00 +02:00
data [ ivm ] = c_rwater
else
data [ ivm ] = c_ice
end
2020-05-23 15:52:16 +02:00
elseif y <= sea_level then
2020-04-09 21:13:38 +02:00
data [ ivm ] = c_water
end
end
2020-05-23 15:52:16 +02:00
ground_above = y <= terrain
2022-01-03 11:56:16 +01:00
ivm = ivm - ystride
2020-07-21 14:01:29 +02:00
if use_distort then
nid = nid + incrY
end
end
if use_distort then
nid = nid + incrX
2020-04-09 21:13:38 +02:00
end
2020-05-23 18:13:00 +02:00
i2d = i2d + 1
2020-04-09 21:13:38 +02:00
end
2020-07-21 14:01:29 +02:00
if use_distort then
nid = nid + incrZ
end
2020-04-09 21:13:38 +02:00
end
2020-11-10 13:25:04 +01:00
if use_biomegen_mod then
2020-11-11 13:58:23 +01:00
biomegen.generate_all ( data , a , vm , minp , maxp , seed )
2020-11-10 13:25:04 +01:00
else
2020-11-11 13:58:23 +01:00
vm : set_data ( data )
2020-11-14 19:13:58 +01:00
minetest.generate_ores ( vm , minp , maxp )
2020-11-10 13:25:04 +01:00
end
2020-04-09 21:13:38 +02:00
vm : set_lighting ( { day = 0 , night = 0 } )
vm : calc_lighting ( )
vm : update_liquids ( )
vm : write_to_map ( )
2021-06-25 21:05:14 +02:00
2022-01-03 15:47:03 +01:00
local t = os.clock ( ) - t0
2021-06-25 21:05:14 +02:00
ngen = ngen + 1
sumtime = sumtime + t
sumtime2 = sumtime2 + t * t
2022-01-03 16:22:55 +01:00
minetest.log ( " verbose " , ( " [mapgen_rivers] Done in %5.3f s " ) : format ( t ) )
2020-04-09 21:13:38 +02:00
end
minetest.register_on_generated ( generate )
2021-06-25 21:05:14 +02:00
minetest.register_on_shutdown ( function ( )
local avg = sumtime / ngen
local std = math.sqrt ( sumtime2 / ngen - avg * avg )
2022-01-03 16:22:55 +01:00
minetest.log ( " action " , ( " [mapgen_rivers] Mapgen statistics: \n - Mapgen calls: %4d \n - Mean time: %5.3f s \n - Standard deviation: %5.3f s " ) : format ( ngen , avg , std ) )
2021-06-25 21:05:14 +02:00
end )