-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathjasmine-mpeg3-plugin.js
283 lines (249 loc) · 12.1 KB
/
jasmine-mpeg3-plugin.js
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
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
// Jasmine Mpeg3Plugin, we play mp3 and mpg (but use mp4 for mpg)
// empty video frames are sent to Squeak, videos are shown as overlay in the browser
// silent audio is sent to Squeak, we play directly via browser
// this is hardcoded to pretend we play videos at 320x240, 25fps
// and audio at 44100Hz, 1 channel
//
// TODO:
// [ ] actually decode video frames and send them to Squeak
// [ ] actually decode audio samples and send them to Squeak
function JasmineMpeg3Plugin() {
"use strict";
var DEBUG = 1; // 0 = off, 1 = some, 2 = lots
return {
getModuleName: function() { return 'Mpeg3Plugin (jasmine)'; },
interpreterProxy: null,
primHandler: null,
vm: null,
setInterpreter: function(anInterpreter) {
this.interpreterProxy = anInterpreter;
this.vm = this.interpreterProxy.vm;
this.primHandler = this.vm.primHandler;
return true;
},
initialiseModule: function() {
// silent audio
this.audioElement = new Audio("data:audio/wav;base64,UklGRigAAABXQVZFZm10IBIAAAABAAEARKwAAIhYAQACABAAAABkYXRhAgAAAAEA");
// for some reason, timing is more accurate on iOS
// if we play through an audio context
var audioCtx = Squeak.startAudioOut();
var node = audioCtx.createMediaElementSource(this.audioElement);
node.connect(audioCtx.destination);
this.audioElement.play();
// empty video
this.videoElement = document.createElement("video");
},
primitiveMPEG3CheckSig: function(argCount) {
// assume any file is valid
return this.primHandler.popNandPushBoolIfOK(argCount + 1, true);
},
primitiveMPEG3Open: function(argCount) {
var pathObj = this.interpreterProxy.stackObjectValue(0);
var path = pathObj.bytesAsString();
// hack to find mp4 in jasmine directory
path = path.replace(/.*\/jasmine\//, "");
path = path.replace(/\.mpg$/, ".mp4");
var isAudio = path.match(/\.mp3$/);
// We use this.audioElement or this.videoElement which have been
// interacted with by the user, so they are allowed to play
var player;
if (isAudio) {
player = this.audioElement;
player.isAudio = true;
} else {
player = this.videoElement;
player.isVideo = true;
player.playsInline = true; // Safari
document.body.appendChild(player);
}
player.src = path; // start playing
player.currentTime = 0; // start at beginning
player.play();
// create handle
var handle = this.primHandler.makeStString("squeakjs-mpeg:" + path);
handle.player = player;
// freeze VM until player is actually playing
this.vm.freeze(function(unfreeze) {
function continueExecution() {
unfreeze();
unfreeze = null; // don't unfreeze twice
}
player.addEventListener('timeupdate',
function() {
if (player.currentTime < 0.1) return; // not playing yet
if (!unfreeze) return; // already failed or started
DEBUG > 0 && console.log("primitiveMPEG3Open: "
+ (player.isVideo? player.videoWidth + "x" + player.videoHeight + ", " : "")
+ player.duration + "s " + player.src);
if (player.isAudio) {
// we pretend to send samples so Squeak can track time
player.sentSamples = 0;
player.totalSamples = Math.floor(player.duration * 44100);
}
continueExecution();
},
);
player.onerror = function(err) {
if (!unfreeze) return; // too late
console.error("primitiveMPEG3Open: error", err);
continueExecution();
};
}.bind(this));
// the primitive will succeeed now, but then the VM
// will pause execution until unfreeze() is called
this.interpreterProxy.popthenPush(argCount + 1, handle);
return true;
},
primitiveMPEG3Close: function(argCount) {
var player = this.playerFromStackArg(0);
if (!player) return false;
DEBUG > 0 && console.log("primitiveMPEG3Close", player.src);
player.pause();
if (player.isVideo) player.remove();
try {
player.removeAttribute('src');
player.load();
} catch (err) {}
this.interpreterProxy.pop(argCount);
return true;
},
playerFromStackArg: function(n) {
var handle = this.interpreterProxy.stackObjectValue(n);
if (this.interpreterProxy.failed()) return null;
return handle.player;
},
primitiveMPEG3HasAudio: function(argCount) {
var player = this.playerFromStackArg(0);
return this.primHandler.popNandPushBoolIfOK(argCount + 1, player && player.isAudio);
},
primitiveMPEG3HasVideo: function(argCount) {
var player = this.playerFromStackArg(0);
return this.primHandler.popNandPushBoolIfOK(argCount + 1, player && player.isVideo);
},
//////////// VIDEO /////////////
primitiveMPEG3TotalVStreams: function(argCount) {
var player = this.playerFromStackArg(0);
return this.primHandler.popNandPushIntIfOK(argCount + 1, player && player.isVideo ? 1 : 0);
},
primitiveMPEG3FrameRate: function(argCount) {
return this.primHandler.popNandPushIntIfOK(argCount + 1, 25);
},
primitiveMPEG3VideoWidth: function(argCount) {
return this.primHandler.popNandPushIntIfOK(argCount + 1, 320);
},
primitiveMPEG3VideoHeight: function(argCount) {
return this.primHandler.popNandPushIntIfOK(argCount + 1, 240);
},
primitiveMPEG3ReadFrame: function(argCount) {
DEBUG > 1 && console.log("primitiveMPEG3ReadFrame");
this.interpreterProxy.pop(argCount);
return true;
},
primitiveMPEG3GetFrame: function(argCount) {
var player = this.playerFromStackArg(1);
if (!player) return false;
var frame = Math.floor(player.currentTime * 25);
var frames = Math.floor(player.duration * 25);
DEBUG > 1 && console.log("primitiveMPEG3GetFrame", frame, "of", frames);
if (frame >= frames) frame = frames - 1;
return this.primHandler.popNandPushIntIfOK(argCount + 1, frame);
},
primitiveMPEG3VideoFrames: function(argCount) {
var player = this.playerFromStackArg(1);
if (!player) return false;
var frames = Math.floor(player.duration * 25);
DEBUG > 1 && console.log("primitiveMPEG3VideoFrames", frames);
return this.primHandler.popNandPushIntIfOK(argCount + 1, frames);
},
primitiveMPEG3SetFrame: function(argCount) {
var player = this.playerFromStackArg(2);
if (!player) return false;
var frame = this.interpreterProxy.stackIntegerValue(1);
player.currentTime = frame / 25;
DEBUG > 0 && console.log("primitiveMPEG3SetFrame", frame, "(current time now " + player.currentTime.toFixed(1) + ")");
this.interpreterProxy.pop(argCount);
return true;
},
primitiveMPEG3DropFrames: function(argCount) {
// var count = this.interpreterProxy.stackIntegerValue(1);
DEBUG > 1 && console.log("primitiveMPEG3DropFrames", count);
this.interpreterProxy.pop(argCount);
return true;
},
primitiveMPEG3ReadFrameBufferOffset: function(argCount) {
return this.primHandler.popNandPushIntIfOK(argCount + 1, 0);
},
primitiveMPEG3EndOfVideo: function(argCount) {
var player = this.playerFromStackArg(1);
if (!player) return false;
DEBUG > 1 && console.log("primitiveMPEG3EndOfVideo:", player.ended);
return this.primHandler.popNandPushBoolIfOK(argCount + 1, player.ended);
},
//////////// AUDIO /////////////
primitiveMPEG3AudioChannels: function(argCount) {
DEBUG > 1 && console.log("primitiveMPEG3AudioChannels", 1);
// even if it was stereo - we're not sending samples to Squeak
// so it doesn't matter
return this.primHandler.popNandPushIntIfOK(argCount + 1, 1);
},
primitiveMPEG3SampleRate: function(argCount) {
DEBUG > 1 && console.log("primitiveMPEG3SampleRate", 44100);
return this.primHandler.popNandPushIntIfOK(argCount + 1, 44100);
},
primitiveMPEG3SetSample: function(argCount) {
var player = this.playerFromStackArg(2);
if (!player) return false;
var sample = this.interpreterProxy.stackIntegerValue(1);
player.currentTime = sample / 44100;
DEBUG > 0 && console.log("primitiveMPEG3SetSample", sample, "(current time now " + player.currentTime.toFixed(1) + ")");
this.interpreterProxy.pop(argCount);
return true;
},
primitiveMPEG3GetSample: function(argCount) {
var player = this.playerFromStackArg(1);
if (!player) return false;
var sent = player.sentSamples;
var actual = Math.floor(player.currentTime * 44100);
var total = player.totalSamples;
var samples = sent;
if (samples > total) samples = total;
if (actual > total || player.ended) actual = total;
DEBUG > 1 && console.log("primitiveMPEG3GetSample sent:", sent, "/", total, "(todo:", total - sent, "), actual:", actual, "todo:", total - actual, "ended:", player.ended);
return this.primHandler.popNandPushIntIfOK(argCount + 1, samples);
},
primitiveMPEG3AudioSamples: function(argCount) {
var player = this.playerFromStackArg(1);
if (!player) return false;
var total = player.totalSamples;
DEBUG > 1 && console.log("primitiveMPEG3AudioSamples", total);
return this.primHandler.popNandPushIntIfOK(argCount + 1, total);
},
primitiveMPEG3ReadAudio: function(argCount) {
// args: handle, buffer, channel, samples, stream
var player = this.playerFromStackArg(4);
var sending = this.interpreterProxy.stackIntegerValue(1);
// we need to keep track of the number of samples we sent
player.sentSamples += sending;
DEBUG > 1 && console.log("primitiveMPEG3ReadAudio: sending", sending, "=", player.sentSamples, "/", player.totalSamples, "todo:", player.totalSamples - player.sentSamples);
this.interpreterProxy.pop(argCount);
return true;
},
primitiveMPEG3EndOfAudio: function(argCount) {
var player = this.playerFromStackArg(1);
if (!player) return false;
var sent = player.sentSamples;
var actual = Math.floor(player.currentTime * 44100);
var total = player.totalSamples;
var samples = sent > actual ? sent : actual; // in case Squeak is behind
var ended = samples >= total || player.ended;
if (ended) DEBUG > 1 && console.log("primitiveMPEG3EndOfAudio:", ended);
return this.primHandler.popNandPushBoolIfOK(argCount + 1, ended);
},
};
}
function registerMpeg3Plugin() {
if (typeof Squeak === "object" && Squeak.registerExternalModule) {
Squeak.registerExternalModule('Mpeg3Plugin', JasmineMpeg3Plugin());
} else self.setTimeout(registerMpeg3Plugin, 100);
}
registerMpeg3Plugin();