Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
Roblox
GitHub Repository: Roblox/luau
Path: blob/master/bench/tests/mesh-normal-vector.lua
2725 views
1
--!strict
2
local function prequire(name) local success, result = pcall(require, name); return success and result end
3
local bench = script and require(script.Parent.bench_support) or prequire("bench_support") or require("../bench_support")
4
5
function test()
6
7
type Vertex = { p: vector, uv: vector, n: vector, t: vector, b: vector, h: number }
8
9
local grid_size = 100
10
11
local mesh: {
12
vertices: {Vertex},
13
indices: {number},
14
triangle_cone_p: {vector},
15
triangle_cone_n: {vector}
16
} = {
17
vertices = table.create(grid_size * grid_size),
18
indices = table.create((grid_size - 1) * (grid_size - 1) * 6),
19
triangle_cone_p = table.create((grid_size - 1) * (grid_size - 1) * 2),
20
triangle_cone_n = table.create((grid_size - 1) * (grid_size - 1) * 2)
21
}
22
23
function init_vertices()
24
local i = 1
25
for y = 1,grid_size do
26
for x = 1,grid_size do
27
local v: Vertex = {}
28
29
v.p = vector.create(x, y, math.cos(x) + math.sin(y))
30
v.uv = vector.create((x-1)/(grid_size-1), (y-1)/(grid_size-1), 0)
31
v.n = vector.create(0, 0, 0)
32
v.b = vector.create(0, 0, 0)
33
v.t = vector.create(0, 0, 0)
34
v.h = 0
35
36
mesh.vertices[i] = v
37
i += 1
38
end
39
end
40
end
41
42
function init_indices()
43
local i = 1
44
for y = 1,grid_size-1 do
45
for x = 1,grid_size-1 do
46
mesh.indices[i] = x + (y-1)*grid_size
47
i += 1
48
mesh.indices[i] = x + y*grid_size
49
i += 1
50
mesh.indices[i] = (x+1) + (y-1)*grid_size
51
i += 1
52
mesh.indices[i] = (x+1) + (y-1)*grid_size
53
i += 1
54
mesh.indices[i] = x + y*grid_size
55
i += 1
56
mesh.indices[i] = (x+1) + y*grid_size
57
i += 1
58
end
59
end
60
end
61
62
function calculate_normals()
63
local norm_sum = 0
64
65
for i = 1,#mesh.indices,3 do
66
local a = mesh.vertices[mesh.indices[i]]
67
local b = mesh.vertices[mesh.indices[i + 1]]
68
local c = mesh.vertices[mesh.indices[i + 2]]
69
70
local n = vector.cross(a.p - b.p, a.p - c.p)
71
72
a.n += n
73
b.n += n
74
c.n += n
75
end
76
77
for _,v in ipairs(mesh.vertices) do
78
v.n = vector.normalize(v.n)
79
80
norm_sum += vector.dot(v.n, v.n)
81
end
82
83
return norm_sum
84
end
85
86
function compute_triangle_cones()
87
local mesh_area = 0
88
89
local pos = 1
90
91
for i = 1,#mesh.indices,3 do
92
local p0 = mesh.vertices[mesh.indices[i]]
93
local p1 = mesh.vertices[mesh.indices[i + 1]]
94
local p2 = mesh.vertices[mesh.indices[i + 2]]
95
96
local p10 = p1.p - p0.p
97
local p20 = p2.p - p0.p
98
99
local normal = vector.cross(p10, p20)
100
101
local area = vector.magnitude(normal)
102
local invarea = (area == 0) and 0 or 1 / area;
103
104
mesh.triangle_cone_p[pos] = (p0.p + p1.p + p2.p) / 3
105
mesh.triangle_cone_n[pos] = normal * invarea
106
pos += 1
107
108
mesh_area += area
109
end
110
111
return mesh_area
112
end
113
114
function compute_tangent_space()
115
local checksum = 0
116
117
for i = 1,#mesh.indices,3 do
118
local a = mesh.vertices[mesh.indices[i]]
119
local b = mesh.vertices[mesh.indices[i + 1]]
120
local c = mesh.vertices[mesh.indices[i + 2]]
121
122
local vba = b.p - a.p
123
local vca = c.p - a.p
124
125
local uvba = b.uv - a.uv
126
local uvca = c.uv - a.uv
127
128
local r = 1.0 / (uvba.X * uvca.Y - uvca.X * uvba.Y);
129
130
local sdir = (uvca.Y * vba - uvba.Y * vca) * r
131
local tdir = (uvba.X * vca - uvca.X * vba) * r
132
133
a.t += sdir
134
b.t += sdir
135
c.t += sdir
136
137
a.b += tdir
138
b.b += tdir
139
c.b += tdir
140
end
141
142
for _,v in ipairs(mesh.vertices) do
143
local t = v.t
144
145
-- Gram-Schmidt orthogonalize
146
v.t = vector.normalize(t - v.n * vector.dot(v.n, t))
147
148
local ht = vector.dot(vector.cross(v.n, t), v.b)
149
150
v.h = ht < 0 and -1 or 1
151
152
checksum += v.t.X + v.h
153
end
154
155
return checksum
156
end
157
158
159
init_vertices()
160
init_indices()
161
calculate_normals()
162
compute_triangle_cones()
163
compute_tangent_space()
164
end
165
166
bench.runCode(test, "mesh-normal-vector")
167
168