X-Git-Url: https://notaz.gp2x.de/cgi-bin/gitweb.cgi?a=blobdiff_plain;f=src%2Fvideo%2Fomapdss%2Fsdlif.c;h=9e374ef74c65234741566d24424b3969fa03d5a1;hb=830411e055408d226f1ab6d91085342c3fd3e71b;hp=7511ee49bfc4bec451fa465aa42fb34fe8a7fad9;hpb=e1837b2c78d48d2811ff77e8702810b7620dd4f8;p=sdl_omap.git diff --git a/src/video/omapdss/sdlif.c b/src/video/omapdss/sdlif.c index 7511ee4..9e374ef 100644 --- a/src/video/omapdss/sdlif.c +++ b/src/video/omapdss/sdlif.c @@ -7,6 +7,7 @@ #include #include +#include #include "../SDL_sysvideo.h" #include "../SDL_pixels_c.h" @@ -132,21 +133,37 @@ static SDL_Surface *omap_SetVideoMode(SDL_VideoDevice *this, SDL_Surface *curren } } - doublebuf = (flags & SDL_DOUBLEBUF) ? 1 : 0; + /* always use doublebuf, when SDL_DOUBLEBUF is not set, + * we'll have to blit manually on UpdateRects() */ + doublebuf = 1; + fbmem = osdl_video_set_mode(pdata, pdata->border_l, pdata->border_r, pdata->border_t, pdata->border_b, - width, height, bpp, &doublebuf); + width, height, bpp, &doublebuf, this->wm_title); if (fbmem == NULL) { err("failing on mode %dx%d@%d, doublebuf %s, border %d,%d,%d,%d", width, height, bpp, (flags & SDL_DOUBLEBUF) ? "on" : "off", pdata->border_l, pdata->border_r, pdata->border_t, pdata->border_b); return NULL; } - if ((flags & SDL_DOUBLEBUF) && !doublebuf) { - log("doublebuffering could not be set\n"); - flags &= ~SDL_DOUBLEBUF; + pdata->front_buffer = osdl_video_get_active_buffer(pdata); + if (pdata->front_buffer == NULL) { + err("osdl_video_get_active_buffer failed\n"); + return NULL; + } + + if (!doublebuf) { + if (flags & SDL_DOUBLEBUF) { + log("doublebuffering could not be set\n"); + flags &= ~SDL_DOUBLEBUF; + } + /* XXX: could just malloc a back buffer here instead */ + pdata->cfg_force_directbuf = 1; } + if (!(flags & SDL_DOUBLEBUF) && pdata->cfg_force_directbuf) + fbmem = pdata->front_buffer; + flags |= SDL_FULLSCREEN | SDL_HWSURFACE; unhandled_flags = flags & ~(SDL_FULLSCREEN | SDL_HWSURFACE | SDL_DOUBLEBUF); if (unhandled_flags != 0) { @@ -186,10 +203,27 @@ static void omap_UnlockHWSurface(SDL_VideoDevice *this, SDL_Surface *surface) static int omap_FlipHWSurface(SDL_VideoDevice *this, SDL_Surface *surface) { struct SDL_PrivateVideoData *pdata = this->hidden; + static int warned; trace("%p", surface); - surface->pixels = osdl_video_flip(pdata); + if (surface != this->screen) { + if (!warned) { + err("flip surface %p which is not screen %p?\n", + surface, this->screen); + warned = 1; + } + return; + } + + if (surface->flags & SDL_DOUBLEBUF) + surface->pixels = osdl_video_flip(pdata); + else { + if (surface->pixels != pdata->front_buffer) + memcpy(surface->pixels, pdata->front_buffer, + surface->pitch * surface->h); + } + pdata->app_uses_flip = 1; return 0; @@ -216,15 +250,53 @@ static int omap_SetColors(SDL_VideoDevice *this, int firstcolor, int ncolors, SD static void omap_UpdateRects(SDL_VideoDevice *this, int nrects, SDL_Rect *rects) { struct SDL_PrivateVideoData *pdata = this->hidden; + SDL_Surface *screen = this->screen; + int fullscreen_blit = 0; + int i, Bpp, x, y, w, h; + char *src, *dst; trace("%d, %p", nrects, rects); - /* for doublebuf forcing on apps */ - if (nrects == 1 && rects->x == 0 && rects->y == 0 - && !pdata->app_uses_flip && (this->screen->flags & SDL_DOUBLEBUF) - && rects->w == this->screen->w && rects->h == this->screen->h) - { - this->screen->pixels = osdl_video_flip(pdata); + fullscreen_blit = + nrects == 1 && rects->x == 0 && rects->y == 0 + && (rects->w == screen->w || rects->w == 0) + && (rects->h == screen->h || rects->h == 0); + + if (screen->flags & SDL_DOUBLEBUF) { + if (fullscreen_blit && !pdata->app_uses_flip) + screen->pixels = osdl_video_flip(pdata); + return; + } + + src = screen->pixels; + dst = pdata->front_buffer; + if (src == dst) + return; + + if (fullscreen_blit) { + memcpy(dst, src, screen->pitch * screen->h); + return; + } + + for (i = 0, Bpp = screen->format->BytesPerPixel; i < nrects; i++) { + /* this supposedly has no clipping, but we'll do it anyway */ + x = rects[i].x, y = rects[i].y, w = rects[i].w, h = rects[i].h; + if (x < 0) + w += x, x = 0; + else if (x + w > screen->w) + w = screen->w - x; + if (w <= 0) + continue; + + if (y < 0) + h += y, y = 0; + else if (y + h > screen->h) + h = screen->h - y; + + for (; h > 0; y++, h--) + memcpy(dst + y * screen->pitch + x * Bpp, + src + y * screen->pitch + x * Bpp, + w * Bpp); } } @@ -233,11 +305,48 @@ static void omap_InitOSKeymap(SDL_VideoDevice *this) trace(); } -static int key_event_cb(void *cb_arg, int sdl_kc, int is_pressed) +static int key_event_cb(void *cb_arg, int sdl_kc, int sdl_sc, int is_pressed) { SDL_keysym keysym = { 0, }; + int shift = 0; + SDLMod mod; + int ret; keysym.sym = sdl_kc; + keysym.scancode = sdl_sc; + + /* 0xff if pandora's Fn, so we exclude it too.. */ + if (is_pressed && sdl_kc < 0xff && SDL_TranslateUNICODE) { + mod = SDL_GetModState(); + if (!(mod & KMOD_CTRL) && (!!(mod & KMOD_SHIFT) ^ !!(mod & KMOD_CAPS))) + shift = 1; + + /* prefer X mapping, if that doesn't work use hardcoded one */ + ret = xenv_keycode_to_keysym(sdl_sc, shift); + if (ret >= 0) { + keysym.unicode = ret; + if ((mod & KMOD_CTRL) + && 0x60 <= keysym.unicode && keysym.unicode <= 0x7f) + { + keysym.unicode -= 0x60; + } + /* hmh.. */ + if ((keysym.unicode & 0xff00) == 0xff00) + keysym.unicode &= ~0xff00; + } + else { + keysym.unicode = sdl_kc; + if ((mod & KMOD_CTRL) && 0x60 <= sdl_kc && sdl_kc <= 0x7f) + { + keysym.unicode = sdl_kc - 0x60; + } + else if (shift && 'a' <= sdl_kc && sdl_kc <= 'z') + { + keysym.unicode = sdl_kc - 'a' + 'A'; + } + } + } + SDL_PrivateKeyboard(is_pressed, &keysym); } @@ -275,7 +384,7 @@ static int ts_event_cb(void *cb_arg, int x, int y, unsigned int pressure) SDL_PrivateMouseMotion(0, 0, x, y); } -static int mouseb_event_cb(void *cb_arg, int x, int y, int button, int is_pressed) +static int xmouseb_event_cb(void *cb_arg, int x, int y, int button, int is_pressed) { SDL_VideoDevice *this = cb_arg; struct SDL_PrivateVideoData *pdata = this->hidden; @@ -284,7 +393,7 @@ static int mouseb_event_cb(void *cb_arg, int x, int y, int button, int is_presse SDL_PrivateMouseButton(is_pressed ? SDL_PRESSED : SDL_RELEASED, button, x, y); } -static int mousem_event_cb(void *cb_arg, int x, int y) +static int xmousem_event_cb(void *cb_arg, int x, int y) { SDL_VideoDevice *this = cb_arg; struct SDL_PrivateVideoData *pdata = this->hidden; @@ -293,6 +402,22 @@ static int mousem_event_cb(void *cb_arg, int x, int y) SDL_PrivateMouseMotion(0, 0, x, y); } +static int xkey_cb(void *cb_arg, int kc, int is_pressed) +{ + SDL_VideoDevice *this = cb_arg; + struct SDL_PrivateVideoData *pdata = this->hidden; + int ret; + + if (kc == XF86XK_MenuKB && is_pressed) { + ret = osdl_video_pause(pdata, 1); + if (ret == 0) { + xenv_minimize(); + osdl_video_pause(pdata, 0); + omapsdl_input_get_events(0, NULL, NULL, NULL); + } + } +} + static void omap_PumpEvents(SDL_VideoDevice *this) { struct SDL_PrivateVideoData *pdata = this->hidden; @@ -302,7 +427,7 @@ static void omap_PumpEvents(SDL_VideoDevice *this) if (pdata->xenv_up) { if (!pdata->cfg_ts_force_tslib) { - xenv_update(NULL, mouseb_event_cb, mousem_event_cb, this); + xenv_update(xkey_cb, xmouseb_event_cb, xmousem_event_cb, this); if (pdata->xenv_mouse) read_tslib = 0; }