2 Skelton for retropc emulator
4 Author : Takeda.Toshiya
13 #define DIRECTSOUND_VERSION 0x900
14 #define DIRECT3D_VERSION 0x900
15 // XXX: if your DirectX 9.0 SDK is newer and does not contain dinput.lib,
16 // please change the definition of DIRECTINPUT_VERSION from 0x500 to 0x800
17 #define DIRECTINPUT_VERSION 0x500
18 //#define DIRECTINPUT_VERSION 0x800
29 #include <d3d9types.h>
35 #include "../common.h"
36 #include "../config.h"
39 // relative path from *.vcproj/*.vcxproj, not from this directory :-(
40 #if defined(_MSC_VER) && (_MSC_VER >= 1800)
42 #pragma comment(lib, "../src/zlib-1.2.11/vc++2013/debug/zlibstat.lib")
44 #pragma comment(lib, "../src/zlib-1.2.11/vc++2013/release/zlibstat.lib")
48 #pragma comment(lib, "../src/zlib-1.2.11/vc++2008/debug/zlibstat.lib")
50 #pragma comment(lib, "../src/zlib-1.2.11/vc++2008/release/zlibstat.lib")
57 #pragma comment(lib, "wsock32.lib")
59 #pragma comment(lib, "comctl32.lib")
60 #pragma comment(lib, "msimg32.lib")
61 #pragma comment(lib, "gdiplus.lib")
62 using namespace Gdiplus;
63 #pragma comment(lib, "d3d9.lib")
64 #pragma comment(lib, "d3dx9.lib")
65 #pragma comment(lib, "vfw32.lib")
66 #pragma comment(lib, "dsound.lib")
67 #if DIRECTINPUT_VERSION >= 0x0800
68 #pragma comment(lib, "dinput8.lib")
70 #pragma comment(lib, "dinput.lib")
72 #pragma comment(lib, "dxguid.lib")
74 #if defined(USE_MOVIE_PLAYER) || defined(USE_VIDEO_CAPTURE)
75 #pragma comment(lib, "strmiids.lib")
78 EXTERN_C const CLSID CLSID_SampleGrabber;
79 EXTERN_C const CLSID CLSID_NullRenderer;
80 EXTERN_C const IID IID_ISampleGrabberCB;
81 MIDL_INTERFACE("0579154A-2B53-4994-B0D0-E773148EFF85")
82 ISampleGrabberCB : public IUnknown {
84 virtual HRESULT STDMETHODCALLTYPE SampleCB( double SampleTime,IMediaSample *pSample) = 0;
85 virtual HRESULT STDMETHODCALLTYPE BufferCB( double SampleTime,BYTE *pBuffer,long BufferLen) = 0;
87 EXTERN_C const IID IID_ISampleGrabber;
88 MIDL_INTERFACE("6B652FFF-11FE-4fce-92AD-0266B5D7C78F")
89 ISampleGrabber : public IUnknown {
91 virtual HRESULT STDMETHODCALLTYPE SetOneShot( BOOL OneShot) = 0;
92 virtual HRESULT STDMETHODCALLTYPE SetMediaType( const AM_MEDIA_TYPE *pType) = 0;
93 virtual HRESULT STDMETHODCALLTYPE GetConnectedMediaType( AM_MEDIA_TYPE *pType) = 0;
94 virtual HRESULT STDMETHODCALLTYPE SetBufferSamples( BOOL BufferThem) = 0;
95 virtual HRESULT STDMETHODCALLTYPE GetCurrentBuffer( /* [out][in] */ long *pBufferSize,/* [out] */ long *pBuffer) = 0;
96 virtual HRESULT STDMETHODCALLTYPE GetCurrentSample( /* [retval][out] */ IMediaSample **ppSample) = 0;
97 virtual HRESULT STDMETHODCALLTYPE SetCallback( ISampleGrabberCB *pCallback,long WhichMethodToCallback) = 0;
100 #ifdef USE_MOVIE_PLAYER
101 class CMySampleGrabberCB : public ISampleGrabberCB {
105 CMySampleGrabberCB(VM *vm_ptr)
109 STDMETHODIMP_(ULONG) AddRef()
113 STDMETHODIMP_(ULONG) Release()
117 STDMETHODIMP QueryInterface(REFIID riid, void **ppv)
119 if(riid == IID_ISampleGrabberCB || riid == IID_IUnknown) {
120 *ppv = (void *) static_cast<ISampleGrabberCB*>(this);
123 return E_NOINTERFACE;
125 STDMETHODIMP SampleCB(double SampleTime, IMediaSample *pSample)
129 STDMETHODIMP BufferCB(double dblSampleTime, BYTE *pBuffer, long lBufferSize)
131 vm->movie_sound_callback(pBuffer, lBufferSize);
137 #define WM_RESIZE (WM_USER + 1)
138 #define WM_SOCKET0 (WM_USER + 2)
139 #define WM_SOCKET1 (WM_USER + 3)
140 #define WM_SOCKET2 (WM_USER + 4)
141 #define WM_SOCKET3 (WM_USER + 5)
145 #define SOCKET_BUFFER_MAX 0x100000
148 #ifdef USE_VIDEO_CAPTURE
149 #define MAX_CAPTURE_DEVS 8
152 #define SUPPORT_WIN32_DLL
154 #define SCREEN_FILTER_NONE 0
155 #define SCREEN_FILTER_RGB 1
156 #define SCREEN_FILTER_RF 2
158 // check memory leaks
160 #define _CRTDBG_MAP_ALLOC
162 #define malloc(s) _malloc_dbg(s, _NORMAL_BLOCK, __FILE__, __LINE__)
163 #define new new(_NORMAL_BLOCK, __FILE__, __LINE__)
166 typedef struct bitmap_s {
168 inline bool initialized()
170 return (hdcDib != NULL);
172 inline scrntype_t* get_buffer(int y)
174 return lpBmp + width * (height - y - 1);
179 HBITMAP hBmp, hOldBmp;
185 typedef struct font_s {
187 inline bool initialized()
189 return (hFont != NULL);
192 int width, height, rotate;
198 typedef struct pen_s {
200 inline bool initialized()
202 return (hPen != NULL);
211 PAVISTREAM pAVICompressed;
213 LPBITMAPINFOHEADER pbmInfoHeader;
218 } rec_video_thread_param_t;
229 HANDLE hStdIn, hStdOut;
232 void initialize_input();
233 void release_input();
235 #if DIRECTINPUT_VERSION >= 0x0800
237 LPDIRECTINPUTDEVICE8 lpdikey;
238 // LPDIRECTINPUTDEVICE8 lpdijoy;
241 LPDIRECTINPUTDEVICE lpdikey;
242 // LPDIRECTINPUTDEVICE lpdijoy;
244 bool dinput_key_available;
245 // bool dinput_joy_available;
247 uint8_t keycode_conv[256];
248 uint8_t key_status[256]; // windows key code mapping
249 uint8_t key_dik_prev[256];
250 #ifdef USE_SHIFT_NUMPAD_KEY
251 uint8_t key_converted[256];
252 bool key_shift_pressed, key_shift_released;
254 bool key_caps_locked;
258 uint32_t joy_status[4]; // joystick #1-#4 (b0 = up, b1 = down, b2 = left, b3 = right, b4- = buttons
260 uint32_t joy_mask[4];
264 int32_t mouse_status[3]; // x, y, button (b0 = left, b1 = right)
269 void initialize_screen();
270 void release_screen();
271 void initialize_screen_buffer(bitmap_t *buffer, int width, int height, int mode);
272 void release_screen_buffer(bitmap_t *buffer);
273 #ifdef USE_SCREEN_FILTER
274 void apply_rgb_filter_to_screen_buffer(bitmap_t *source, bitmap_t *dest);
275 void apply_rgb_filter_x3_y3(bitmap_t *source, bitmap_t *dest);
276 void apply_rgb_filter_x3_y2(bitmap_t *source, bitmap_t *dest);
277 void apply_rgb_filter_x2_y3(bitmap_t *source, bitmap_t *dest);
278 void apply_rgb_filter_x2_y2(bitmap_t *source, bitmap_t *dest);
279 void apply_rgb_filter_x1_y1(bitmap_t *source, bitmap_t *dest);
281 //#ifdef USE_SCREEN_ROTATE
282 void rotate_screen_buffer(bitmap_t *source, bitmap_t *dest);
284 void stretch_screen_buffer(bitmap_t *source, bitmap_t *dest);
285 bool initialize_d3d9();
286 bool initialize_d3d9_surface(bitmap_t *buffer);
288 void release_d3d9_surface();
289 void copy_to_d3d9_surface(bitmap_t *buffer);
290 int add_video_frames();
292 bitmap_t vm_screen_buffer;
293 #ifdef USE_SCREEN_FILTER
294 bitmap_t filtered_screen_buffer;
295 bitmap_t tmp_filtered_screen_buffer;
297 //#ifdef USE_SCREEN_ROTATE
298 bitmap_t rotated_screen_buffer;
300 bitmap_t stretched_screen_buffer;
301 bitmap_t shrinked_screen_buffer;
302 bitmap_t video_screen_buffer;
304 bitmap_t* draw_screen_buffer;
306 int host_window_width, host_window_height;
307 bool host_window_mode;
308 int vm_screen_width, vm_screen_height;
309 int vm_window_width, vm_window_height;
310 int vm_window_width_aspect, vm_window_height_aspect;
311 int draw_screen_width, draw_screen_height;
313 Gdiplus::GdiplusStartupInput gdiSI;
317 LPDIRECT3DDEVICE9 lpd3d9Device;
318 LPDIRECT3DSURFACE9 lpd3d9Surface;
319 LPDIRECT3DSURFACE9 lpd3d9OffscreenSurface;
321 _TCHAR video_file_path[_MAX_PATH];
323 double rec_video_run_frames;
324 double rec_video_frames;
326 LPBITMAPINFO lpDibRec;
328 PAVISTREAM pAVIStream;
329 PAVISTREAM pAVICompressed;
330 AVICOMPRESSOPTIONS AVIOpts;
334 rec_video_thread_param_t rec_video_thread_param;
336 bool first_draw_screen;
337 bool first_invalidate;
338 bool self_invalidate;
341 void initialize_sound(int rate, int samples);
342 void release_sound();
344 int sound_rate, sound_samples;
345 bool sound_available, sound_started, sound_muted;
348 LPDIRECTSOUNDBUFFER lpdsPrimaryBuffer, lpdsSecondaryBuffer;
349 bool sound_first_half;
351 _TCHAR sound_file_path[_MAX_PATH];
352 FILEIO* rec_sound_fio;
354 int rec_sound_buffer_ptr;
357 #if defined(USE_MOVIE_PLAYER) || defined(USE_VIDEO_CAPTURE)
358 void initialize_video();
359 void release_video();
361 IGraphBuilder *pGraphBuilder;
362 IBaseFilter *pVideoBaseFilter;
363 IBaseFilter *pCaptureBaseFilter;
364 ICaptureGraphBuilder2 *pCaptureGraphBuilder2;
365 ISampleGrabber *pVideoSampleGrabber;
366 IBaseFilter *pSoundBaseFilter;
367 ISampleGrabber *pSoundSampleGrabber;
368 CMySampleGrabberCB *pSoundCallBack;
369 IMediaControl *pMediaControl;
370 IMediaSeeking *pMediaSeeking;
371 IMediaPosition *pMediaPosition;
372 IVideoWindow *pVideoWindow;
373 IBasicVideo *pBasicVideo;
374 IBasicAudio *pBasicAudio;
375 bool bTimeFormatFrame;
376 bool bVerticalReversed;
378 bitmap_t direct_show_screen_buffer;
379 bitmap_t direct_show_stretch_buffer;
380 int direct_show_width, direct_show_height;
381 bool direct_show_mute[2];
383 #ifdef USE_MOVIE_PLAYER
384 double movie_frame_rate;
385 int movie_sound_rate;
387 #ifdef USE_VIDEO_CAPTURE
388 void enum_capture_devs();
389 bool connect_capture_dev(int index, bool pin);
390 int cur_capture_dev_index;
391 int num_capture_devs;
392 _TCHAR capture_dev_name[MAX_CAPTURE_DEVS][256];
397 void initialize_socket();
398 void release_socket();
401 bool is_tcp[SOCKET_MAX];
402 struct sockaddr_in udpaddr[SOCKET_MAX];
403 int socket_delay[SOCKET_MAX];
404 char recv_buffer[SOCKET_MAX][SOCKET_BUFFER_MAX];
405 int recv_r_ptr[SOCKET_MAX], recv_w_ptr[SOCKET_MAX];
418 void initialize(int rate, int samples);
427 return (lock_count != 0);
429 void force_unlock_vm();
430 void sleep(uint32_t ms);
433 void open_console(const _TCHAR* title);
434 void close_console();
435 unsigned int get_console_code_page();
436 bool is_console_active();
437 void set_console_text_attribute(unsigned short attr);
438 void write_console(_TCHAR* buffer, unsigned int length);
439 int read_console_input(_TCHAR* buffer);
440 bool is_console_key_pressed(int vk);
441 void close_debugger_console();
445 void key_down(int code, bool repeat);
446 void key_up(int code);
447 void key_down_native(int code, bool repeat);
448 void key_up_native(int code);
449 void key_lost_focus()
455 void disable_mouse();
457 bool is_mouse_enabled()
459 return mouse_enabled;
462 uint8_t* get_key_buffer()
467 uint32_t* get_joy_buffer()
473 int32_t* get_mouse_buffer()
483 int get_window_mode_width(int mode);
484 int get_window_mode_height(int mode);
485 void set_host_window_size(int window_width, int window_height, bool window_mode);
486 void set_vm_screen_size(int screen_width, int screen_height, int window_width, int window_height, int window_width_aspect, int window_height_aspect);
487 int get_vm_window_width()
489 return vm_window_width;
491 int get_vm_window_height()
493 return vm_window_height;
495 int get_vm_window_width_aspect()
497 return vm_window_width_aspect;
499 int get_vm_window_height_aspect()
501 return vm_window_height_aspect;
503 scrntype_t* get_vm_screen_buffer(int y);
505 #ifdef ONE_BOARD_MICRO_COMPUTER
508 first_invalidate = true;
511 void capture_screen();
512 bool start_record_video(int fps);
513 void stop_record_video();
514 void restart_record_video();
515 void add_extra_frames(int extra_frames);
516 bool now_record_video;
517 #ifdef USE_SCREEN_FILTER
518 bool screen_skip_line;
522 void update_sound(int* extra_frames);
525 void start_record_sound();
526 void stop_record_sound();
527 void restart_record_sound();
528 bool now_record_sound;
530 // common video device
531 #if defined(USE_MOVIE_PLAYER) || defined(USE_VIDEO_CAPTURE)
532 void get_video_buffer();
533 void mute_video_dev(bool l, bool r);
535 #ifdef USE_MOVIE_PLAYER
536 bool open_movie_file(const _TCHAR* file_path);
537 void close_movie_file();
541 double get_movie_frame_rate()
543 return movie_frame_rate;
545 int get_movie_sound_rate()
547 return movie_sound_rate;
549 void set_cur_movie_frame(int frame, bool relative);
550 uint32_t get_cur_movie_frame();
551 bool now_movie_play, now_movie_pause;
553 #ifdef USE_VIDEO_CAPTURE
554 int get_cur_capture_dev_index()
556 return cur_capture_dev_index;
558 int get_num_capture_devs()
560 return num_capture_devs;
562 _TCHAR* get_capture_dev_name(int index)
564 return capture_dev_name[index];
566 void open_capture_dev(int index, bool pin);
567 void close_capture_dev();
568 void show_capture_dev_filter();
569 void show_capture_dev_pin();
570 void show_capture_dev_source();
571 void set_capture_dev_channel(int ch);
576 void create_bitmap(bitmap_t *bitmap, int width, int height);
577 void release_bitmap(bitmap_t *bitmap);
578 void create_font(font_t *font, const _TCHAR *family, int width, int height, int rotate, bool bold, bool italic);
579 void release_font(font_t *font);
580 void create_pen(pen_t *pen, int width, uint8_t r, uint8_t g, uint8_t b);
581 void release_pen(pen_t *pen);
582 void clear_bitmap(bitmap_t *bitmap, uint8_t r, uint8_t g, uint8_t b);
583 int get_text_width(bitmap_t *bitmap, font_t *font, const char *text);
584 void draw_text_to_bitmap(bitmap_t *bitmap, font_t *font, int x, int y, const char *text, uint8_t r, uint8_t g, uint8_t b);
585 void draw_line_to_bitmap(bitmap_t *bitmap, pen_t *pen, int sx, int sy, int ex, int ey);
586 void draw_rectangle_to_bitmap(bitmap_t *bitmap, int x, int y, int width, int height, uint8_t r, uint8_t g, uint8_t b);
587 void draw_point_to_bitmap(bitmap_t *bitmap, int x, int y, uint8_t r, uint8_t g, uint8_t b);
588 void stretch_bitmap(bitmap_t *dest, int dest_x, int dest_y, int dest_width, int dest_height, bitmap_t *source, int source_x, int source_y, int source_width, int source_height);
590 void write_bitmap_to_file(bitmap_t *bitmap, const _TCHAR *file_path);
594 int get_socket(int ch)
598 void notify_socket_connected(int ch);
599 void notify_socket_disconnected(int ch);
600 void update_socket();
601 bool initialize_socket_tcp(int ch);
602 bool initialize_socket_udp(int ch);
603 bool connect_socket(int ch, uint32_t ipaddr, int port);
604 void disconnect_socket(int ch);
605 bool listen_socket(int ch);
606 void send_socket_data_tcp(int ch);
607 void send_socket_data_udp(int ch, uint32_t ipaddr, int port);
608 void send_socket_data(int ch);
609 void recv_socket_data(int ch);
613 void invalidate_screen();
614 void update_screen(HDC hdc);
615 HWND main_window_handle;
616 HINSTANCE instance_handle;