Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
Roblox
GitHub Repository: Roblox/luau
Path: blob/master/bench/tests/sunspider/n-body-oop.lua
2727 views
1
local function prequire(name) local success, result = pcall(require, name); return success and result end
2
local bench = script and require(script.Parent.bench_support) or prequire("bench_support") or require("../../bench_support")
3
4
local PI = 3.141592653589793
5
local SOLAR_MASS = 4 * PI * PI
6
local DAYS_PER_YEAR = 365.24
7
8
local Body = {}
9
Body.__index = Body
10
11
function Body.new(x, y, z, vx, vy, vz, mass)
12
local self = {}
13
self.x = x
14
self.y = y
15
self.z = z
16
self.vx = vx
17
self.vy = vy
18
self.vz = vz
19
self.mass = mass
20
return setmetatable(self, Body)
21
end
22
23
function Body:offsetMomentum(px, py, pz)
24
self.vx = -px / SOLAR_MASS
25
self.vy = -py / SOLAR_MASS
26
self.vz = -pz / SOLAR_MASS
27
28
return self
29
end
30
31
local function Jupiter()
32
return Body.new(
33
4.841431442464721e0,
34
-1.1603200440274284e0,
35
-1.036220444711231e-1,
36
1.660076642744037e-3 * DAYS_PER_YEAR,
37
7.6990111841974045e-3 * DAYS_PER_YEAR,
38
-6.90460016972063e-5 * DAYS_PER_YEAR,
39
9.547919384243267e-4 * SOLAR_MASS
40
)
41
end
42
local function Saturn()
43
return Body.new(8.34336671824458e0, 4.124798564124305e0, -4.035234171143213e-1, -2.767425107268624e-3 * DAYS_PER_YEAR, 4.998528012349173e-3 * DAYS_PER_YEAR, 2.3041729757376395e-5 * DAYS_PER_YEAR, 2.8588598066613082e-4 * SOLAR_MASS)
44
end
45
local function Uranus()
46
return Body.new(1.2894369562139132e1, -1.511115140169863e1, -2.2330757889265573e-1, 2.964601375647616e-3 * DAYS_PER_YEAR, 2.3784717395948096e-3 * DAYS_PER_YEAR, -2.9658956854023755e-5 * DAYS_PER_YEAR, 4.366244043351563e-5 * SOLAR_MASS)
47
end
48
local function Neptune()
49
return Body.new(1.5379697114850917e1, -2.5919314609987962e1, 1.7925877295037118e-1, 2.680677724903893e-3 * DAYS_PER_YEAR, 1.628241700382423e-3 * DAYS_PER_YEAR, -9.515922545197158e-5 * DAYS_PER_YEAR, 5.151389020466114e-5 * SOLAR_MASS)
50
end
51
local function Sun()
52
return Body.new(0, 0, 0, 0, 0, 0, SOLAR_MASS)
53
end
54
55
local NBodySystem = {}
56
NBodySystem.__index = NBodySystem
57
58
59
function NBodySystem.new(bodies)
60
local self = {}
61
self.bodies = bodies
62
63
local px = 0
64
local py = 0
65
local pz = 0
66
local size = #self.bodies
67
68
for i=1, size do
69
local b = self.bodies[i]
70
local m = b.mass
71
72
px = px + b.vx * m
73
py = py + b.vy * m
74
pz = pz + b.vz * m
75
end
76
77
self.bodies[1]:offsetMomentum(px, py, pz)
78
79
return setmetatable(self, NBodySystem)
80
end
81
82
function NBodySystem:advance(dt)
83
local dx, dy, dz, distance, mag
84
local size = #self.bodies
85
86
for i=1, size do
87
local bodyi = self.bodies[i]
88
for j=i+1, size do
89
local bodyj = self.bodies[j]
90
dx = bodyi.x - bodyj.x
91
dy = bodyi.y - bodyj.y
92
dz = bodyi.z - bodyj.z
93
94
distance = math.sqrt(dx*dx + dy*dy + dz*dz)
95
mag = dt / (distance * distance * distance)
96
97
bodyi.vx -= dx * bodyj.mass * mag
98
bodyi.vy -= dy * bodyj.mass * mag
99
bodyi.vz -= dz * bodyj.mass * mag
100
101
bodyj.vx += dx * bodyi.mass * mag
102
bodyj.vy += dy * bodyi.mass * mag
103
bodyj.vz += dz * bodyi.mass * mag
104
end
105
end
106
for i=1, size do
107
local body = self.bodies[i]
108
109
body.x = body.x + dt * body.vx
110
body.y = body.y + dt * body.vy
111
body.z = body.z + dt * body.vz
112
end
113
end
114
115
function NBodySystem:energy()
116
local dx, dy, dz, distance
117
local e = 0.0
118
local size = #self.bodies
119
120
for i=1, size do
121
local bodyi = self.bodies[i]
122
123
e = e + 0.5 * bodyi.mass * (bodyi.vx * bodyi.vx + bodyi.vy * bodyi.vy + bodyi.vz * bodyi.vz)
124
125
for j=i+1, size do
126
local bodyj = self.bodies[j]
127
dx = bodyi.x - bodyj.x
128
dy = bodyi.y - bodyj.y
129
dz = bodyi.z - bodyj.z
130
131
distance = math.sqrt(dx*dx + dy*dy + dz*dz)
132
e -= (bodyi.mass * bodyj.mass) / distance
133
end
134
end
135
136
return e
137
end
138
139
local function run()
140
local ret = 0
141
local n = 3
142
while n <= 24 do
143
(function()
144
local bodies = NBodySystem.new({
145
Sun(),Jupiter(),Saturn(),Uranus(),Neptune()
146
})
147
local max = n * 100
148
149
ret += bodies:energy()
150
for i=1, max do
151
bodies:advance(0.01)
152
end
153
ret += bodies:energy()
154
end)()
155
n *= 2
156
end
157
local expected = -1.3524862408537381
158
159
if ret ~= expected then
160
error('ERROR: bad result: expected ' .. expected .. ' but got ' .. ret)
161
end
162
end
163
164
function runIteration()
165
for i=1, 5 do
166
run()
167
end
168
end
169
170
bench.runCode(runIteration, "n-body-oop")
171
172