aboutsummaryrefslogtreecommitdiff
path: root/src/shader/code/stream.glsl
blob: a8aec1d9ac5bf6b072a7cca0bf8cc057e6145ffc (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
static const std::string STREAM_SHADER_CODE = R"(
#version 430

layout (local_size_x = 1, local_size_y = 1) in;

layout (std430, binding=1) buffer bufferCollide  { float collideCells[];  };
layout (std430, binding=2) buffer bufferStream   { float streamCells[];   };
layout (std430, binding=4) buffer bufferGeometry { int   materialCells[]; };

/// LBM constants

uniform uint nX;
uniform uint nY;

const uint q = 9;

// Array indexing

uint indexOfDirection(int i, int j) {
	return 3*(i+1) + (j+1);
}

uint indexOfLatticeCell(uint x, uint y) {
	return q*nX*y + q*x;
}

/// Data access

float get(uint x, uint y, int i, int j) {
	return collideCells[indexOfLatticeCell(x,y) + indexOfDirection(i,j)];
}

void set(uint x, uint y, int i, int j, float v) {
	streamCells[indexOfLatticeCell(x,y) + indexOfDirection(i,j)] = v;
}

int getMaterial(uint x, uint y) {
	return materialCells[nX*y + x];
}

/// Domain description

bool isEndOfWorld(uint x, uint y) {
	return x == 0 || x == nX-1 || y == 0 || y == nY-1;
}

bool isOuterWall(uint x, uint y) {
	return x == 1 || x == nX-2 || y == 1 || y == nY-2;
}

/// Boundary conditions

void bounceBack(uint x, uint y) {
	for ( int i = -1; i <= 1; ++i ) {
		for ( int j = -1; j <= 1; ++j ) {
			set(x,y,i,j, get(x,y,(-1)*i,(-1)*j));
		}
	}
}

/// Actual stream kernel

void main() {
	const uint x = gl_GlobalInvocationID.x;
	const uint y = gl_GlobalInvocationID.y;

	if ( !(x < nX && y < nY) ) {
		return;
	}

	const int material = getMaterial(x,y);

	if ( material == 0 ) {
		return;
	}

	if ( material == 2 ) {
		bounceBack(x,y);
	}

	for ( int i = -1; i <= 1; ++i ) {
		for ( int j = -1; j <= 1; ++j ) {
			set(x+i,y+j,i,j, get(x,y,i,j));
		}
	}
}
)";