Skip to content

Commit f0ed079

Browse files
authoredNov 20, 2017
Unravel (codecombat#4544)
* PlayGameDevLevelView running world in main thread for better performance This reduces memory and CPU usage, improves smoothness of playback, and makes keyboard and mouse input more responsive. It would be close to being able to do test mode in main thread, too, if we wanted. With a little more work, we would then have no worker threads at all in game dev mode. Random misc half of fix for loading keyValueDb while playing game dev levels. Made ?dev=true work for PlayGameDevLevelView. Would need to think about if we always want Box2D loaded on the main thread. Would need to standardize on @world.rand.shuffle instead of _.shuffle to use this in maze levels. Math.random() and _.shuffle in player code may need a fix to make deterministic, not that we teach those yet. There may be a slight memory leak of old worlds yet to clean up after level restarts. If we can figure out how to clean up LankBoss Thang stateChanged updates, we'll gain even more performance. Unrelated to this change, I noticed that our EaselJS click handler gets really unperformant with many sprites on the screen, so for best speed, we'll want to fix that, too. * Fix some misc stuff * Add world.rand.shuffleCompat to migrate from using _.shuffle in game engine * Update CreateJS from somewhere around 0.8 (Nov 2014) to 1.0 TODO: get Sounds working all the way TODO: take advantage of new StageGL instead of old SpriteStage, where we can now use more than one texture See createjs.js line 7735 monkeypatching of hit testing for performance increases that we'll want to make sure to preserve in future updates * Other updates to go along with CreateJS update * Fix a couple SoundJS and test issues with the CreateJS update * Try using main thread game dev mode for testing levels, too. TODO: investigate whether memory is leaking. * Removed test for one-spritesheet-per-container (restriction not needed in new StageGL) * Fixed the memory leak * Minor cleanup * Fix sounds in synchronous game dev * Fix bug in optimized click hit test in level editor * Fix game dev levels running out of time when synchronous and indefinite * Disable rerendering of spritesheets for performance with new StageGL * Fix rendering spritesheets syncronously blowing up max spritesheet size * Don't consolidate walls when simulating synchronously to avoid messing with graphics * Fix wall display during game dev levels * Don't go crazy on game-dev particles in campaign map * Clean up some unused code
1 parent c763e21 commit f0ed079

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

41 files changed

+31000
-27140
lines changed
 

‎app/lib/Angel.coffee

+80-42
Original file line numberDiff line numberDiff line change
@@ -149,7 +149,7 @@ module.exports = class Angel extends CocoClass
149149
finishBeholdingWorld: (goalStates) -> (world) =>
150150
return if @aborting or @destroyed
151151
finished = world.frames.length is world.totalFrames
152-
if @work?.indefiniteLength and world.victory?
152+
if world.indefiniteLength and world.victory?
153153
finished = true
154154
world.totalFrames = world.frames.length
155155
firstChangedFrame = if @work?.indefiniteLength then 0 else world.findFirstChangedFrame @shared.world
@@ -163,8 +163,8 @@ module.exports = class Angel extends CocoClass
163163
@shared.goalManager?.world = world
164164
@finishWork()
165165
else
166-
@deserializationQueue.shift() # Finished with this deserialization.
167-
if deserializationArgs = @deserializationQueue[0] # Start another?
166+
@deserializationQueue?.shift() # Finished with this deserialization.
167+
if deserializationArgs = @deserializationQueue?[0] # Start another?
168168
@beholdWorld deserializationArgs...
169169

170170
finishWork: ->
@@ -218,11 +218,11 @@ module.exports = class Angel extends CocoClass
218218
return @say 'Not initialized for work yet.' unless @initialized
219219
if @shared.workQueue.length
220220
@work = @shared.workQueue.shift()
221-
return _.defer @simulateSync, @work if @work.synchronous
222221
@say 'Running world...'
223222
@running = true
224223
@shared.busyAngels.push @
225224
@deserializationQueue = []
225+
return _.defer @simulateSync, @work if @work.synchronous
226226
@worker.postMessage func: 'runWorld', args: @work
227227
clearTimeout @purgatoryTimer
228228
@say 'Infinite loop timer started at interval of', @infiniteLoopIntervalDuration
@@ -232,16 +232,19 @@ module.exports = class Angel extends CocoClass
232232
@hireWorker()
233233

234234
abort: ->
235-
return unless @worker and @running
235+
return unless @running
236236
@say 'Aborting...'
237237
@running = false
238+
@work?.world?.goalManager?.destroy()
239+
@work?.aborted = true
238240
@work = null
239241
@streamingWorld = null
240242
@deserializationQueue = []
241243
_.remove @shared.busyAngels, @
242-
@abortTimeout = _.delay @fireWorker, @abortTimeoutDuration
243-
@aborting = true
244-
@worker.postMessage func: 'abort'
244+
if @worker
245+
@abortTimeout = _.delay @fireWorker, @abortTimeoutDuration
246+
@aborting = true
247+
@worker.postMessage func: 'abort'
245248

246249
fireWorker: (rehire=true) =>
247250
return if @destroyed
@@ -255,6 +258,7 @@ module.exports = class Angel extends CocoClass
255258
clearInterval @purgatoryTimer
256259
@say 'Fired worker.'
257260
@initialized = false
261+
@work?.world?.destroy?()
258262
@work = null
259263
@streamingWorld = null
260264
@deserializationQueue = []
@@ -273,20 +277,29 @@ module.exports = class Angel extends CocoClass
273277
@worker.addEventListener 'message', @onWorkerMessage
274278
@worker.creationTime = new Date()
275279

276-
onFlagEvent: (e) ->
280+
onFlagEvent: (flagEvent) ->
277281
return unless @running and @work.realTime
278-
@worker.postMessage func: 'addFlagEvent', args: e
282+
if @work.synchronous
283+
@work.world.addFlagEvent flagEvent
284+
else
285+
@worker.postMessage func: 'addFlagEvent', args: flagEvent
279286

280287
onAddRealTimeInputEvent: (realTimeInputEvent) ->
281288
return unless @running and @work.realTime
282-
@worker.postMessage func: 'addRealTimeInputEvent', args: realTimeInputEvent.toJSON()
289+
if @work.synchronous
290+
@work.world.addRealTimeInputEvent realTimeInputEvent.toJSON()
291+
else
292+
@worker.postMessage func: 'addRealTimeInputEvent', args: realTimeInputEvent.toJSON()
283293

284294
onStopRealTimePlayback: (e) ->
285295
return unless @running and @work.realTime
296+
if @work.synchronous
297+
return @abort()
286298
@work.realTime = false
287299
@lastRealTimeWork = new Date()
288300
@worker.postMessage func: 'stopRealTimePlayback'
289301

302+
290303
onEscapePressed: (e) ->
291304
return unless @running and not @work.realTime
292305
return if (new Date() - @lastRealTimeWork) < 1000 # Fires right after onStopRealTimePlayback
@@ -296,51 +309,76 @@ module.exports = class Angel extends CocoClass
296309
simulateSync: (work) =>
297310
console?.profile? "World Generation #{(Math.random() * 1000).toFixed(0)}" if imitateIE9?
298311
work.t0 = now()
299-
work.world = testWorld = new World work.userCodeMap
312+
work.world = new World work.userCodeMap
313+
work.world.synchronous = true
300314
work.world.levelSessionIDs = work.levelSessionIDs
301315
work.world.submissionCount = work.submissionCount
302316
work.world.fixedSeed = work.fixedSeed
303317
work.world.flagHistory = work.flagHistory ? []
304-
work.world.difficulty = work.difficulty
305-
work.world.loadFromLevel work.level
318+
work.world.realTimeInputEvents = work.realTimeInputEvents ? []
319+
work.world.difficulty = work.difficulty ? 0
320+
work.world.loadFromLevel work.level, true
306321
work.world.preloading = work.preload
307322
work.world.headless = work.headless
308323
work.world.realTime = work.realTime
324+
work.world.indefiniteLength = work.indefiniteLength;
325+
work.world.justBegin = work.justBegin;
326+
work.world.keyValueDb = work.keyValueDb;
309327
if @shared.goalManager
310-
testGM = new GoalManager(testWorld)
311-
testGM.setGoals work.goals
312-
testGM.setCode work.userCodeMap
313-
testGM.worldGenerationWillBegin()
314-
testWorld.setGoalManager testGM
315-
@doSimulateWorld work
328+
goalManager = new GoalManager(work.world)
329+
goalManager.setGoals work.goals
330+
goalManager.setCode work.userCodeMap
331+
goalManager.worldGenerationWillBegin()
332+
work.world.setGoalManager goalManager
333+
@beginSimulationSync work
334+
335+
beginSimulationSync: (work) ->
336+
work.t1 = now()
337+
work.world.worldLoadStartTime = work.t1
338+
work.world.lastRealTimeUpdate = 0
339+
work.world.realTimeSpeedFactor = 1
340+
@simulateFramesSync work, 0
341+
342+
simulateFramesSync: (work, i) ->
343+
return if @destroyed or work.aborted
344+
world = work.world
345+
i ?= world.frames.length
346+
simulationLoopStartTime = now()
347+
while i < world.totalFrames
348+
if work.realTime
349+
progress = world.frames.length / world.totalFrames
350+
progress = Math.min(progress, 0.9) if world.indefiniteLength
351+
@publishGodEvent 'world-load-progress-changed', progress: progress # Debounce? Need to publish at all?
352+
@streamFrameSync work
353+
if world.indefiniteLength and world.victory?
354+
world.indefiniteLength = false
355+
continuing = world.shouldContinueLoading simulationLoopStartTime, (->), false, (=> @simulateFramesSync(work) unless @destroyed)
356+
return unless continuing
357+
if world.indefiniteLength and i is world.totalFrames - 1
358+
++world.totalFrames
359+
frame = world.getFrame i++ # TODO: better handle non-user-code errors and infinite-loops
360+
@finishSimulationSync work
361+
362+
streamFrameSync: (work) ->
363+
goalStates = work.world.goalManager.getGoalStates()
364+
@finishBeholdingWorld(goalStates)(work.world)
365+
366+
finishSimulationSync: (work) ->
367+
@publishGodEvent 'world-load-progress-changed', progress: 1
368+
work.world.ended = true
369+
system.finish work.world.thangs for system in work.world.systems
370+
work.t2 = now()
316371
console?.profileEnd?() if imitateIE9?
317-
console.log 'Construction:', (work.t1 - work.t0).toFixed(0), 'ms. Simulation:', (work.t2 - work.t1).toFixed(0), 'ms --', ((work.t2 - work.t1) / testWorld.frames.length).toFixed(3), 'ms per frame, profiled.'
372+
console.log 'Construction:', (work.t1 - work.t0).toFixed(0), 'ms. Simulation:', (work.t2 - work.t1).toFixed(0), 'ms --', ((work.t2 - work.t1) / work.world.frames.length).toFixed(3), 'ms per frame, profiled.'
318373

319374
# If performance was really a priority in IE9, we would rework things to be able to skip this step.
320-
goalStates = testGM?.getGoalStates()
375+
goalStates = work.world.goalManager?.getGoalStates()
321376
work.world.goalManager.worldGenerationEnded() if work.world.ended
377+
@running = false
322378

323379
if work.headless
324380
simulationFrameRate = work.world.frames.length / (work.t2 - work.t1) * 1000 * 30 / work.world.frameRate
325-
@beholdGoalStates {goalStates, overallStatus: testGM.checkOverallStatus(), preload: false, totalFrames: work.world.totalFrames, lastFrameHash: work.world.frames[work.world.totalFrames - 2]?.hash, simulationFrameRate: simulationFrameRate}
381+
@beholdGoalStates {goalStates, overallStatus: work.world.goalManager.checkOverallStatus(), preload: false, totalFrames: work.world.totalFrames, lastFrameHash: work.world.frames[work.world.totalFrames - 2]?.hash, simulationFrameRate: simulationFrameRate}
326382
return
327383

328-
serialized = world.serialize()
329-
window.BOX2D_ENABLED = false
330-
World.deserialize serialized.serializedWorld, @shared.worldClassMap, @shared.lastSerializedWorldFrames, @finishBeholdingWorld(goalStates), serialized.startFrame, serialized.endFrame, work.level
331-
window.BOX2D_ENABLED = true
332-
@shared.lastSerializedWorldFrames = serialized.serializedWorld.frames
333-
334-
doSimulateWorld: (work) ->
335-
work.t1 = now()
336-
Math.random = work.world.rand.randf # so user code is predictable
337-
Aether.replaceBuiltin('Math', Math)
338-
replacedLoDash = _.runInContext(window)
339-
_[key] = replacedLoDash[key] for key, val of replacedLoDash
340-
i = 0
341-
while i < work.world.totalFrames
342-
frame = work.world.getFrame i++
343-
@publishGodEvent 'world-load-progress-changed', progress: 1
344-
work.world.ended = true
345-
system.finish work.world.thangs for system in work.world.systems
346-
work.t2 = now()
384+
@shared.lastSerializedWorldFrames = work.world.frames

‎app/lib/AudioPlayer.coffee

+2-7
Original file line numberDiff line numberDiff line change
@@ -8,12 +8,7 @@ createjs = require 'lib/createjs-parts'
88
rot13 = (s) -> s.replace /[A-z]/g, (c) -> String.fromCharCode c.charCodeAt(0) + (if c.toUpperCase() <= 'M' then 13 else -13)
99
swears = (rot13 s for s in ['nefrubyr', 'nffubyr', 'onfgneq', 'ovgpu', 'oybbql', 'obyybpxf', 'ohttre', 'pbpx', 'penc', 'phag', 'qnza', 'qnea', 'qvpx', 'qbhpur', 'snt', 'shpx', 'cvff', 'chffl', 'fuvg', 'fyhg', 'svqqyrfgvpxf'])
1010

11-
# IE(11, 10, 9) throws an exception if createjs.FlashPlugin is undefined
12-
# Chrome and Firefox don't seem to care that it's undefined
13-
if createjs.FlashPlugin?
14-
soundPlugins = [createjs.WebAudioPlugin, createjs.FlashPlugin, createjs.HTMLAudioPlugin]
15-
else
16-
soundPlugins = [createjs.WebAudioPlugin, createjs.HTMLAudioPlugin]
11+
soundPlugins = [createjs.WebAudioPlugin, createjs.HTMLAudioPlugin]
1712
createjs.Sound.registerPlugins(soundPlugins)
1813

1914
class Manifest
@@ -44,7 +39,7 @@ class AudioPlayer extends CocoClass
4439

4540
constructor: () ->
4641
super()
47-
@ext = if createjs.Sound.getCapability('mp3') then '.mp3' else '.ogg'
42+
@ext = if createjs.Sound.capabilities.mp3 then '.mp3' else '.ogg'
4843
@camera = null
4944
@listenToSound()
5045
@createNewManifest()

0 commit comments

Comments
 (0)
Please sign in to comment.