Mercurial > mm7
comparison Media/MediaPlayer.cpp @ 2502:a77c34acdbc9
Media folder
author | Ritor1 |
---|---|
date | Fri, 19 Sep 2014 05:13:32 +0600 |
parents | |
children | 6b8c2f8c7fad |
comparison
equal
deleted
inserted
replaced
2501:0ff6a9e9bf7f | 2502:a77c34acdbc9 |
---|---|
1 #define _CRTDBG_MAP_ALLOC | |
2 #include <stdlib.h> | |
3 #include <crtdbg.h> | |
4 | |
5 #include <vector> | |
6 #include <deque> | |
7 | |
8 #define _CRT_SECURE_NO_WARNINGS | |
9 | |
10 | |
11 #include "IO/Mouse.h" | |
12 #include "GUI/GUIWindow.h" | |
13 #include "Engine/mm7_data.h" | |
14 #include "Media/Audio/OpenALSoundProvider.h" | |
15 #include "Engine/Log.h" | |
16 #include "MediaPlayer.h" | |
17 #include "Media/Video/Bink_Smacker.h" | |
18 #include "Media/Audio/AudioPlayer.h" | |
19 #include "Engine/Timer.h" | |
20 #include "Engine/Graphics/Render.h" | |
21 #include "Engine/Game.h" | |
22 | |
23 | |
24 #pragma comment(lib, "Version.lib") | |
25 | |
26 using namespace Media; | |
27 | |
28 Media::MPlayer *pMediaPlayer = nullptr; | |
29 Media::IMovie *pMovie_Track; | |
30 Media::ITrack *pAudio_Track; | |
31 Movie *movie; | |
32 | |
33 int mSourceID; | |
34 | |
35 void PlayMovie(const wchar_t * pFilename); | |
36 void PlayAudio(const wchar_t * pFilename); | |
37 void LoadMovie(const char *); | |
38 | |
39 class MemoryStream | |
40 { | |
41 public: | |
42 inline MemoryStream(void *data, size_t data_size) | |
43 { | |
44 this->data_size = data_size; | |
45 this->data = data; | |
46 this->current_pos = 0; | |
47 } | |
48 inline MemoryStream() | |
49 { | |
50 this->data_size = 0; | |
51 this->data = nullptr; | |
52 this->current_pos = 0; | |
53 } | |
54 | |
55 inline ~MemoryStream() | |
56 { | |
57 //Log::Warning(L"Destructor: data delete %u", data); | |
58 if (data) | |
59 delete [] data; | |
60 } | |
61 | |
62 inline size_t Write(void *buffer, size_t num_bytes) | |
63 { | |
64 if (!data) | |
65 { | |
66 data_size = 32 + num_bytes; | |
67 data = new char[data_size]; | |
68 //Log::Warning(L"new data %u", data); | |
69 current_pos = 0; | |
70 } | |
71 else if (current_pos + num_bytes >= data_size) | |
72 { | |
73 int new_data_size = data_size + num_bytes + data_size / 8 + 4; | |
74 auto new_data = new char[new_data_size]; | |
75 //Log::Warning(L"new new_data %u", new_data); | |
76 | |
77 memcpy(new_data, data, data_size); | |
78 //Log::Warning(L"data delete %u", data); | |
79 delete [] data; | |
80 | |
81 data_size = new_data_size; | |
82 data = new_data; | |
83 } | |
84 memcpy((char *)data + current_pos, buffer, num_bytes); | |
85 current_pos += num_bytes; | |
86 return num_bytes; | |
87 } | |
88 | |
89 inline size_t Read(void *buffer, size_t num_bytes) | |
90 { | |
91 size_t read_size = min(num_bytes, data_size - current_pos); | |
92 if (read_size) | |
93 { | |
94 memcpy(buffer, (char *)data + current_pos, read_size); | |
95 current_pos += read_size; | |
96 } | |
97 return read_size; | |
98 } | |
99 | |
100 inline void Reset() | |
101 { | |
102 current_pos = 0; | |
103 } | |
104 inline void SeekToEnd() | |
105 { | |
106 current_pos = data_size; | |
107 } | |
108 | |
109 inline size_t Unwind(size_t bytes) | |
110 { | |
111 if (bytes > current_pos) | |
112 current_pos = 0; | |
113 else | |
114 current_pos -= bytes; | |
115 return current_pos; | |
116 } | |
117 | |
118 inline size_t Rewind(size_t bytes) | |
119 { | |
120 if (current_pos + bytes >= data_size) | |
121 current_pos = data_size; | |
122 else | |
123 current_pos += bytes; | |
124 return current_pos; | |
125 } | |
126 | |
127 inline size_t Size() const {return data_size;} | |
128 inline size_t Current() const {return current_pos;} | |
129 inline void *Ptr() const {return data;} | |
130 | |
131 private: | |
132 void *data; | |
133 size_t data_size; | |
134 size_t current_pos; | |
135 }; | |
136 | |
137 OpenALSoundProvider *provider = nullptr; | |
138 | |
139 static int av_num_bytes_per_sample(AVSampleFormat sample_fmt) | |
140 { | |
141 switch (sample_fmt) | |
142 { | |
143 case AV_SAMPLE_FMT_U8: | |
144 case AV_SAMPLE_FMT_U8P: | |
145 return 1; | |
146 | |
147 case AV_SAMPLE_FMT_S16: | |
148 case AV_SAMPLE_FMT_S16P: | |
149 return 2; | |
150 | |
151 case AV_SAMPLE_FMT_S32: | |
152 case AV_SAMPLE_FMT_S32P: | |
153 case AV_SAMPLE_FMT_FLT: | |
154 case AV_SAMPLE_FMT_FLTP: | |
155 return 4; | |
156 | |
157 case AV_SAMPLE_FMT_DBL: | |
158 case AV_SAMPLE_FMT_DBLP: | |
159 return 8; | |
160 | |
161 default: | |
162 case AV_SAMPLE_FMT_NONE: | |
163 Error("Invalid av sample format: %u", sample_fmt); | |
164 } | |
165 return 0; | |
166 } | |
167 | |
168 struct AVStreamWrapper | |
169 { | |
170 inline AVStreamWrapper() | |
171 { | |
172 this->type = AVMEDIA_TYPE_UNKNOWN; | |
173 this->stream_idx = -1; | |
174 this->stream = nullptr; | |
175 this->dec = nullptr; | |
176 this->dec_ctx = nullptr; | |
177 } | |
178 | |
179 inline void Release() | |
180 { | |
181 type = AVMEDIA_TYPE_UNKNOWN; | |
182 stream_idx = -1; | |
183 stream = nullptr; | |
184 dec = nullptr; | |
185 if (dec_ctx) | |
186 { | |
187 // Close the codec | |
188 avcodec_close(dec_ctx); | |
189 Log::Warning(L"close decoder context file\n"); | |
190 dec_ctx = nullptr; | |
191 } | |
192 } | |
193 | |
194 AVMediaType type; | |
195 int stream_idx; | |
196 AVStream *stream; | |
197 AVCodec *dec; | |
198 AVCodecContext *dec_ctx; | |
199 }; | |
200 | |
201 struct AVAudioStream: public AVStreamWrapper | |
202 { | |
203 inline AVAudioStream():AVStreamWrapper() | |
204 { | |
205 this->bytes_per_sample = 0; | |
206 this->bytes_per_second = 0; | |
207 } | |
208 | |
209 int bytes_per_sample; | |
210 int bytes_per_second; | |
211 }; | |
212 | |
213 struct AVVideoStream: public AVStreamWrapper | |
214 { | |
215 inline AVVideoStream(): AVStreamWrapper() | |
216 { | |
217 this->frames_per_second = 0.0; | |
218 } | |
219 | |
220 double frames_per_second; | |
221 }; | |
222 | |
223 static bool av_open_stream(AVFormatContext *format_ctx, AVMediaType type, AVStreamWrapper *out_stream) | |
224 { | |
225 int stream_idx = av_find_best_stream(format_ctx, type, -1, -1, nullptr, 0); | |
226 if (stream_idx >= 0) | |
227 { | |
228 auto stream = format_ctx->streams[stream_idx]; | |
229 auto dec_ctx = stream->codec; | |
230 | |
231 // Find the decoder for the video stream | |
232 auto dec = avcodec_find_decoder(dec_ctx->codec_id); | |
233 if (dec) | |
234 { | |
235 // Open codec | |
236 if (avcodec_open2(dec_ctx, dec, nullptr) >= 0) | |
237 { | |
238 out_stream->type = type; | |
239 out_stream->stream_idx = stream_idx; | |
240 out_stream->stream = stream; | |
241 out_stream->dec = dec; | |
242 out_stream->dec_ctx = dec_ctx; | |
243 return true; | |
244 } | |
245 fprintf(stderr, "ffmpeg: Could not open codec\n"); | |
246 return false; | |
247 } | |
248 fprintf(stderr, "ffmpeg: Unable to open codec\n"); | |
249 return false; | |
250 } | |
251 fprintf(stderr, "ffmpeg: Didn't find a stream\n"); | |
252 return false; | |
253 } | |
254 | |
255 static bool av_open_audio_stream(AVFormatContext *format_ctx, AVAudioStream *out_stream) | |
256 { | |
257 if (!av_open_stream(format_ctx, AVMEDIA_TYPE_AUDIO, out_stream)) | |
258 return Error("Audio stream not found"), false; | |
259 | |
260 // we support only 2-channel audio for now | |
261 //if (out_stream->dec_ctx->channels != 2) | |
262 //{ | |
263 // out_stream->Release(); | |
264 // return Error("Unsupported number of channels: %u", out_stream->dec_ctx->channels), false; | |
265 //} | |
266 | |
267 out_stream->bytes_per_sample = av_num_bytes_per_sample(out_stream->dec_ctx->sample_fmt); | |
268 out_stream->bytes_per_second = out_stream->dec_ctx->channels * out_stream->dec_ctx->sample_rate * out_stream->bytes_per_sample; | |
269 | |
270 return true; | |
271 } | |
272 | |
273 static bool av_open_video_stream(AVFormatContext *format_ctx, AVVideoStream *out_stream) | |
274 { | |
275 if (!av_open_stream(format_ctx, AVMEDIA_TYPE_VIDEO, out_stream)) | |
276 return Error("Video stream not found"), false; | |
277 | |
278 out_stream->frames_per_second = (double)out_stream->dec_ctx->time_base.den / (double)out_stream->dec_ctx->time_base.num; | |
279 return true; | |
280 } | |
281 | |
282 void InterleaveAudioData(MemoryStream *stream, AVSampleFormat src_format, int num_channels, int num_samples, uint8_t **channels) | |
283 { | |
284 unsigned int bytes_per_sample; | |
285 switch (src_format) | |
286 { | |
287 case AV_SAMPLE_FMT_U8: | |
288 case AV_SAMPLE_FMT_U8P: | |
289 __debugbreak(); | |
290 | |
291 case AV_SAMPLE_FMT_S16: | |
292 bytes_per_sample = sizeof(__int16); | |
293 stream->Write(channels[0], num_channels * num_samples * bytes_per_sample); | |
294 break; | |
295 | |
296 case AV_SAMPLE_FMT_S16P: | |
297 { | |
298 bytes_per_sample = sizeof(__int16); | |
299 for (int i = 0; i < num_samples; ++i) | |
300 for (int j = 0; j < num_channels; ++j) | |
301 stream->Write(channels[j] + i * bytes_per_sample, bytes_per_sample); | |
302 } | |
303 break; | |
304 | |
305 case AV_SAMPLE_FMT_FLT: | |
306 { | |
307 SwrContext *converter = swr_alloc(); | |
308 av_opt_set_int(converter, "in_channel_layout", av_get_default_channel_layout(2), 0); | |
309 //av_opt_set_int(converter, "in_sample_rate", sample_ra, 0); | |
310 av_opt_set_sample_fmt(converter, "in_sample_fmt", AV_SAMPLE_FMT_FLT, 0); | |
311 | |
312 av_opt_set_int(converter, "out_channel_layout", av_get_default_channel_layout(2), 0); | |
313 //av_opt_set_int(converter, "out_sample_rate", dst_sample_rate, 0); | |
314 av_opt_set_sample_fmt(converter, "out_sample_fmt", AV_SAMPLE_FMT_S16, 0); | |
315 | |
316 if (swr_init(converter) < 0) | |
317 { | |
318 __debugbreak(); | |
319 swr_free(&converter); | |
320 return; | |
321 } | |
322 | |
323 uint8_t **dst_channels; | |
324 int dst_linesize[8]; | |
325 //int dst_nb_channels = av_get_channel_layout_nb_channels(dst_channel_layout); | |
326 if (av_samples_alloc_array_and_samples(&dst_channels, dst_linesize, 2, num_channels * num_samples, AV_SAMPLE_FMT_S16, 0) < 0) | |
327 { | |
328 __debugbreak(); | |
329 swr_free(&converter); | |
330 return; | |
331 } | |
332 | |
333 if (swr_convert(converter, dst_channels, num_channels * num_samples, (const uint8_t **)channels, num_channels * num_samples) >= 0) //Invalid partial memory access, Uninitialized memory access | |
334 | |
335 stream->Write(dst_channels[0], num_channels * num_samples * sizeof(__int16)); | |
336 else | |
337 __debugbreak(); | |
338 | |
339 av_free(dst_channels[0]); | |
340 swr_free(&converter); | |
341 } | |
342 break; | |
343 | |
344 default: | |
345 __debugbreak(); | |
346 //if (Resample(next_frame->avframe, next_frame->avframe->channel_layout, next_frame->avframe->sample_rate, | |
347 // av_get_default_channel_layout(2), next_frame->avframe->sample_rate, AV_SAMPLE_FMT_S16P, resampled_data)) | |
348 } | |
349 } | |
350 | |
351 const uint16_t ff_wma_critical_freqs[25] = { | |
352 100, 200, 300, 400, 510, 630, 770, 920, | |
353 1080, 1270, 1480, 1720, 2000, 2320, 2700, 3150, | |
354 3700, 4400, 5300, 6400, 7700, 9500, 12000, 15500, | |
355 24500, | |
356 }; | |
357 extern const uint16_t ff_wma_critical_freqs[25]; | |
358 static float quant_table[96]; | |
359 | |
360 bool DecodeAudioFrame(AVCodecContext *dec_ctx, AVPacket *avpacket, AVFrame *avframe, MemoryStream *out_audio_data, int *out_num_audio_samples) | |
361 { | |
362 volatile int decoded = false; | |
363 do | |
364 { | |
365 if (avcodec_decode_audio4(dec_ctx, avframe, (int *)&decoded, avpacket) < 0) //Uninitialized portail memory access | |
366 { | |
367 log("Cannot decode audio frame\n"); | |
368 return false; | |
369 } | |
370 | |
371 if (!decoded) | |
372 log("Cannot decode audio frame in one piece\n"); | |
373 } while (!decoded); | |
374 | |
375 switch (dec_ctx->codec_id) | |
376 { | |
377 case AV_CODEC_ID_BINKAUDIO_DCT: | |
378 { | |
379 __debugbreak(); | |
380 } | |
381 case AV_CODEC_ID_BINKAUDIO_RDFT: | |
382 {//pts samples dpts | |
383 // 0 960 | |
384 //17280 960 17280 18x960 | |
385 //18240 960 960 1x960 | |
386 //20160 960 1920 2x960 | |
387 //21120 960 960 1x960 | |
388 //23040 960 1920 2x960 | |
389 /*static int bink_next_pts = 0; | |
390 | |
391 // there's a gap in the sound - fill empty samples in | |
392 if (bink_next_pts < avpacket->pts) | |
393 { | |
394 short silence[1024]; | |
395 memset(silence, 0, sizeof(silence)); | |
396 | |
397 int samples_to_fill = /*dec_ctx->channels * (avpacket->pts - bink_next_pts); | |
398 while (samples_to_fill > 0) | |
399 { | |
400 int samples_to_fill_this_step = samples_to_fill >= 1024 ? 1024 : samples_to_fill; | |
401 out_audio_data->Write(silence, samples_to_fill_this_step * sizeof(short)); | |
402 | |
403 samples_to_fill -= samples_to_fill_this_step; | |
404 } | |
405 } | |
406 | |
407 bink_next_pts = avpacket->pts + /*dec_ctx->channels * avframe->nb_samples; */ | |
408 | |
409 AVFrame frame; | |
410 int first; | |
411 int version_b; | |
412 int frame_len; | |
413 int overlap_len; | |
414 int block_size; | |
415 int num_bands; | |
416 unsigned int *bands; | |
417 float root; | |
418 int sample_rate = dec_ctx->sample_rate; | |
419 int sample_rate_half; | |
420 int i; | |
421 int frame_len_bits; | |
422 int channels; | |
423 | |
424 //compresses audio in chunks of varying sizes depending on sample rate: | |
425 // if sample rate < 22050, frame size is 2048 samples | |
426 // if sample rate < 44100, frame size is 4096 samples | |
427 // else, frame size is 8192 samples | |
428 | |
429 | |
430 /* determine frame length */ | |
431 if (dec_ctx->sample_rate < 22050) | |
432 frame_len_bits = 9; | |
433 else if (dec_ctx->sample_rate < 44100) | |
434 frame_len_bits = 10; | |
435 else | |
436 frame_len_bits = 11; | |
437 | |
438 if (dec_ctx->channels < 1 || dec_ctx->channels > 2) | |
439 { | |
440 av_log(dec_ctx, AV_LOG_ERROR, "invalid number of channels: %d\n", dec_ctx->channels); | |
441 return AVERROR_INVALIDDATA; | |
442 } | |
443 | |
444 version_b = dec_ctx->extradata_size >= 4 && dec_ctx->extradata[3] == 'b'; | |
445 if (version_b) | |
446 __debugbreak(); | |
447 | |
448 if (dec_ctx->codec->id == AV_CODEC_ID_BINKAUDIO_RDFT) | |
449 { | |
450 // audio is already interleaved for the RDFT format variant | |
451 dec_ctx->sample_fmt = AV_SAMPLE_FMT_FLT; | |
452 sample_rate *= dec_ctx->channels; | |
453 channels = 1; | |
454 if (!version_b) | |
455 frame_len_bits += av_log2(dec_ctx->channels); | |
456 } | |
457 else | |
458 { | |
459 channels = dec_ctx->channels; | |
460 dec_ctx->sample_fmt = AV_SAMPLE_FMT_FLTP; | |
461 } | |
462 | |
463 frame_len = 1 << frame_len_bits; //2048 | |
464 | |
465 //a frame is windowed with the previous frame; the size of the window is frame size / 16 | |
466 overlap_len = frame_len / 16; //128 | |
467 block_size = (frame_len - overlap_len) * channels; //1920 | |
468 | |
469 //compute half the sample rate as (sample rate + 1) / 2; | |
470 //initialize an array of band frequencies corresponding to an array of 25 critical frequencies (same as WMA, apparently), | |
471 // any for which the critical frequencies are less than half the sample rate | |
472 | |
473 sample_rate_half = (sample_rate + 1) / 2; //22050 | |
474 if (dec_ctx->codec->id == AV_CODEC_ID_BINKAUDIO_RDFT) | |
475 root = 2.0 / (sqrt(float(frame_len)) * 32768.0); | |
476 else | |
477 root = frame_len / (sqrt(float(frame_len)) * 32768.0); | |
478 for (i = 0; i < 96; i++) | |
479 { | |
480 /* constant is result of 0.066399999/log10(M_E) */ | |
481 quant_table[i] = expf(i * 0.15289164787221953823f) * root; | |
482 } | |
483 | |
484 /* calculate number of bands */ | |
485 //bands calculation: | |
486 //bands[0] = 1; | |
487 //foreach (i in 1..# of bands-1): | |
488 //bands[i] = crit_freq[i-1] * (frame length / 2) / (sample rate / 2); | |
489 //bands[# of bands] = frame length / 2 | |
490 for (num_bands = 1; num_bands < 25; num_bands++) | |
491 if (sample_rate_half <= ff_wma_critical_freqs[num_bands - 1]) | |
492 break; | |
493 | |
494 bands = (unsigned int *)(av_malloc((num_bands + 1) * sizeof(*bands))); | |
495 if (!bands) | |
496 return AVERROR(ENOMEM); | |
497 | |
498 /* populate bands data */ | |
499 bands[0] = 2; | |
500 for (i = 1; i < num_bands; i++) | |
501 bands[i] = (ff_wma_critical_freqs[i - 1] * frame_len / sample_rate_half) & ~1; | |
502 bands[num_bands] = frame_len; | |
503 | |
504 first = 1; | |
505 | |
506 //ff_rdft_init(&trans.rdft, frame_len_bits, DFT_C2R); | |
507 | |
508 avcodec_get_frame_defaults(&frame); | |
509 dec_ctx->coded_frame = &frame; | |
510 } | |
511 break; | |
512 /* | |
513 case AV_CODEC_ID_SMACKAUDIO: | |
514 { | |
515 static int smack_debug_next_audio_time = 0; | |
516 if (smack_debug_next_audio_time != packet->pts) | |
517 { | |
518 Error("There's a gap in the sound before frame %u\n", num_audio_frames); | |
519 __debugbreak(); // there's a gap in the sound | |
520 } | |
521 | |
522 int num_actual_data_channels = 0; | |
523 switch (dec_ctx->sample_fmt) | |
524 { | |
525 case AV_SAMPLE_FMT_U8: | |
526 case AV_SAMPLE_FMT_S16: | |
527 case AV_SAMPLE_FMT_S32: | |
528 case AV_SAMPLE_FMT_FLT: | |
529 case AV_SAMPLE_FMT_DBL: | |
530 num_actual_data_channels = 1; | |
531 break; | |
532 | |
533 case AV_SAMPLE_FMT_U8P: | |
534 case AV_SAMPLE_FMT_S16P: | |
535 case AV_SAMPLE_FMT_S32P: | |
536 case AV_SAMPLE_FMT_FLTP: | |
537 case AV_SAMPLE_FMT_DBLP: | |
538 num_actual_data_channels = dec_ctx->channels; | |
539 break; | |
540 | |
541 default: | |
542 case AV_SAMPLE_FMT_NONE: | |
543 case AV_SAMPLE_FMT_NB: | |
544 __debugbreak(); | |
545 } | |
546 | |
547 smack_debug_next_audio_time += dec_ctx->channels * frame->nb_samples * bytes_per_sample; | |
548 Assert(frame->avframe->linesize[0] == audio.dec_ctx->channels * frame->avframe->nb_samples * audio.bytes_per_sample / num_actual_data_channels, | |
549 "Smack audio size mismatch in frame %u in %s\n", audio_num_read_frames, movie_filename); | |
550 | |
551 frame->play_time = (double)frame->avpacket->pts / (double)audio.bytes_per_second; | |
552 } | |
553 break; | |
554 | |
555 case AV_CODEC_ID_MP3: | |
556 { | |
557 static int mp3_samples_decoded_so_far = 0; | |
558 static int mp3_prev_samples_count = frame->avframe->nb_samples; // mp3 seems to always feed same amount of samples | |
559 frame->play_time = (double)mp3_samples_decoded_so_far / (double)audio.dec_ctx->sample_rate; | |
560 | |
561 mp3_samples_decoded_so_far += frame->avframe->nb_samples; | |
562 Assert(mp3_prev_samples_count == frame->avframe->nb_samples, | |
563 "MP3 audio have variable sample count in frame %u in %s\n", audio_num_read_frames, movie_filename); | |
564 } | |
565 break; | |
566 | |
567 default: | |
568 { | |
569 __debugbreak(); | |
570 double samples_per_second = (double)audio.dec_ctx->time_base.den / (double)audio.dec_ctx->time_base.num; | |
571 double play_length = frame->avframe->nb_samples / samples_per_second; | |
572 frame->play_time = (double)frame->avpacket->pts / samples_per_second; | |
573 } | |
574 break;*/ | |
575 } | |
576 | |
577 if (!avframe->channel_layout) | |
578 { | |
579 log("Audio channel layout not specified, rolling back to default\n"); | |
580 avframe->channel_layout = av_get_default_channel_layout(dec_ctx->channels); | |
581 } | |
582 | |
583 *out_num_audio_samples = dec_ctx->channels * avframe->nb_samples; | |
584 InterleaveAudioData(out_audio_data, dec_ctx->sample_fmt, | |
585 dec_ctx->channels, avframe->nb_samples, avframe->data); | |
586 return true; | |
587 } | |
588 | |
589 bool LoadAudioTrack(AVFormatContext *format_ctx, AVCodecContext *dec_ctx, int audio_stream_idx, MemoryStream *out_audio_stream, int *out_num_audio_frames, int *out_num_audio_samples) | |
590 { | |
591 out_audio_stream->Reset(); | |
592 | |
593 AVFrame *frame = avcodec_alloc_frame(); | |
594 | |
595 AVPacket *packet = new AVPacket; | |
596 av_init_packet(packet); | |
597 | |
598 int num_audio_frames = 0; | |
599 int num_audio_samples = 0; | |
600 | |
601 while (av_read_frame(format_ctx, packet) >= 0) | |
602 { | |
603 // Is this a packet from the audio stream? | |
604 if (packet->stream_index != audio_stream_idx) | |
605 { | |
606 //log("Suspicious stream id %u in %s", packet->stream_index, filenamea); | |
607 continue; | |
608 } | |
609 | |
610 // Decode audio frame | |
611 int num_samples_decoded; | |
612 DecodeAudioFrame(dec_ctx, packet, frame, out_audio_stream, &num_samples_decoded); | |
613 | |
614 num_audio_samples += num_samples_decoded; | |
615 num_audio_frames++; | |
616 } | |
617 *out_num_audio_frames = num_audio_frames; | |
618 *out_num_audio_samples = num_audio_samples; | |
619 | |
620 avcodec_free_frame(&frame); | |
621 delete frame; | |
622 av_free_packet(packet); | |
623 delete packet; | |
624 | |
625 return true; | |
626 } | |
627 | |
628 class Track: public Media::ITrack | |
629 { | |
630 public: | |
631 inline Track() | |
632 { | |
633 this->format_ctx = nullptr; | |
634 this->audio_num_samples = 0; | |
635 } | |
636 | |
637 void Release() | |
638 { | |
639 ReleaseAvcodec(); | |
640 } | |
641 | |
642 void ReleaseAvcodec() | |
643 { | |
644 audio.Release(); | |
645 if (format_ctx) | |
646 { | |
647 av_close_input_file(format_ctx); | |
648 Log::Warning(L"close audio format context file\n"); | |
649 format_ctx = nullptr; | |
650 } | |
651 } | |
652 | |
653 bool LoadAudio(const wchar_t *filename) | |
654 { | |
655 char filenamea[1024]; | |
656 sprintf(filenamea, "%S", filename); | |
657 // Open audio file | |
658 if (avformat_open_input(&format_ctx, filenamea, nullptr, nullptr) >= 0) | |
659 { | |
660 // Retrieve stream information | |
661 if (avformat_find_stream_info(format_ctx, nullptr) >= 0) | |
662 { | |
663 // Dump information about file onto standard error | |
664 av_dump_format(format_ctx, 0, filenamea, 0); | |
665 | |
666 if (!av_open_audio_stream(format_ctx, &audio)) | |
667 { | |
668 Error("Cannot open strack: %s", filenamea); | |
669 return Release(), false; | |
670 } | |
671 | |
672 MemoryStream audio_plain_data; | |
673 int num_audio_frames; | |
674 int num_audio_samples; | |
675 | |
676 if (LoadAudioTrack(format_ctx, audio.dec_ctx, audio.stream_idx, &audio_plain_data, &num_audio_frames, &num_audio_samples)) | |
677 { | |
678 /*#ifdef _DEBUG | |
679 char debug_filename[1024]; | |
680 sprintf(debug_filename, "%s.wav", filenamea); | |
681 FILE *wav = fopen(debug_filename, "w+b"); | |
682 | |
683 extern void write_wav_header(FILE *wav, int channel_count = 2, int sample_rate = 22050, int bytes_per_sample = 2); | |
684 write_wav_header(wav, audio.dec_ctx->channels, audio.dec_ctx->sample_rate, audio.bytes_per_sample); | |
685 | |
686 fwrite(audio_plain_data.Ptr(), audio_plain_data.Current(), 1, wav); | |
687 | |
688 extern void fix_wav_header(FILE *wav, int wav_bytes_in_stream); | |
689 fix_wav_header(wav, audio_plain_data.Current()); | |
690 #endif*/ | |
691 | |
692 device_buffer = provider->CreateTrack16(audio.dec_ctx->channels, audio.dec_ctx->sample_rate, 2, num_audio_samples, audio_plain_data.Ptr()); | |
693 | |
694 Release(); | |
695 return true; | |
696 } | |
697 } | |
698 Release(); | |
699 fprintf(stderr, "ffmpeg: Unable to find stream info\n"); | |
700 return false; | |
701 } | |
702 fprintf(stderr, "ffmpeg: Unable to open input file\n"); | |
703 return false; | |
704 } | |
705 | |
706 virtual void Play(bool loop) | |
707 { | |
708 provider->PlayTrack16(device_buffer, loop); | |
709 mSourceID = device_buffer->source_id; | |
710 } | |
711 | |
712 protected: | |
713 AVFormatContext *format_ctx; | |
714 AVAudioStream audio; | |
715 int audio_num_samples; | |
716 | |
717 bool stopped; | |
718 | |
719 OpenALSoundProvider::TrackBuffer *device_buffer; | |
720 }; | |
721 | |
722 class Movie: public Media::IMovie | |
723 { | |
724 public: | |
725 inline Movie() | |
726 { | |
727 this->movie_filename[0] = 0; | |
728 this->width = 0; | |
729 this->height = 0; | |
730 this->format_ctx = nullptr; | |
731 this->end_of_file = false; | |
732 this->playback_time = 0.0; | |
733 | |
734 this->num_audio_frames = 0; | |
735 this->num_audio_samples = 0; | |
736 | |
737 this->last_resampled_frame_num = -1; | |
738 memset(last_resampled_frame_data, 0, sizeof(last_resampled_frame_data)); | |
739 memset(last_resampled_frame_linesize, 0, sizeof(last_resampled_frame_linesize)); | |
740 | |
741 audio_data_in_device = nullptr; | |
742 decoding_packet = nullptr; | |
743 ioBuffer = nullptr; | |
744 format_ctx = nullptr; | |
745 avioContext = nullptr; | |
746 } | |
747 | |
748 virtual ~Movie() {} | |
749 | |
750 virtual void Release() | |
751 { | |
752 ReleaseAVCodec(); | |
753 | |
754 if (audio_data_in_device) | |
755 provider->DeleteStreamingTrack(&audio_data_in_device); | |
756 } | |
757 | |
758 inline void ReleaseAVCodec() | |
759 { | |
760 audio.Release(); | |
761 video.Release(); | |
762 | |
763 if (format_ctx) | |
764 { | |
765 // Close the video file | |
766 av_close_input_file(format_ctx); | |
767 Log::Warning(L"close video format context file\n"); | |
768 format_ctx = nullptr; | |
769 } | |
770 if(avioContext) | |
771 { | |
772 av_free(avioContext); | |
773 avioContext = nullptr; | |
774 } | |
775 if (ioBuffer) | |
776 { | |
777 //av_free(ioBuffer); | |
778 ioBuffer = nullptr; | |
779 } | |
780 av_free_packet(decoding_packet); | |
781 delete decoding_packet; | |
782 avcodec_free_frame(&decoding_frame); | |
783 delete decoding_frame; | |
784 if (last_resampled_frame_data[0]) | |
785 av_freep(&last_resampled_frame_data[0]); | |
786 | |
787 } | |
788 | |
789 bool Load(const wchar_t *filename, int dst_width, int dst_height, int cache_ms) //Загрузка | |
790 { | |
791 char filenamea[1024]; | |
792 sprintf(filenamea, "%S", filename); | |
793 sprintf(movie_filename, "%S", filename); | |
794 | |
795 width = dst_width; | |
796 height = dst_height; | |
797 // Open video file | |
798 if (avformat_open_input(&format_ctx, filenamea, nullptr, nullptr) >= 0) | |
799 { | |
800 // Retrieve stream information | |
801 if (avformat_find_stream_info(format_ctx, nullptr) >= 0) | |
802 { | |
803 // Dump information about file onto standard error | |
804 av_dump_format(format_ctx, 0, filenamea, 0); | |
805 | |
806 if (!av_open_audio_stream(format_ctx, &audio)) | |
807 { | |
808 Error("Cannot open audio stream: %s", filenamea); | |
809 return Release(), false; | |
810 } | |
811 | |
812 if (!av_open_video_stream(format_ctx, &video)) | |
813 { | |
814 Error("Cannot open video stream: %s", filenamea); | |
815 return Release(), false; | |
816 } | |
817 | |
818 //Ritor1: include | |
819 if (_stricmp("binkvideo", video.dec->name) ) | |
820 { | |
821 pMediaPlayer->current_movie_width = video.dec_ctx->width; | |
822 pMediaPlayer->current_movie_height = video.dec_ctx->height; | |
823 } | |
824 else | |
825 { | |
826 pMediaPlayer->current_movie_width = width; | |
827 pMediaPlayer->current_movie_height = height; | |
828 } | |
829 // | |
830 decoding_packet = new AVPacket; | |
831 av_init_packet(decoding_packet); | |
832 | |
833 // Allocate video frame | |
834 decoding_frame = avcodec_alloc_frame(); | |
835 | |
836 audio_data_in_device = provider->CreateStreamingTrack16(audio.dec_ctx->channels, audio.dec_ctx->sample_rate, 2); | |
837 return true; | |
838 } | |
839 fprintf(stderr, "ffmpeg: Unable to find stream info\n"); | |
840 return Release(), false; | |
841 } | |
842 fprintf(stderr, "ffmpeg: Unable to open input file\n"); | |
843 return Release(), false; | |
844 } | |
845 | |
846 bool LoadFromLOD(HANDLE h, int readFunction(void*, uint8_t*, int), int64_t seekFunction(void*, int64_t, int), int width, int height) | |
847 { | |
848 if (!ioBuffer) | |
849 ioBuffer = (unsigned char *)av_malloc(0x4000 + FF_INPUT_BUFFER_PADDING_SIZE); // can get av_free()ed by libav | |
850 if (!avioContext) | |
851 avioContext = avio_alloc_context(ioBuffer, 0x4000, 0, h, readFunction, NULL, seekFunction); | |
852 if (!format_ctx) | |
853 format_ctx = avformat_alloc_context(); | |
854 format_ctx->pb = avioContext; | |
855 return Load(L"dummyFilename", width, height, 0); | |
856 } | |
857 | |
858 virtual void GetNextFrame(double dt, void *dst_surface) | |
859 { | |
860 playback_time += dt; | |
861 | |
862 AVPacket *avpacket = decoding_packet; | |
863 AVFrame *avframe = decoding_frame; | |
864 | |
865 avcodec_get_frame_defaults(avframe); | |
866 | |
867 int desired_frame_number = floor(playback_time * video.dec_ctx->time_base.den / video.dec_ctx->time_base.num + 0.5); | |
868 if (last_resampled_frame_num == desired_frame_number) | |
869 { | |
870 memcpy(dst_surface, last_resampled_frame_data[0], pMediaPlayer->current_movie_height * last_resampled_frame_linesize[0]); | |
871 return; | |
872 } | |
873 | |
874 volatile int frameFinished = false; | |
875 | |
876 // keep reading packets until we hit the end or find a video packet | |
877 do | |
878 { | |
879 if (pMediaPlayer->loop_current_file) | |
880 { | |
881 //Now seek back to the beginning of the stream | |
882 if (video.dec_ctx->frame_number >= video.stream->duration - 1 ) | |
883 pMediaPlayer->bPlaying_Movie = false; | |
884 } | |
885 if (av_read_frame(format_ctx, avpacket) < 0) | |
886 { | |
887 // probably movie is finished | |
888 pMediaPlayer->bPlaying_Movie = false; | |
889 av_free_packet(avpacket); | |
890 return; | |
891 } | |
892 // Is this a packet from the video stream? | |
893 // audio packet - queue into playing | |
894 if (avpacket->stream_index == audio.stream_idx) | |
895 { | |
896 MemoryStream audio_data; | |
897 if (DecodeAudioFrame(audio.dec_ctx, avpacket, avframe, &audio_data, &num_audio_samples)) | |
898 provider->Stream16(audio_data_in_device, num_audio_samples, audio_data.Ptr()); | |
899 //continue; | |
900 } | |
901 | |
902 // Decode video frame | |
903 // video packet - decode & maybe show | |
904 else if (avpacket->stream_index == video.stream_idx) | |
905 { | |
906 do | |
907 { | |
908 if (avcodec_decode_video2(video.dec_ctx, avframe, (int *)&frameFinished, avpacket) < 0) | |
909 __debugbreak(); | |
910 } while (!frameFinished); | |
911 } | |
912 else __debugbreak(); // unknown stream | |
913 } | |
914 while (avpacket->stream_index != video.stream_idx || | |
915 avpacket->pts != desired_frame_number); | |
916 | |
917 if (frameFinished) | |
918 { | |
919 if (last_resampled_frame_data[0]) | |
920 av_freep(&last_resampled_frame_data[0]); | |
921 | |
922 AVPixelFormat rescaled_format = AV_PIX_FMT_RGB32; | |
923 uint8_t *rescaled_data[4] = {nullptr, nullptr, nullptr, nullptr}; | |
924 int rescaled_linesize[4] = {0, 0, 0, 0}; | |
925 | |
926 if (av_image_alloc(rescaled_data, rescaled_linesize, pMediaPlayer->current_movie_width, pMediaPlayer->current_movie_height, rescaled_format, 1) >= 0) | |
927 { | |
928 SwsContext *converter = sws_getContext(avframe->width, avframe->height, (AVPixelFormat)avframe->format, | |
929 pMediaPlayer->current_movie_width, pMediaPlayer->current_movie_height, rescaled_format, | |
930 SWS_BICUBIC, nullptr, nullptr, nullptr); | |
931 sws_scale(converter, avframe->data, avframe->linesize, 0, avframe->height, rescaled_data, rescaled_linesize); | |
932 sws_freeContext(converter); | |
933 | |
934 memcpy(dst_surface, rescaled_data[0], pMediaPlayer->current_movie_height * rescaled_linesize[0]); | |
935 | |
936 last_resampled_frame_num = desired_frame_number; | |
937 memcpy(last_resampled_frame_data, rescaled_data, sizeof(rescaled_data)); | |
938 memcpy(last_resampled_frame_linesize, rescaled_linesize, sizeof(rescaled_linesize)); | |
939 } | |
940 } | |
941 else | |
942 memset(dst_surface, 0, width * pMediaPlayer->current_movie_height * 4); | |
943 | |
944 // Free the packet that was allocated by av_read_frame | |
945 av_free_packet(avpacket); | |
946 } | |
947 | |
948 virtual void Play() | |
949 { | |
950 } | |
951 | |
952 protected: | |
953 char movie_filename[256]; | |
954 int width; | |
955 int height; | |
956 bool stopped; | |
957 AVFormatContext *format_ctx; | |
958 double playback_time; | |
959 bool end_of_file; | |
960 | |
961 AVPacket *decoding_packet; | |
962 AVFrame *decoding_frame; | |
963 | |
964 AVAudioStream audio; | |
965 int num_audio_frames; | |
966 int num_audio_samples; | |
967 unsigned char *ioBuffer; | |
968 AVIOContext *avioContext; | |
969 OpenALSoundProvider::StreamingTrackBuffer *audio_data_in_device; | |
970 | |
971 AVVideoStream video; | |
972 int last_resampled_frame_num; | |
973 uint8_t *last_resampled_frame_data[4]; | |
974 int last_resampled_frame_linesize[4]; | |
975 }; | |
976 | |
977 ITrack *MPlayer::LoadTrack(const wchar_t *filename) | |
978 { | |
979 auto audio_track = new Track; | |
980 Log::Warning(L"allocation dynamic memory for audio_track\n"); | |
981 if (!audio_track->LoadAudio(filename)) | |
982 { | |
983 delete audio_track; | |
984 Log::Warning(L"delete dynamic memory for audio_track\n"); | |
985 audio_track = nullptr; | |
986 } | |
987 return audio_track; | |
988 } | |
989 | |
990 IMovie *MPlayer::LoadMovie(const wchar_t *filename, int width, int height, int cache_ms) //Загрузить видео | |
991 { | |
992 movie = new Movie; | |
993 Log::Warning(L"allocation dynamic memory for movie\n"); | |
994 if (!movie->Load(filename, width, height, cache_ms)) | |
995 { | |
996 delete movie; | |
997 Log::Warning(L"delete dynamic memory for movie\n"); | |
998 movie = nullptr; | |
999 } | |
1000 return movie; | |
1001 } | |
1002 | |
1003 | |
1004 //for video///////////////////////////////////////////////////////////////// | |
1005 | |
1006 //----- (004BE9D8) -------------------------------------------------------- | |
1007 void MPlayer::Initialize(OSWindow *target_window) | |
1008 { | |
1009 DWORD NumberOfBytesRead; // [sp+10h] [bp-4h]@9 | |
1010 | |
1011 window = target_window; | |
1012 | |
1013 unsigned int uBinkVersionMajor = -1, | |
1014 uBinkVersionMinor = -1; | |
1015 //GetDllVersion(L"BINKW32.DLL", &uBinkVersionMajor, &uBinkVersionMinor); | |
1016 //uBinkVersion = (unsigned __int64)uBinkVersionMajor << 32 | uBinkVersionMinor; | |
1017 | |
1018 strcpy(pTmpBuf.data(), "anims\\might7.vid"); | |
1019 hMightVid = CreateFileW(L"anims\\might7.vid", GENERIC_READ, FILE_SHARE_READ, 0, OPEN_EXISTING, 0x8000080, 0); | |
1020 if ( hMightVid == INVALID_HANDLE_VALUE ) | |
1021 { | |
1022 sprintf(pTmpBuf2.data(), "Can't open file - anims\\%s.smk", pTmpBuf.data()); | |
1023 MessageBoxA(0, pTmpBuf2.data(), "Video File Error", 0); | |
1024 return; | |
1025 } | |
1026 strcpy(pTmpBuf.data(), "anims\\magic7.vid"); | |
1027 hMagicVid = CreateFileW(L"anims\\magic7.vid", GENERIC_READ, FILE_SHARE_READ, 0, OPEN_EXISTING, 0x8000080, 0); | |
1028 if ( hMagicVid == INVALID_HANDLE_VALUE ) | |
1029 { | |
1030 if ( !bCanLoadFromCD ) | |
1031 { | |
1032 sprintf(pTmpBuf2.data(), "Can't open file - anims\\%s.smk", pTmpBuf.data()); | |
1033 MessageBoxA(0, pTmpBuf2.data(), "Video File Error", 0); | |
1034 return; | |
1035 } | |
1036 sprintf(pTmpBuf2.data(), "%c:\\%s", (unsigned __int8)cMM7GameCDDriveLetter, pTmpBuf.data()); | |
1037 hMagicVid = CreateFileA(pTmpBuf2.data(), GENERIC_READ, FILE_SHARE_READ, 0, OPEN_EXISTING, 0x8000080, 0); | |
1038 if ( hMagicVid == (HANDLE)INVALID_HANDLE_VALUE ) | |
1039 { | |
1040 sprintf(pTmpBuf2.data(), "Can't open file - %s", pTmpBuf.data()); | |
1041 MessageBoxA(0, pTmpBuf2.data(), "Video File Error", 0); | |
1042 return; | |
1043 } | |
1044 } | |
1045 ReadFile(hMightVid, &uNumMightVideoHeaders, 4, &NumberOfBytesRead, 0); | |
1046 ReadFile(hMagicVid, &uNumMagicVideoHeaders, 4, &NumberOfBytesRead, 0); | |
1047 pMightVideoHeaders = (MovieHeader *)malloc(sizeof(MovieHeader) * uNumMightVideoHeaders + 2); | |
1048 pMagicVideoHeaders = (MovieHeader *)malloc(sizeof(MovieHeader) * uNumMagicVideoHeaders + 2); | |
1049 ReadFile(hMightVid, pMightVideoHeaders, 44 * uNumMightVideoHeaders, &NumberOfBytesRead, 0); | |
1050 ReadFile(hMagicVid, pMagicVideoHeaders, 44 * uNumMagicVideoHeaders, &NumberOfBytesRead, 0); | |
1051 } | |
1052 | |
1053 //----- (004BF411) -------------------------------------------------------- | |
1054 void MPlayer::OpenFullscreenMovie(const char *pFilename, unsigned int bLoop/*, int ScreenSizeFlag*/) | |
1055 { | |
1056 if (!this->bPlaying_Movie) | |
1057 { | |
1058 pEventTimer->Pause(); | |
1059 if (pAudioPlayer->hAILRedbook) | |
1060 AIL_redbook_pause(pAudioPlayer->hAILRedbook); | |
1061 | |
1062 bStopBeforeSchedule = false; | |
1063 bFirstFrame = false; | |
1064 this->bLoopPlaying = bLoop; | |
1065 LoadMovie(pFilename); | |
1066 return; | |
1067 } | |
1068 } | |
1069 | |
1070 //----- (004BF28F) -------------------------------------------------------- | |
1071 void MPlayer::OpenHouseMovie(const char *pMovieName, unsigned int a3_1) | |
1072 { | |
1073 if (!this->bPlaying_Movie) | |
1074 { | |
1075 //Prepare(); | |
1076 pEventTimer->Pause(); | |
1077 if (pAudioPlayer->hAILRedbook) | |
1078 AIL_redbook_pause(pAudioPlayer->hAILRedbook); | |
1079 | |
1080 bStopBeforeSchedule = false; | |
1081 bFirstFrame = false; | |
1082 | |
1083 this->bLoopPlaying = a3_1; | |
1084 /*if ( LOBYTE(this->field_104) == 1 ) | |
1085 { | |
1086 MessageBoxA(nullptr, "Unsupported Bink playback!", "E:\\WORK\\MSDEV\\MM7\\MM7\\Code\\Video.cpp:925", 0); | |
1087 return; | |
1088 }*/ | |
1089 | |
1090 LoadMovie(pMovieName); | |
1091 time_video_begin = GetTickCount(); | |
1092 } | |
1093 } | |
1094 | |
1095 //----- (004BE70E) -------------------------------------------------------- | |
1096 void MPlayer::FullscreenMovieLoop(const char *pMovieName, int a2/*, int ScreenSizeFlag, int a4*/) | |
1097 { | |
1098 int v4; // ebp@1 | |
1099 MSG Msg; // [sp+Ch] [bp-1Ch]@12 | |
1100 | |
1101 v4 = a2; | |
1102 if ( dword_6BE364_game_settings_1 & (GAME_SETTINGS_NO_HOUSE_ANIM | GAME_SETTINGS_NO_INTRO) || | |
1103 bNoVideo) | |
1104 return; | |
1105 | |
1106 if ( a2 == 2 ) | |
1107 v4 = 0; | |
1108 ShowCursor(0); | |
1109 OpenFullscreenMovie(pMovieName, 0); | |
1110 bPlaying_Movie = 1; | |
1111 field_44 = v4; | |
1112 pRenderer->ClearTarget(0); | |
1113 pCurrentScreen = SCREEN_VIDEO; | |
1114 | |
1115 auto hwnd = pMediaPlayer->window->GetApiHandle(); | |
1116 | |
1117 RECT rc_client; | |
1118 GetClientRect(hwnd, &rc_client); | |
1119 int client_width = rc_client.right - rc_client.left, | |
1120 client_height = rc_client.bottom - rc_client.top; | |
1121 | |
1122 HDC dc = GetDC(hwnd); | |
1123 HDC back_dc = CreateCompatibleDC(dc); | |
1124 HBITMAP back_bmp = CreateCompatibleBitmap(dc, client_width, client_height); | |
1125 auto frame_buffer = new char[client_width * client_height * 4]; | |
1126 SelectObject(back_dc, back_bmp); | |
1127 | |
1128 DWORD t = GetTickCount(); | |
1129 | |
1130 bPlaying_Movie = true; | |
1131 | |
1132 while (true) | |
1133 { | |
1134 if (pMediaPlayer->bStopBeforeSchedule) | |
1135 break; | |
1136 while (PeekMessageA(&Msg, hwnd, 0, 0, PM_REMOVE)) | |
1137 { | |
1138 if (Msg.message == WM_QUIT) | |
1139 Game_DeinitializeAndTerminate(0); | |
1140 if (Msg.message == WM_PAINT) | |
1141 break; | |
1142 TranslateMessage(&Msg); | |
1143 DispatchMessageA(&Msg); | |
1144 } | |
1145 | |
1146 double dt = (GetTickCount() - t) / 1000.0; | |
1147 t = GetTickCount(); | |
1148 | |
1149 pMovie_Track->GetNextFrame(dt, frame_buffer); | |
1150 | |
1151 if (!bPlaying_Movie) | |
1152 break; | |
1153 | |
1154 if (frame_buffer) | |
1155 { | |
1156 // draw to hwnd | |
1157 BITMAPINFO bmi; | |
1158 bmi.bmiHeader.biSize = sizeof(BITMAPINFOHEADER); | |
1159 bmi.bmiHeader.biWidth = client_width; | |
1160 bmi.bmiHeader.biHeight = -client_height; | |
1161 bmi.bmiHeader.biPlanes = 1; | |
1162 bmi.bmiHeader.biBitCount = 32; | |
1163 bmi.bmiHeader.biCompression = BI_RGB; | |
1164 bmi.bmiHeader.biSizeImage = 0; | |
1165 bmi.bmiHeader.biXPelsPerMeter = 0; | |
1166 bmi.bmiHeader.biYPelsPerMeter = 0; | |
1167 bmi.bmiHeader.biClrUsed = 0; | |
1168 bmi.bmiHeader.biClrImportant = 0; | |
1169 GetDIBits(back_dc, back_bmp, 0, client_height, 0, &bmi, DIB_RGB_COLORS); | |
1170 SetDIBits(back_dc, back_bmp, 0, client_height, frame_buffer, &bmi, DIB_RGB_COLORS); | |
1171 BitBlt(dc, 0, 0, client_width, client_height, back_dc, 0, 0, SRCCOPY); | |
1172 } | |
1173 | |
1174 GUI_MainMenuMessageProc(); | |
1175 | |
1176 if (pMediaPlayer->bStopBeforeSchedule == 1) | |
1177 Sleep(1000); | |
1178 } | |
1179 delete [] frame_buffer; | |
1180 DeleteObject(back_bmp); | |
1181 DeleteObject(back_dc); | |
1182 ReleaseDC(hwnd, dc); | |
1183 | |
1184 pMediaPlayer->Unload(); | |
1185 | |
1186 //if (a4 == 1) | |
1187 pCurrentScreen = SCREEN_GAME; | |
1188 | |
1189 pMediaPlayer->bPlaying_Movie = false; | |
1190 | |
1191 ShowCursor(1); | |
1192 | |
1193 /*if ( pCurrentScreen == SCREEN_VIDEO ) | |
1194 pCurrentScreen = SCREEN_GAME;*/ | |
1195 } | |
1196 | |
1197 void MPlayer::HouseMovieLoop() | |
1198 { | |
1199 if (pMovie_Track && !bNoVideo) | |
1200 { | |
1201 pRenderer->BeginScene(); | |
1202 pMouse->DrawCursorToTarget(); | |
1203 | |
1204 Log::Warning(L"smacker"); | |
1205 loop_current_file = true; | |
1206 pRenderer->BeginScene(); | |
1207 if (!bPlaying_Movie)//reload | |
1208 { | |
1209 unsigned int width = game_viewport_width; | |
1210 unsigned int height = game_viewport_height; | |
1211 MovieRelease(); | |
1212 | |
1213 SetFilePointer(hVidFile, uOffset, nullptr, FILE_BEGIN); | |
1214 pMovie_Track = nullptr; | |
1215 Log::Warning(L"reload pMovie_Track"); | |
1216 pMovie_Track = pMediaPlayer->LoadMovieFromLOD(hVidFile, &readFunction, &seekFunction, width, height); | |
1217 bPlaying_Movie = true; | |
1218 } | |
1219 //else | |
1220 //{ | |
1221 double dt = (GetTickCount() - time_video_begin) / 1000.0; | |
1222 //dt = 1.0/15.0; | |
1223 time_video_begin = GetTickCount(); | |
1224 | |
1225 //log("dt=%.5f\n", dt); | |
1226 | |
1227 auto image = new char[current_movie_width * current_movie_height * 4]; | |
1228 | |
1229 pMovie_Track->GetNextFrame(dt, image); | |
1230 | |
1231 int image_array[460 * 344];//game_viewport_width * game_viewport_height | |
1232 if (image) | |
1233 { | |
1234 memcpy(image_array, image, sizeof (image_array)); | |
1235 for (unsigned int y = 8; y < 8 + game_viewport_height; ++y)//координаты местоположения видеоролика | |
1236 { | |
1237 for (unsigned int x = 8; x < 8 + game_viewport_width; ++x) | |
1238 { | |
1239 auto p = (unsigned __int32 *)pRenderer->pTargetSurface + x + y * pRenderer->uTargetSurfacePitch; | |
1240 *p = image_array[((x - 8) + ((y - 8)*game_viewport_width))]; | |
1241 } | |
1242 } | |
1243 delete[] image; | |
1244 } | |
1245 //} | |
1246 pRenderer->EndScene(); | |
1247 pMouse->ReadCursorWithItem(); | |
1248 pRenderer->EndScene(); | |
1249 } | |
1250 } | |
1251 | |
1252 //----- (004BF73A) -------------------------------------------------------- | |
1253 void MPlayer::SelectMovieType() | |
1254 { | |
1255 char Source[32]; // [sp+Ch] [bp-40h]@1 | |
1256 | |
1257 strcpy(Source, this->pCurrentMovieName); | |
1258 pMediaPlayer->Unload(); | |
1259 if ( this->uMovieType == 1 ) | |
1260 OpenHouseMovie(Source, LOBYTE(this->bLoopPlaying)); | |
1261 else if ( this->uMovieType == 2 ) | |
1262 OpenFullscreenMovie(Source, LOBYTE(this->bLoopPlaying)); | |
1263 else | |
1264 __debugbreak(); | |
1265 } | |
1266 | |
1267 void MPlayer::LoadMovie(const char *pFilename) | |
1268 { | |
1269 char pVideoNameBik[120]; // [sp+Ch] [bp-28h]@2 | |
1270 char pVideoNameSmk[120]; // [sp+Ch] [bp-28h]@2 | |
1271 | |
1272 sprintf(pVideoNameBik, "%s.bik", pFilename); | |
1273 sprintf(pVideoNameSmk, "%s.smk", pFilename); | |
1274 for (uint i = 0; i < uNumMightVideoHeaders; ++i) | |
1275 { | |
1276 if (!_stricmp(pVideoNameSmk, pMightVideoHeaders[i].pVideoName)) | |
1277 { | |
1278 hVidFile = hMightVid; | |
1279 uOffset = pMightVideoHeaders[i].uFileOffset; | |
1280 uSize = pMightVideoHeaders[i + 1].uFileOffset - uOffset; | |
1281 this->uMovieType = 2; | |
1282 } | |
1283 } | |
1284 for (uint i = 0; i < uNumMagicVideoHeaders; ++i) | |
1285 { | |
1286 if (!_stricmp(pVideoNameBik, pMagicVideoHeaders[i].pVideoName)) | |
1287 { | |
1288 hVidFile = hMagicVid; | |
1289 uOffset = pMagicVideoHeaders[i].uFileOffset; | |
1290 uSize = pMagicVideoHeaders[i + 1].uFileOffset - uOffset; | |
1291 this->uMovieType = 1; | |
1292 } | |
1293 if (!_stricmp(pVideoNameSmk, pMagicVideoHeaders[i].pVideoName)) | |
1294 { | |
1295 hVidFile = hMagicVid; | |
1296 uOffset = pMagicVideoHeaders[i].uFileOffset; | |
1297 uSize = pMagicVideoHeaders[i + 1].uFileOffset - uOffset; | |
1298 this->uMovieType = 2; | |
1299 } | |
1300 } | |
1301 if (!hVidFile) | |
1302 { | |
1303 pMediaPlayer->Unload(); | |
1304 MessageBoxA(0, "MediaPlayer error", "MediaPlayer Error", 0); | |
1305 return; | |
1306 } | |
1307 | |
1308 SetFilePointer(hVidFile, uOffset, 0, FILE_BEGIN); | |
1309 strcpy(this->pCurrentMovieName, pFilename); | |
1310 | |
1311 auto hwnd = pMediaPlayer->window->GetApiHandle(); | |
1312 RECT rc_client; | |
1313 GetClientRect(hwnd, &rc_client); | |
1314 int client_width = rc_client.right - rc_client.left, | |
1315 client_height = rc_client.bottom - rc_client.top; | |
1316 | |
1317 pMovie_Track = pMediaPlayer->LoadMovieFromLOD(hVidFile, &readFunction, &seekFunction, client_width, client_height); | |
1318 } | |
1319 | |
1320 //----- (004BF794) -------------------------------------------------------- | |
1321 void MPlayer::ShowMM7IntroVideo_and_LoadingScreen() | |
1322 { | |
1323 RGBTexture tex; // [sp+Ch] [bp-30h]@1 | |
1324 unsigned int uTrackStartMS; // [sp+34h] [bp-8h]@8 | |
1325 unsigned int uTrackEndMS; // [sp+38h] [bp-4h]@8 | |
1326 | |
1327 pMediaPlayer->bStopBeforeSchedule = false; | |
1328 // pMediaPlayer->pResetflag = 0; | |
1329 bGameoverLoop = true; | |
1330 if (!bNoVideo) | |
1331 { | |
1332 pRenderer->PresentBlackScreen(); | |
1333 if ( !pMediaPlayer->bStopBeforeSchedule ) | |
1334 PlayFullscreenMovie(MOVIE_Intro, true); | |
1335 } | |
1336 | |
1337 tex.Load("mm6title.pcx", 2); | |
1338 pRenderer->BeginScene(); | |
1339 pRenderer->DrawTextureRGB(0, 0, &tex); | |
1340 free(tex.pPixels); | |
1341 tex.pPixels = 0; | |
1342 | |
1343 //LoadFonts_and_DrawCopyrightWindow(); | |
1344 DrawMM7CopyrightWindow(); | |
1345 | |
1346 pRenderer->EndScene(); | |
1347 pRenderer->Present(); | |
1348 | |
1349 #ifndef _DEBUG | |
1350 Sleep(1500); // let the copyright window stay for a while | |
1351 #endif | |
1352 | |
1353 if (!bNoSound && pAudioPlayer->hAILRedbook ) | |
1354 { | |
1355 pAudioPlayer->SetMusicVolume((signed __int64)(pSoundVolumeLevels[uMusicVolimeMultiplier] * 64.0)); | |
1356 AIL_redbook_stop(pAudioPlayer->hAILRedbook); | |
1357 AIL_redbook_track_info(pAudioPlayer->hAILRedbook, 14, &uTrackStartMS, &uTrackEndMS); | |
1358 AIL_redbook_play(pAudioPlayer->hAILRedbook, uTrackStartMS + 1, uTrackEndMS); | |
1359 } | |
1360 bGameoverLoop = false; | |
1361 } | |
1362 | |
1363 //----- (004BEBD7) -------------------------------------------------------- | |
1364 void MPlayer::Unload() | |
1365 { | |
1366 bPlaying_Movie = false; | |
1367 uMovieType = 0; | |
1368 memset(pCurrentMovieName, 0, 0x40); | |
1369 if ( pAudioPlayer->hAILRedbook && !bGameoverLoop ) | |
1370 AIL_redbook_resume(pAudioPlayer->hAILRedbook); | |
1371 pEventTimer->Resume(); | |
1372 | |
1373 pMovie_Track->Release(); | |
1374 delete pMovie_Track; | |
1375 pMovie_Track = nullptr; | |
1376 } | |
1377 | |
1378 int MPlayer::readFunction(void* opaque, uint8_t* buf, int buf_size) | |
1379 { | |
1380 HANDLE stream = (HANDLE)opaque; | |
1381 //int numBytes = stream->read((char*)buf, buf_size); | |
1382 int numBytes; | |
1383 ReadFile(stream, (char *)buf, buf_size, (LPDWORD)&numBytes, NULL); | |
1384 return numBytes; | |
1385 } | |
1386 | |
1387 int64_t MPlayer::seekFunction(void* opaque, int64_t offset, int whence) | |
1388 { | |
1389 if (whence == AVSEEK_SIZE) | |
1390 return pMediaPlayer->uSize; | |
1391 HANDLE h = (HANDLE)opaque; | |
1392 LARGE_INTEGER li; | |
1393 li.QuadPart = offset; | |
1394 | |
1395 if (!SetFilePointerEx(h, li, (PLARGE_INTEGER)&li, FILE_BEGIN)) | |
1396 return -1; | |
1397 return li.QuadPart; | |
1398 } | |
1399 | |
1400 //for video////////////////////////////////////////////////////////////////// | |
1401 | |
1402 | |
1403 | |
1404 IMovie *MPlayer::LoadMovieFromLOD(HANDLE h, int readFunction(void*, uint8_t*, int), int64_t seekFunction(void*, int64_t, int), int width, int height) | |
1405 { | |
1406 movie = new Movie; | |
1407 Log::Warning(L"allocation dynamic memory for movie\n"); | |
1408 if (movie) | |
1409 { | |
1410 if (movie->LoadFromLOD(h, readFunction, seekFunction, width, height)) | |
1411 return movie; | |
1412 delete movie; | |
1413 Log::Warning(L"delete dynamic memory for movie\n"); | |
1414 } | |
1415 return nullptr; | |
1416 } | |
1417 | |
1418 void MovieRelease() | |
1419 { | |
1420 movie->Release(); | |
1421 delete movie; | |
1422 Log::Warning(L"delete dynamic memory for movie\n"); | |
1423 movie = nullptr; | |
1424 } | |
1425 | |
1426 | |
1427 //for audio/////////////////////////////////////////////////////// | |
1428 //----- (004AB818) -------------------------------------------------------- | |
1429 void MPlayer::LoadAudioSnd() | |
1430 { | |
1431 DWORD NumberOfBytesRead; // [sp+Ch] [bp-4h]@3 | |
1432 | |
1433 hAudioSnd = CreateFileA("Sounds\\Audio.snd", GENERIC_READ, FILE_SHARE_READ, 0, OPEN_EXISTING, 0x8000080u, 0); | |
1434 if (hAudioSnd == INVALID_HANDLE_VALUE) | |
1435 { | |
1436 Log::Warning(L"Can't open file: %s", L"Sounds\\Audio.snd"); | |
1437 return; | |
1438 } | |
1439 | |
1440 ReadFile(hAudioSnd, &uNumSoundHeaders, 4, &NumberOfBytesRead, 0); | |
1441 pSoundHeaders = nullptr; | |
1442 pSoundHeaders = (SoundHeader *)malloc(52 * uNumSoundHeaders + 2); | |
1443 ReadFile(hAudioSnd, pSoundHeaders, 52 * uNumSoundHeaders, &NumberOfBytesRead, 0); | |
1444 } | |
1445 //for audio/////////////////////////////////////////////////////// | |
1446 | |
1447 void av_logger(void *, int, const char *format, va_list args) | |
1448 { | |
1449 va_list va; | |
1450 va_start(va, format); | |
1451 char msg[256]; | |
1452 vsprintf(msg, format, va); | |
1453 va_end(va); | |
1454 | |
1455 log("av: %s", msg); | |
1456 } | |
1457 | |
1458 MPlayer::MPlayer() | |
1459 { | |
1460 bPlaying_Movie = false; | |
1461 | |
1462 static int libavcodec_initialized = false; | |
1463 | |
1464 if (!libavcodec_initialized) | |
1465 { | |
1466 av_log_set_callback(av_logger); | |
1467 avcodec_register_all(); | |
1468 | |
1469 // Register all available file formats and codecs | |
1470 av_register_all(); | |
1471 | |
1472 libavcodec_initialized = true; | |
1473 } | |
1474 | |
1475 bStopBeforeSchedule = false; | |
1476 pMovie_Track = nullptr; | |
1477 | |
1478 if (!provider) | |
1479 { | |
1480 provider = new OpenALSoundProvider; | |
1481 Log::Warning(L"allocation dynamic memory for provider\n"); | |
1482 provider->Initialize(); | |
1483 } | |
1484 LoadAudioSnd(); | |
1485 } | |
1486 | |
1487 MPlayer::~MPlayer() | |
1488 { | |
1489 delete provider; | |
1490 Log::Warning(L"delete dynamic memory for provider\n"); | |
1491 | |
1492 bStopBeforeSchedule = false; | |
1493 // pResetflag = 0; | |
1494 pVideoFrame.Release(); | |
1495 } | |
1496 | |
1497 void PlayAudio(const wchar_t * pFilename) | |
1498 { | |
1499 pAudio_Track = pMediaPlayer->LoadTrack(pFilename); | |
1500 pAudio_Track->Play(); | |
1501 delete pAudio_Track; | |
1502 Log::Warning(L"delete dynamic memory for pAudio_Track\n"); | |
1503 pAudio_Track = nullptr; | |
1504 } | |
1505 | |
1506 void PlayMovie(const wchar_t * pFilename) | |
1507 { | |
1508 Media::IMovie *Movie_track = pMediaPlayer->LoadMovie(pFilename, 640, 480, 0); | |
1509 Movie_track->Play(); | |
1510 } | |
1511 |