diff options
Diffstat (limited to 'src/wx')
-rw-r--r-- | src/wx/main.cpp | 306 | ||||
-rw-r--r-- | src/wx/main.h | 28 |
2 files changed, 229 insertions, 105 deletions
diff --git a/src/wx/main.cpp b/src/wx/main.cpp index b195636..b4813af 100644 --- a/src/wx/main.cpp +++ b/src/wx/main.cpp @@ -19,26 +19,15 @@ #include "../types.h" #include "main.h" #include "../version.h" +#include "../Config.h" #include "../NDS.h" #include "../GPU.h" -const int DefaultKeyMapping[18] = -{ - 15, 54, - 44, 40, - 22, 4, 26, 29, - 19, 20, - 0, 0, 0, 0, 0, 0, - 18, 14 -}; - -int KeyMapping[18]; -int JoyMapping[18]; - bool Touching; int WindowX, WindowY; +int WindowW, WindowH; wxIMPLEMENT_APP(wxApp_melonDS); @@ -54,6 +43,8 @@ bool wxApp_melonDS::OnInit() printf("melonDS " MELONDS_VERSION "\n" MELONDS_URL "\n"); + Config::Load(); + MainFrame* melon = new MainFrame(); melon->Show(true); @@ -62,13 +53,18 @@ bool wxApp_melonDS::OnInit() wxBEGIN_EVENT_TABLE(MainFrame, wxFrame) + EVT_CLOSE(MainFrame::OnClose) + EVT_MENU(ID_OPENROM, MainFrame::OnOpenROM) + EVT_MENU(ID_EXIT, MainFrame::OnCloseFromMenu) + EVT_PAINT(MainFrame::OnPaint) + EVT_IDLE(MainFrame::OnIdle) wxEND_EVENT_TABLE() MainFrame::MainFrame() - : wxFrame(NULL, wxID_ANY, "melonDS") + : wxFrame(NULL, wxID_ANY, "melonDS " MELONDS_VERSION) { wxMenu* filemenu = new wxMenu(); filemenu->Append(ID_OPENROM, "Open ROM..."); @@ -81,18 +77,29 @@ MainFrame::MainFrame() systemmenu->AppendSeparator(); systemmenu->Append(ID_RESET, "Reset"); + wxMenu* settingsmenu = new wxMenu(); + settingsmenu->Append(ID_INPUTCONFIG, "Input"); + wxMenuBar* melonbar = new wxMenuBar(); melonbar->Append(filemenu, "File"); melonbar->Append(systemmenu, "System"); + melonbar->Append(settingsmenu, "Settings"); SetMenuBar(melonbar); SetClientSize(256, 384); + SetMinSize(GetSize()); + + emustatus = 2; + + emustatuschangemutex = new wxMutex(); + emustatuschange = new wxCondition(*emustatuschangemutex); - emumutex = new wxMutex(); - emucond = new wxCondition(*emumutex); + emustopmutex = new wxMutex(); + emustop = new wxCondition(*emustopmutex); + emustopmutex->Lock(); - emuthread = new EmuThread(this, emumutex, emucond); + emuthread = new EmuThread(this); if (emuthread->Run() != wxTHREAD_NO_ERROR) { printf("thread shat itself :( giving up now\n"); @@ -102,17 +109,53 @@ MainFrame::MainFrame() sdlwin = SDL_CreateWindowFrom(GetHandle()); - sdlrend = SDL_CreateRenderer(sdlwin, -1, SDL_RENDERER_ACCELERATED); // SDL_RENDERER_PRESENTVSYNC + sdlrend = SDL_CreateRenderer(sdlwin, -1, SDL_RENDERER_ACCELERATED);// | SDL_RENDERER_PRESENTVSYNC); sdltex = SDL_CreateTexture(sdlrend, SDL_PIXELFORMAT_RGBA32, SDL_TEXTUREACCESS_STREAMING, 256, 384); - NDS::Init(); + texmutex = new wxMutex(); + SDL_LockTexture(sdltex, NULL, &texpixels, &texstride); + texmutex->Unlock(); - memcpy(KeyMapping, DefaultKeyMapping, 18*sizeof(int)); - memset(JoyMapping, 0, 18*sizeof(int)); + NDS::Init(); Touching = false; SDL_GetWindowPosition(sdlwin, &WindowX, &WindowY); + SDL_GetWindowSize(sdlwin, &WindowW, &WindowH); +} + +void MainFrame::OnClose(wxCloseEvent& event) +{ + emustatus = 0; + emustatuschangemutex->Lock(); + emustatuschange->Signal(); + emustatuschangemutex->Unlock(); + + emuthread->Wait(); + delete emuthread; + delete emustatuschange; + delete emustatuschangemutex; + delete emustop; + delete emustopmutex; + + NDS::DeInit(); + + SDL_UnlockTexture(sdltex); + delete texmutex; + + SDL_DestroyTexture(sdltex); + SDL_DestroyRenderer(sdlrend); + + SDL_DestroyWindow(sdlwin); + + SDL_Quit(); + + Destroy(); +} + +void MainFrame::OnCloseFromMenu(wxCommandEvent& event) +{ + Close(); } void MainFrame::OnOpenROM(wxCommandEvent& event) @@ -121,35 +164,118 @@ void MainFrame::OnOpenROM(wxCommandEvent& event) if (opener.ShowModal() == wxID_CANCEL) return; + if (emustatus == 1) + { + emustatus = 2; + emustop->Wait(); + } + wxString filename = opener.GetPath(); NDS::LoadROM(filename.mb_str(), true); - emuthread->EmuStatus = 1; - emumutex->Lock(); - emucond->Signal(); - emumutex->Unlock(); + emustatus = 1; + emustatuschangemutex->Lock(); + emustatuschange->Signal(); + emustatuschangemutex->Unlock(); +} + +void MainFrame::ProcessSDLEvents() +{ + bool running = (emustatus == 1); + SDL_Event evt; + + while (SDL_PollEvent(&evt)) + { + switch (evt.type) + { + case SDL_WINDOWEVENT: + if (evt.window.event != SDL_WINDOWEVENT_EXPOSED) + { + SDL_GetWindowPosition(sdlwin, &WindowX, &WindowY); + SDL_GetWindowSize(sdlwin, &WindowW, &WindowH); + } + break; + + case SDL_MOUSEBUTTONDOWN: + if (!running) return; + if (evt.button.y >= 192 && evt.button.button == SDL_BUTTON_LEFT) + { + Touching = true; + NDS::PressKey(16+6); + } + break; + + case SDL_KEYDOWN: + if (!running) return; + for (int i = 0; i < 10; i++) + if (evt.key.keysym.scancode == Config::KeyMapping[i]) NDS::PressKey(i); + if (evt.key.keysym.scancode == Config::KeyMapping[10]) NDS::PressKey(16); + if (evt.key.keysym.scancode == Config::KeyMapping[11]) NDS::PressKey(17); + break; + + case SDL_KEYUP: + if (!running) return; + for (int i = 0; i < 10; i++) + if (evt.key.keysym.scancode == Config::KeyMapping[i]) NDS::ReleaseKey(i); + if (evt.key.keysym.scancode == Config::KeyMapping[10]) NDS::ReleaseKey(16); + if (evt.key.keysym.scancode == Config::KeyMapping[11]) NDS::ReleaseKey(17); + break; + } + } + + if (Touching) + { + int mx, my; + u32 btn = SDL_GetGlobalMouseState(&mx, &my); + if (!(btn & SDL_BUTTON(SDL_BUTTON_LEFT))) + { + Touching = false; + NDS::ReleaseKey(16+6); + NDS::ReleaseScreen(); + } + else + { + mx -= WindowX; + my -= (WindowY + 192); + + if (mx < 0) mx = 0; + else if (mx > 255) mx = 255; + + if (my < 0) my = 0; + else if (my > 191) my = 191; + + NDS::TouchScreen(mx, my); + } + } } void MainFrame::OnPaint(wxPaintEvent& event) { - /*wxPaintDC dc(this); - wxGraphicsContext* gc = wxGraphicsContext::Create(dc); - if (!gc) return; + wxPaintDC dc(this); - // + texmutex->Lock(); + SDL_UnlockTexture(sdltex); - delete gc;*/ + //SDL_RenderClear(sdlrend); + SDL_RenderCopy(sdlrend, sdltex, NULL, NULL); + SDL_RenderPresent(sdlrend); + + SDL_LockTexture(sdltex, NULL, &texpixels, &texstride); + texmutex->Unlock(); + + ProcessSDLEvents(); +} + +void MainFrame::OnIdle(wxIdleEvent& event) +{ + ProcessSDLEvents(); } -EmuThread::EmuThread(MainFrame* parent, wxMutex* mutex, wxCondition* cond) +EmuThread::EmuThread(MainFrame* parent) : wxThread(wxTHREAD_JOINABLE) { this->parent = parent; - this->mutex = mutex; - this->cond = cond; - - EmuStatus = 2; } EmuThread::~EmuThread() @@ -158,87 +284,75 @@ EmuThread::~EmuThread() wxThread::ExitCode EmuThread::Entry() { - mutex->Lock(); + parent->emustatuschangemutex->Lock(); for (;;) { - cond->Wait(); - - if (EmuStatus == 0) - break; + parent->emustatuschange->Wait(); - while (EmuStatus == 1) + if (parent->emustatus == 1) { - NDS::RunFrame(); + u32 nframes = 0; + u32 lasttick = SDL_GetTicks(); + u32 fpslimitcount = 0; - SDL_Event evt; - while (SDL_PollEvent(&evt)) + while (parent->emustatus == 1) { - switch (evt.type) - { - case SDL_WINDOWEVENT: - SDL_GetWindowPosition(parent->sdlwin, &WindowX, &WindowY); - break; + u32 starttick = SDL_GetTicks(); - case SDL_MOUSEBUTTONDOWN: - if (evt.button.y >= 192 && evt.button.button == SDL_BUTTON_LEFT) - { - Touching = true; - NDS::PressKey(16+6); - } - break; - - case SDL_KEYDOWN: - for (int i = 0; i < 10; i++) - if (evt.key.keysym.scancode == KeyMapping[i]) NDS::PressKey(i); - if (evt.key.keysym.scancode == KeyMapping[16]) NDS::PressKey(16); - if (evt.key.keysym.scancode == KeyMapping[17]) NDS::PressKey(17); - break; - - case SDL_KEYUP: - for (int i = 0; i < 10; i++) - if (evt.key.keysym.scancode == KeyMapping[i]) NDS::ReleaseKey(i); - if (evt.key.keysym.scancode == KeyMapping[16]) NDS::ReleaseKey(16); - if (evt.key.keysym.scancode == KeyMapping[17]) NDS::ReleaseKey(17); - break; - } - } + NDS::RunFrame(); - if (Touching) - { - int mx, my; - u32 btn = SDL_GetGlobalMouseState(&mx, &my); - if (!(btn & SDL_BUTTON(SDL_BUTTON_LEFT))) + parent->texmutex->Lock(); + + if (parent->texstride == 256*4) { - Touching = false; - NDS::ReleaseKey(16+6); - NDS::ReleaseScreen(); + memcpy(parent->texpixels, GPU::Framebuffer, 256*384*4); } else { - mx -= WindowX; - my -= (WindowY + 192); + int dsty = 0; + for (int y = 0; y < 256*384; y+=256) + { + memcpy(&((u8*)parent->texpixels)[dsty], &GPU::Framebuffer[y], 256*4); + dsty += parent->texstride; + } + } + + parent->texmutex->Unlock(); + parent->Refresh(); + + fpslimitcount++; + if (fpslimitcount >= 3) fpslimitcount = 0; + u32 frametime = (fpslimitcount == 0) ? 16 : 17; - if (mx < 0) mx = 0; - else if (mx > 255) mx = 255; + u32 endtick = SDL_GetTicks(); + u32 diff = endtick - starttick; + if (diff < frametime) + Sleep(frametime - diff); - if (my < 0) my = 0; - else if (my > 191) my = 191; + nframes++; + if (nframes >= 30) + { + u32 tick = SDL_GetTicks(); + u32 diff = tick - lasttick; + lasttick = tick; + + u32 fps = (nframes * 1000) / diff; + nframes = 0; - NDS::TouchScreen(mx, my); + char melontitle[100]; + sprintf(melontitle, "%d FPS - melonDS " MELONDS_VERSION, fps); + parent->SetTitle(melontitle); } } - void* pixels; int zorp; - SDL_LockTexture(parent->sdltex, NULL, &pixels, &zorp); - memcpy(pixels, GPU::Framebuffer, 256*384*4); - SDL_UnlockTexture(parent->sdltex); - - SDL_SetRenderTarget(parent->sdlrend, parent->sdltex); - SDL_RenderClear(parent->sdlrend); - SDL_RenderCopy(parent->sdlrend, parent->sdltex, NULL, NULL); - SDL_RenderPresent(parent->sdlrend); + parent->emustopmutex->Lock(); + parent->emustop->Signal(); + parent->emustopmutex->Unlock(); } + + if (parent->emustatus == 0) + break; } return (wxThread::ExitCode)0; diff --git a/src/wx/main.h b/src/wx/main.h index 0ee41be..f60e381 100644 --- a/src/wx/main.h +++ b/src/wx/main.h @@ -34,6 +34,8 @@ enum ID_RUN, ID_PAUSE, ID_RESET, + + ID_INPUTCONFIG, }; class EmuThread; @@ -53,32 +55,40 @@ public: SDL_Renderer* sdlrend; SDL_Texture* sdltex; + wxMutex* texmutex; + void* texpixels; + int texstride; + + int emustatus; + EmuThread* emuthread; + wxMutex* emustatuschangemutex; + wxCondition* emustatuschange; + wxMutex* emustopmutex; + wxCondition* emustop; + private: wxDECLARE_EVENT_TABLE(); + void OnClose(wxCloseEvent& event); + void OnCloseFromMenu(wxCommandEvent& event); void OnOpenROM(wxCommandEvent& event); - void OnPaint(wxPaintEvent& event); + void ProcessSDLEvents(); - EmuThread* emuthread; - wxMutex* emumutex; - wxCondition* emucond; + void OnPaint(wxPaintEvent& event); + void OnIdle(wxIdleEvent& event); }; class EmuThread : public wxThread { public: - EmuThread(MainFrame* parent, wxMutex* mutex, wxCondition* cond); + EmuThread(MainFrame* parent); ~EmuThread(); - u32 EmuStatus; - protected: virtual ExitCode Entry(); MainFrame* parent; - wxMutex* mutex; - wxCondition* cond; }; #endif // WX_MAIN_H |