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 "../config.h"
39 #pragma comment(lib, "wsock32.lib")
41 #pragma comment(lib, "comctl32.lib")
42 #pragma comment(lib, "msimg32.lib")
43 #pragma comment(lib, "gdiplus.lib")
44 using namespace Gdiplus;
45 #pragma comment(lib, "d3d9.lib")
46 #pragma comment(lib, "d3dx9.lib")
47 #pragma comment(lib, "vfw32.lib")
48 #pragma comment(lib, "dsound.lib")
49 #if DIRECTINPUT_VERSION >= 0x0800
50 #pragma comment(lib, "dinput8.lib")
52 #pragma comment(lib, "dinput.lib")
54 #pragma comment(lib, "dxguid.lib")
56 #if defined(USE_MOVIE_PLAYER) || defined(USE_VIDEO_CAPTURE)
57 #pragma comment(lib, "strmiids.lib")
60 EXTERN_C const CLSID CLSID_SampleGrabber;
61 EXTERN_C const CLSID CLSID_NullRenderer;
62 EXTERN_C const IID IID_ISampleGrabberCB;
63 MIDL_INTERFACE("0579154A-2B53-4994-B0D0-E773148EFF85")
64 ISampleGrabberCB : public IUnknown {
66 virtual HRESULT STDMETHODCALLTYPE SampleCB( double SampleTime,IMediaSample *pSample) = 0;
67 virtual HRESULT STDMETHODCALLTYPE BufferCB( double SampleTime,BYTE *pBuffer,long BufferLen) = 0;
69 EXTERN_C const IID IID_ISampleGrabber;
70 MIDL_INTERFACE("6B652FFF-11FE-4fce-92AD-0266B5D7C78F")
71 ISampleGrabber : public IUnknown {
73 virtual HRESULT STDMETHODCALLTYPE SetOneShot( BOOL OneShot) = 0;
74 virtual HRESULT STDMETHODCALLTYPE SetMediaType( const AM_MEDIA_TYPE *pType) = 0;
75 virtual HRESULT STDMETHODCALLTYPE GetConnectedMediaType( AM_MEDIA_TYPE *pType) = 0;
76 virtual HRESULT STDMETHODCALLTYPE SetBufferSamples( BOOL BufferThem) = 0;
77 virtual HRESULT STDMETHODCALLTYPE GetCurrentBuffer( /* [out][in] */ long *pBufferSize,/* [out] */ long *pBuffer) = 0;
78 virtual HRESULT STDMETHODCALLTYPE GetCurrentSample( /* [retval][out] */ IMediaSample **ppSample) = 0;
79 virtual HRESULT STDMETHODCALLTYPE SetCallback( ISampleGrabberCB *pCallback,long WhichMethodToCallback) = 0;
82 #ifdef USE_MOVIE_PLAYER
83 class CMySampleGrabberCB : public ISampleGrabberCB {
87 CMySampleGrabberCB(VM *vm_ptr)
91 STDMETHODIMP_(ULONG) AddRef()
95 STDMETHODIMP_(ULONG) Release()
99 STDMETHODIMP QueryInterface(REFIID riid, void **ppv)
101 if(riid == IID_ISampleGrabberCB || riid == IID_IUnknown) {
102 *ppv = (void *) static_cast<ISampleGrabberCB*>(this);
105 return E_NOINTERFACE;
107 STDMETHODIMP SampleCB(double SampleTime, IMediaSample *pSample)
111 STDMETHODIMP BufferCB(double dblSampleTime, BYTE *pBuffer, long lBufferSize)
113 vm->movie_sound_callback(pBuffer, lBufferSize);
119 #define WM_RESIZE (WM_USER + 1)
120 #define WM_SOCKET0 (WM_USER + 2)
121 #define WM_SOCKET1 (WM_USER + 3)
122 #define WM_SOCKET2 (WM_USER + 4)
123 #define WM_SOCKET3 (WM_USER + 5)
127 #define SOCKET_BUFFER_MAX 0x100000
130 #ifdef USE_VIDEO_CAPTURE
131 #define MAX_CAPTURE_DEVS 8
134 #define SUPPORT_WIN32_DLL
136 // check memory leaks
138 #define _CRTDBG_MAP_ALLOC
140 #define malloc(s) _malloc_dbg(s, _NORMAL_BLOCK, __FILE__, __LINE__)
141 #define new new(_NORMAL_BLOCK, __FILE__, __LINE__)
144 typedef struct bitmap_s {
146 inline bool initialized()
148 return (hdcDib != NULL);
150 inline scrntype_t* get_buffer(int y)
152 return lpBmp + width * (height - y - 1);
157 HBITMAP hBmp, hOldBmp;
163 typedef struct font_s {
165 inline bool initialized()
167 return (hFont != NULL);
170 int width, height, rotate;
176 typedef struct pen_s {
178 inline bool initialized()
180 return (hPen != NULL);
189 PAVISTREAM pAVICompressed;
191 LPBITMAPINFOHEADER pbmInfoHeader;
196 } rec_video_thread_param_t;
207 HANDLE hStdIn, hStdOut;
210 void initialize_input();
211 void release_input();
213 #if DIRECTINPUT_VERSION >= 0x0800
215 LPDIRECTINPUTDEVICE8 lpdikey;
216 // LPDIRECTINPUTDEVICE8 lpdijoy;
219 LPDIRECTINPUTDEVICE lpdikey;
220 // LPDIRECTINPUTDEVICE lpdijoy;
222 bool dinput_key_available;
223 // bool dinput_joy_available;
225 uint8_t keycode_conv[256];
226 uint8_t key_status[256]; // windows key code mapping
227 uint8_t key_dik_prev[256];
228 #ifdef USE_SHIFT_NUMPAD_KEY
229 uint8_t key_converted[256];
230 bool key_shift_pressed, key_shift_released;
232 bool key_caps_locked;
236 uint32_t joy_status[4]; // joystick #1-#4 (b0 = up, b1 = down, b2 = left, b3 = right, b4- = buttons
238 uint32_t joy_mask[4];
242 int32_t mouse_status[3]; // x, y, button (b0 = left, b1 = right)
247 void initialize_screen();
248 void release_screen();
249 void initialize_screen_buffer(bitmap_t *buffer, int width, int height, int mode);
250 void release_screen_buffer(bitmap_t *buffer);
251 #ifdef USE_CRT_FILTER
252 void apply_crt_fileter_to_screen_buffer(bitmap_t *source, bitmap_t *dest);
253 void apply_crt_filter_x3_y3(bitmap_t *source, bitmap_t *dest);
254 void apply_crt_filter_x3_y2(bitmap_t *source, bitmap_t *dest);
255 void apply_crt_filter_x2_y3(bitmap_t *source, bitmap_t *dest);
256 void apply_crt_filter_x2_y2(bitmap_t *source, bitmap_t *dest);
257 void apply_crt_filter_x1_y1(bitmap_t *source, bitmap_t *dest);
259 #ifdef USE_SCREEN_ROTATE
260 void rotate_screen_buffer(bitmap_t *source, bitmap_t *dest);
262 void stretch_screen_buffer(bitmap_t *source, bitmap_t *dest);
263 bool initialize_d3d9();
264 bool initialize_d3d9_surface(bitmap_t *buffer);
266 void release_d3d9_surface();
267 void copy_to_d3d9_surface(bitmap_t *buffer);
268 int add_video_frames();
270 bitmap_t vm_screen_buffer;
271 #ifdef USE_CRT_FILTER
272 bitmap_t filtered_screen_buffer;
273 bitmap_t tmp_filtered_screen_buffer;
275 #ifdef USE_SCREEN_ROTATE
276 bitmap_t rotated_screen_buffer;
278 bitmap_t stretched_screen_buffer;
279 bitmap_t shrinked_screen_buffer;
280 bitmap_t video_screen_buffer;
282 bitmap_t* draw_screen_buffer;
284 int host_window_width, host_window_height;
285 bool host_window_mode;
286 int vm_screen_width, vm_screen_height;
287 int vm_window_width, vm_window_height;
288 int vm_window_width_aspect, vm_window_height_aspect;
289 int draw_screen_width, draw_screen_height;
291 Gdiplus::GdiplusStartupInput gdiSI;
295 LPDIRECT3DDEVICE9 lpd3d9Device;
296 LPDIRECT3DSURFACE9 lpd3d9Surface;
297 LPDIRECT3DSURFACE9 lpd3d9OffscreenSurface;
299 _TCHAR video_file_path[_MAX_PATH];
301 double rec_video_run_frames;
302 double rec_video_frames;
304 LPBITMAPINFO lpDibRec;
306 PAVISTREAM pAVIStream;
307 PAVISTREAM pAVICompressed;
308 AVICOMPRESSOPTIONS AVIOpts;
312 rec_video_thread_param_t rec_video_thread_param;
314 bool first_draw_screen;
315 bool first_invalidate;
316 bool self_invalidate;
319 void initialize_sound(int rate, int samples);
320 void release_sound();
322 int sound_rate, sound_samples;
323 bool sound_available, sound_started, sound_muted;
326 LPDIRECTSOUNDBUFFER lpdsPrimaryBuffer, lpdsSecondaryBuffer;
327 bool sound_first_half;
329 _TCHAR sound_file_path[_MAX_PATH];
330 FILEIO* rec_sound_fio;
332 int rec_sound_buffer_ptr;
335 #if defined(USE_MOVIE_PLAYER) || defined(USE_VIDEO_CAPTURE)
336 void initialize_video();
337 void release_video();
339 IGraphBuilder *pGraphBuilder;
340 IBaseFilter *pVideoBaseFilter;
341 IBaseFilter *pCaptureBaseFilter;
342 ICaptureGraphBuilder2 *pCaptureGraphBuilder2;
343 ISampleGrabber *pVideoSampleGrabber;
344 IBaseFilter *pSoundBaseFilter;
345 ISampleGrabber *pSoundSampleGrabber;
346 CMySampleGrabberCB *pSoundCallBack;
347 IMediaControl *pMediaControl;
348 IMediaSeeking *pMediaSeeking;
349 IMediaPosition *pMediaPosition;
350 IVideoWindow *pVideoWindow;
351 IBasicVideo *pBasicVideo;
352 IBasicAudio *pBasicAudio;
353 bool bTimeFormatFrame;
354 bool bVerticalReversed;
356 bitmap_t direct_show_screen_buffer;
357 bitmap_t direct_show_stretch_buffer;
358 int direct_show_width, direct_show_height;
359 bool direct_show_mute[2];
361 #ifdef USE_MOVIE_PLAYER
362 double movie_frame_rate;
363 int movie_sound_rate;
365 #ifdef USE_VIDEO_CAPTURE
366 void enum_capture_devs();
367 bool connect_capture_dev(int index, bool pin);
368 int cur_capture_dev_index;
369 int num_capture_devs;
370 _TCHAR capture_dev_name[MAX_CAPTURE_DEVS][256];
375 void initialize_socket();
376 void release_socket();
379 bool is_tcp[SOCKET_MAX];
380 struct sockaddr_in udpaddr[SOCKET_MAX];
381 int socket_delay[SOCKET_MAX];
382 char recv_buffer[SOCKET_MAX][SOCKET_BUFFER_MAX];
383 int recv_r_ptr[SOCKET_MAX], recv_w_ptr[SOCKET_MAX];
396 void initialize(int rate, int samples);
405 return (lock_count != 0);
407 void force_unlock_vm();
408 void sleep(uint32_t ms);
411 void open_console(const _TCHAR* title);
412 void close_console();
413 unsigned int get_console_code_page();
414 bool is_console_active();
415 void set_console_text_attribute(unsigned short attr);
416 void write_console(_TCHAR* buffer, unsigned int length);
417 int read_console_input(_TCHAR* buffer);
418 bool is_console_key_pressed(int vk);
419 void close_debugger_console();
423 void key_down(int code, bool repeat);
424 void key_up(int code);
425 void key_down_native(int code, bool repeat);
426 void key_up_native(int code);
427 void key_lost_focus()
433 void disable_mouse();
435 bool is_mouse_enabled()
437 return mouse_enabled;
440 uint8_t* get_key_buffer()
445 uint32_t* get_joy_buffer()
451 int32_t* get_mouse_buffer()
461 int get_window_mode_width(int mode);
462 int get_window_mode_height(int mode);
463 void set_host_window_size(int window_width, int window_height, bool window_mode);
464 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);
465 int get_vm_window_width()
467 return vm_window_width;
469 int get_vm_window_height()
471 return vm_window_height;
473 int get_vm_window_width_aspect()
475 return vm_window_width_aspect;
477 int get_vm_window_height_aspect()
479 return vm_window_height_aspect;
481 scrntype_t* get_vm_screen_buffer(int y);
483 #ifdef ONE_BOARD_MICRO_COMPUTER
486 first_invalidate = true;
489 void capture_screen();
490 bool start_record_video(int fps);
491 void stop_record_video();
492 void restart_record_video();
493 void add_extra_frames(int extra_frames);
494 bool now_record_video;
495 #ifdef USE_CRT_FILTER
496 bool screen_skip_line;
500 void update_sound(int* extra_frames);
503 void start_record_sound();
504 void stop_record_sound();
505 void restart_record_sound();
506 bool now_record_sound;
508 // common video device
509 #if defined(USE_MOVIE_PLAYER) || defined(USE_VIDEO_CAPTURE)
510 void get_video_buffer();
511 void mute_video_dev(bool l, bool r);
513 #ifdef USE_MOVIE_PLAYER
514 bool open_movie_file(const _TCHAR* file_path);
515 void close_movie_file();
519 double get_movie_frame_rate()
521 return movie_frame_rate;
523 int get_movie_sound_rate()
525 return movie_sound_rate;
527 void set_cur_movie_frame(int frame, bool relative);
528 uint32_t get_cur_movie_frame();
529 bool now_movie_play, now_movie_pause;
531 #ifdef USE_VIDEO_CAPTURE
532 int get_cur_capture_dev_index()
534 return cur_capture_dev_index;
536 int get_num_capture_devs()
538 return num_capture_devs;
540 _TCHAR* get_capture_dev_name(int index)
542 return capture_dev_name[index];
544 void open_capture_dev(int index, bool pin);
545 void close_capture_dev();
546 void show_capture_dev_filter();
547 void show_capture_dev_pin();
548 void show_capture_dev_source();
549 void set_capture_dev_channel(int ch);
554 void create_bitmap(bitmap_t *bitmap, int width, int height);
555 void release_bitmap(bitmap_t *bitmap);
556 void create_font(font_t *font, const _TCHAR *family, int width, int height, int rotate, bool bold, bool italic);
557 void release_font(font_t *font);
558 void create_pen(pen_t *pen, int width, uint8_t r, uint8_t g, uint8_t b);
559 void release_pen(pen_t *pen);
560 void clear_bitmap(bitmap_t *bitmap, uint8_t r, uint8_t g, uint8_t b);
561 int get_text_width(bitmap_t *bitmap, font_t *font, const char *text);
562 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);
563 void draw_line_to_bitmap(bitmap_t *bitmap, pen_t *pen, int sx, int sy, int ex, int ey);
564 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);
565 void draw_point_to_bitmap(bitmap_t *bitmap, int x, int y, uint8_t r, uint8_t g, uint8_t b);
566 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);
568 void write_bitmap_to_file(bitmap_t *bitmap, const _TCHAR *file_path);
572 int get_socket(int ch)
576 void notify_socket_connected(int ch);
577 void notify_socket_disconnected(int ch);
578 void update_socket();
579 bool initialize_socket_tcp(int ch);
580 bool initialize_socket_udp(int ch);
581 bool connect_socket(int ch, uint32_t ipaddr, int port);
582 void disconnect_socket(int ch);
583 bool listen_socket(int ch);
584 void send_socket_data_tcp(int ch);
585 void send_socket_data_udp(int ch, uint32_t ipaddr, int port);
586 void send_socket_data(int ch);
587 void recv_socket_data(int ch);
591 void update_screen(HDC hdc);
592 HWND main_window_handle;
593 HINSTANCE instance_handle;