• Log InLog In
  • Register
Liquid`
Team Liquid Liquipedia
EDT 08:43
CEST 14:43
KST 21:43
  • Home
  • Forum
  • Calendar
  • Streams
  • Liquipedia
  • Features
  • Store
  • EPT
  • TL+
  • StarCraft 2
  • Brood War
  • Smash
  • Heroes
  • Counter-Strike
  • Overwatch
  • Liquibet
  • Fantasy StarCraft
  • TLPD
  • StarCraft 2
  • Brood War
  • Blogs
Forum Sidebar
Events/Features
News
Featured News
Code S Season 1 - RO8 Preview1[ASL21] Ro8 Preview Pt2: Progenitors8Code S Season 1 - RO12 Group A: Rogue, Percival, Solar, Zoun13[ASL21] Ro8 Preview Pt1: Inheritors16[ASL21] Ro16 Preview Pt2: All Star10
Community News
Weekly Cups (April 27-May 4): Clem takes triple0RSL Revival: Season 5 - Qualifiers and Main Event11Code S Season 1 (2026) - RO12 Results12026 GSL Season 1 Qualifiers25Maestros of the Game 2 announced9
StarCraft 2
General
Code S Season 1 - RO8 Preview Behind the Blue - Team Liquid History Book Weekly Cups (April 27-May 4): Clem takes triple Blizzard Classic Cup @ BlizzCon 2026 - $100k prize pool Code S Season 1 (2026) - RO12 Results
Tourneys
GSL Code S Season 1 (2026) Sparkling Tuna Cup - Weekly Open Tournament RSL Revival: Season 5 - Qualifiers and Main Event StarCraft Evolution League (SC Evo Biweekly) 2026 GSL Season 2 Qualifiers
Strategy
Custom Maps
[D]RTS in all its shapes and glory <3 [A] Nemrods 1/4 players [M] (2) Frigid Storage
External Content
Mutation # 524 Death and Taxes The PondCast: SC2 News & Results Mutation # 523 Firewall Mutation # 522 Flip My Base
Brood War
General
(Spoiler) Asl ro8 D winner interview BW General Discussion BGH Auto Balance -> http://bghmmr.eu/ Do we have a pimpest plays list? AI Question
Tourneys
[ASL21] Ro8 Day 3 [ASL21] Ro8 Day 4 [Megathread] Daily Proleagues [ASL21] Ro8 Day 2
Strategy
Simple Questions, Simple Answers Fighting Spirit mining rates What's the deal with APM & what's its true value Any training maps people recommend?
Other Games
General Games
Dawn of War IV Stormgate/Frost Giant Megathread OutLive 25 (RTS Game) Daigo vs Menard Best of 10 Nintendo Switch Thread
Dota 2
The Story of Wings Gaming
League of Legends
G2 just beat GenG in First stand
Heroes of the Storm
Simple Questions, Simple Answers Heroes of the Storm 2.0
Hearthstone
Deck construction bug Heroes of StarCraft mini-set
TL Mafia
Vanilla Mini Mafia Mafia Game Mode Feedback/Ideas TL Mafia Community Thread Five o'clock TL Mafia
Community
General
European Politico-economics QA Mega-thread US Politics Mega-thread Russo-Ukrainian War Thread 3D technology/software discussion Canadian Politics Mega-thread
Fan Clubs
The IdrA Fan Club
Media & Entertainment
Anime Discussion Thread [Manga] One Piece [Req][Books] Good Fantasy/SciFi books
Sports
2024 - 2026 Football Thread Formula 1 Discussion McBoner: A hockey love story
World Cup 2022
Tech Support
streaming software Strange computer issues (software) [G] How to Block Livestream Ads
TL Community
The Automated Ban List
Blogs
Movie Stars In Video Games: …
TrAiDoS
ramps on octagon
StaticNine
Broowar part 2
qwaykee
Funny Nicknames
LUCKY_NOOB
Customize Sidebar...

Website Feedback

Closed Threads



Active: 1880 users

Terrain Generation I --- Noise

Blogs > Namrufus
Post a Reply
Namrufus
Profile Blog Joined August 2011
United States396 Posts
Last Edited: 2013-11-12 04:56:15
November 12 2013 04:42 GMT
#1
Introduction


In this series of posts, I'll describe my methods and experience writing a terrain generator for Minecraft using the craftbukkit world generator API -- (the generator itself will be written agnostic of any specific framework or library - so a port to Forge or even to a similar type of game may be possible)

This is the first time I've written anything remotely like this, so, if you'd like, please give feedback on what you liked or didn't, and any thoughts on how you think the post could be better.

edit: 300th post!



Terrain Generation I --- Noise


If you visit any online resource about terrain or world generation, chances are good that it will mention Perlin noise and associated "Coherent Randomness" algorithms. Perlin noise is in fact used in a critical role in MInecraft's native generator, as Notch describes in Terrain Generation, Part 1 (of 1). This project will be no different.

Perlin noise generation is a technique first engineered to reduce the memory requirements of textures in early computer graphics. This - and similar coherent randomness algorithms -- are useful because they allow us to specify things such as the period (the scale of the noise, in the context of a terrain generator, will allow the generated noise form the basis of things of all scales, from tiny hills to entire continents) and the amplitude (the "height" of the noise). By composing multiple noise generators, it will be possible to generate a world.

Because I want this generator to work similarly to Minecraft's world generator, there are a couple of unique requirements:
  • "Infinite" generation: The noise generator should able to generate noise without any built in limits, terrain should not repeat. Technical limits are OK (the same type of limit at the root of Minecraft's fabled "8 times the surface of the earth" claim)
  • Realtime arbitrary generation order: Minecraft terrain is generated on the fly, as the player explores the world, this means that terrain chunks can be generated at any order, at any time, across different play sessions, Minecraft versions, Java versions, generator versions, or even devices.

Makin' some noise


For this project I've implemented a type of closely related algorithm known as value noise (slightly simpler than Perlin noise - in both implementation and efficiency - but sufficient for the purposes of this project)

As decribed on the wiki, value noise works by generating a grid or lattice of randomized value points, then interpolating between the values in order to create a smooth, randomized surface.

[image loading]
Grid points. As you can see, each grid point generates a value and the surface is interpolated between those values

The interpolation is relatively simple, the real trick here is the grid point value generation.


To Infinity...


Perlin noise, in it's original form is slightly unsuited to this task, due to the fact that the values used at the grid points are precomputed. A grid of points are generated, if the noise is sampled outside of the grid, the pattern is simply repeated; this has the advantage of being fast - requiring only an array lookup instead of a random number generation - but has the disadvantage of creating a repeating pattern.

In light of the requirements above, grid values will need to be effectively infinite in number, generated on the fly, have no visible patterns or repeats, and be reproducible across program runs. In short we need a function of the format:

 noise(seed, grid x, grid y, grid z)  ->  value 


the seed will be an integer value that will be stored across program runs.

After some experimentation, my solution is as follows.

noise(seed, x, y, z)
a <- cantor(y, z)
b <- cantor(x, a)
c <- cantor(seed, b)
result <- randomize(c)


randomize(k)
is this 64 bit linear congruential pseudo-random number generator - this algorithm takes an input and multiplies it by a huge number, "shooting" it past the 64 bit overflow value (the largest possible 64 bit value) causing it to "overflow", resulting in a number that (by casual inspection) is unrelated to the original value.This is not the best RNG function, nor is this the intended use case for it, but it is fast and seems to produce good results (visually).

cantor(k1, k2)
is the cantor pairing function. As mentioned in the wikipedia article, this function takes two positive integers and combines them into a single positive integer, with each pair of numbers resulting in a unique result.

The algorithm combines all of the grid coordinates together (with some spicyness from the seed value) into a unique integer using repeated application of the cantor pairing function, that value is then "randomized" using the linear congruential generator to obtain the final randomized value.

My first images using this method looked like this:

[image loading]

Uh-oh, symmetry. I had forgotten that the cantor pairing function was designed for positive integers only. Adding a function to "interleave" the inputs into positive integers fixed the problem.

for the record: the cantor pairing function:


interleave(k) -> -2 * k k < 0
-> 2 * k + 1 k >= 0

cantor(k1, k2)
k1' <- interleave(k1)
k2' <- interleave(k2)
result <- ((k1' + k2') * (k1' + k2' + 1)) / 2 + k2'


The final result? A seeded, infinite, (as far as I can tell) unrepeating noise generator (with basically zero memory imprint)

[image loading]


A bit square looking, mostly because of the way that I am interpolating between the values. I think it compliments Minecraft's blocky aesthetic.


I love it LOUD!


What can be created using noise generators? I won't talk about the main terrain generation yet (mostly because I am still experimenting), however I will describe the "secret sauce" that makes all this worth while.

The secret ingredient? Fractals.

[image loading]
Not quite
source

In my experience, discussion of fractals in relation to terrain generation tends to be slightly mystical: "How do you make realistic looking terrain?" "fractals". The reality is quite simple: real landscapes have varying levels of detail, this detail tends to be self-similar (Hills look like down-scaled mountains, lumpy coastlines resolve into smaller bumps as you zoom in). Self-similarity is the defining feature of a fractal.

In practice this generally means we use an approximation of Fractal brownian motion. A fractal brownian surface has the prominance (amplitude, or strength) of features be inversely proportional to the frequency (or size) of said features - if a feature is twice the frequency (half as large) it will have half the strength, at 3 times the frequency (1 third scale) it will have 1/3 strength, and so on). The general trend that emerges is that the surface as a whole is dominated by the largest features (in the context of a terrain generator, continents) which are in turn "perturbed" by smaller scale features like coastline bumps, inlets, islands, lakes and other interesting features.

First, assemble a couple of generators, double the frequency for each successive generator

[image loading]

Halve the amplitude of each successive generator.

[image loading]

sum them all together and you get this, pretty cool, right? It really masks the "square" appearance of the noise.

[image loading]

Just for fun, add a simple gradient and a "sea level".

[image loading]

Not all that impressive. We're not done yet, of course.


Next Time


Infinite seeded Voronoi cells using the same techniques as used to make the value noise generator. Also: the basic high-level bones of the generator: continents! mountain ranges! climate! I'll explain how my generator will be different from Minecraft's native terrain generator.

I'm also planning on putting the WIP project up on my Github, I'll have a link on a later post.

If you have any questions, ask and I'll answer as well as I can.

Thanks for reading!

*****
This is it... the alpaca lips.
v0rtex
Profile Joined November 2011
123 Posts
November 12 2013 07:57 GMT
#2
Very interested to see where you go with this! Nice work!
JD, Snute, TLO, Soulkey, $o$, HerO, Suppy, Hendralisk, MKP, Maru
jrkirby
Profile Blog Joined August 2010
United States1510 Posts
November 12 2013 09:45 GMT
#3
I implemented perlin noise a couple months ago: http://imgur.com/8DU6vwh

I also recently saw a video on procedural generation that proposed an awesome technique:


Good stuff.
Namrufus
Profile Blog Joined August 2011
United States396 Posts
Last Edited: 2013-11-12 21:34:45
November 12 2013 21:34 GMT
#4
On November 12 2013 16:57 v0rtex wrote:
Very interested to see where you go with this! Nice work!


thanks! To give you an idea of some of the stuff I'm aiming for: + Show Spoiler +
[image loading]
an image from a much older version of the project.


On November 12 2013 18:45 jrkirby wrote:
I implemented perlin noise a couple months ago: http://imgur.com/8DU6vwh

I also recently saw a video on procedural generation that proposed an awesome technique: https://www.youtube.com/watch?v=GJWuVwZO98s

Good stuff.


cool. Nice video, seems like something like that would be good for a game set in sapce, planets connected by warp gates or something.
This is it... the alpaca lips.
Please log in or register to reply.
Live Events Refresh
GSL
09:30
2026 Season 1: Ro8 Group A
Maru vs ClassicLIVE!
IntoTheiNu 859
Ryung 459
CranKy Ducklings SOOP110
herO (SOOP)80
Rex57
GSL EN (SOOP)0
LiquipediaDiscussion
[ Submit Event ]
Live Streams
Refresh
StarCraft 2
Ryung 459
herO (SOOP) 80
Rex 57
StarCraft: Brood War
Britney 54321
Calm 6890
EffOrt 1107
BeSt 1007
Horang2 794
Hyuk 557
actioN 510
Mini 354
Stork 290
Snow 246
[ Show more ]
Soma 231
Soulkey 218
Last 210
Rush 187
ggaemo 167
ZerO 141
Mind 137
Larva 124
Backho 107
Mong 102
hero 99
Hyun 84
Dewaltoss 81
Pusan 79
Killer 59
sorry 53
Aegong 44
Shine 42
Sacsri 41
sSak 32
Bale 29
Barracks 27
soO 20
Icarus 17
Noble 15
Terrorterran 15
GoRush 12
IntoTheRainbow 11
ajuk12(nOOB) 8
Dota 2
qojqva1438
monkeys_forever125
League of Legends
Reynor52
Counter-Strike
olofmeister2631
byalli552
x6flipin434
allub256
edward143
kRYSTAL_44
Super Smash Bros
Westballz15
Other Games
singsing1820
B2W.Neo790
hiko330
Lowko326
ArmadaUGS123
Liquid`VortiX46
DeMusliM28
Liquid`LucifroN15
ZerO(Twitch)9
Organizations
StarCraft: Brood War
UltimateBattle 1241
Dota 2
PGL Dota 2 - Main Stream36
StarCraft 2
Blizzard YouTube
StarCraft: Brood War
BSLTrovo
[ Show 14 non-featured ]
StarCraft 2
• StrangeGG 12
• AfreecaTV YouTube
• intothetv
• Kozan
• IndyKCrew
• LaughNgamezSOOP
• Migwel
• sooper7s
StarCraft: Brood War
• BSLYoutube
• STPLYoutube
• ZZZeroYoutube
Dota 2
• Noizen26
League of Legends
• Jankos1158
Other Games
• WagamamaTV201
Upcoming Events
GSL
20h 47m
SHIN vs Zoun
ByuN vs herO
OSC
22h 17m
OSC
1d
Replay Cast
1d 11h
Escore
1d 21h
The PondCast
1d 21h
WardiTV Invitational
1d 22h
Zoun vs Ryung
Lambo vs ShoWTimE
OSC
2 days
Replay Cast
2 days
CranKy Ducklings
2 days
[ Show More ]
RSL Revival
2 days
SHIN vs Bunny
ByuN vs Shameless
WardiTV Invitational
2 days
Krystianer vs TriGGeR
Cure vs Rogue
uThermal 2v2 Circuit
3 days
BSL
3 days
Replay Cast
3 days
Sparkling Tuna Cup
3 days
RSL Revival
3 days
Cure vs Zoun
Clem vs Lambo
WardiTV Invitational
3 days
BSL
4 days
GSL
4 days
Afreeca Starleague
4 days
Soma vs Leta
Monday Night Weeklies
5 days
CranKy Ducklings
5 days
Afreeca Starleague
5 days
Light vs Flash
Replay Cast
6 days
Liquipedia Results

Completed

Proleague 2026-05-05
WardiTV TLMC #16
Nations Cup 2026

Ongoing

BSL Season 22
ASL Season 21
CSL 2026 SPRING (S20)
IPSL Spring 2026
KCM Race Survival 2026 Season 2
Acropolis #4
YSL S3
SCTL 2026 Spring
RSL Revival: Season 5
2026 GSL S1
BLAST Rivals Spring 2026
IEM Rio 2026
PGL Bucharest 2026
Stake Ranked Episode 1
BLAST Open Spring 2026
ESL Pro League S23 Finals
ESL Pro League S23 Stage 1&2
PGL Cluj-Napoca 2026

Upcoming

Escore Tournament S2: W6
KK 2v2 League Season 1
BSL 22 Non-Korean Championship
Escore Tournament S2: W7
Escore Tournament S2: W8
CSLAN 4
Kung Fu Cup 2026 Grand Finals
HSC XXIX
uThermal 2v2 2026 Main Event
Maestros of the Game 2
2026 GSL S2
Stake Ranked Episode 3
XSE Pro League 2026
IEM Cologne Major 2026
Stake Ranked Episode 2
CS Asia Championships 2026
IEM Atlanta 2026
Asian Champions League 2026
PGL Astana 2026
TLPD

1. ByuN
2. TY
3. Dark
4. Solar
5. Stats
6. Nerchio
7. sOs
8. soO
9. INnoVation
10. Elazer
1. Rain
2. Flash
3. EffOrt
4. Last
5. Bisu
6. Soulkey
7. Mini
8. Sharp
Sidebar Settings...

Advertising | Privacy Policy | Terms Of Use | Contact Us

Original banner artwork: Jim Warren
The contents of this webpage are copyright © 2026 TLnet. All Rights Reserved.