src/video/bwindow/SDL_bframebuffer.cc
author Nathan Heisey <nathanheisey@gmail.com>
Fri, 12 Aug 2011 16:58:54 +0000
changeset 5936 116b7457e73a
parent 5934 01531d8d1e01
child 5938 b85b6326c498
permissions -rw-r--r--
Reduced SEGFAULT occurence on resize?
nathanheisey@5931
     1
/*
nathanheisey@5931
     2
  Simple DirectMedia Layer
nathanheisey@5931
     3
  Copyright (C) 1997-2011 Sam Lantinga <slouken@libsdl.org>
nathanheisey@5931
     4
nathanheisey@5931
     5
  This software is provided 'as-is', without any express or implied
nathanheisey@5931
     6
  warranty.  In no event will the authors be held liable for any damages
nathanheisey@5931
     7
  arising from the use of this software.
nathanheisey@5931
     8
nathanheisey@5931
     9
  Permission is granted to anyone to use this software for any purpose,
nathanheisey@5931
    10
  including commercial applications, and to alter it and redistribute it
nathanheisey@5931
    11
  freely, subject to the following restrictions:
nathanheisey@5931
    12
nathanheisey@5931
    13
  1. The origin of this software must not be misrepresented; you must not
nathanheisey@5931
    14
     claim that you wrote the original software. If you use this software
nathanheisey@5931
    15
     in a product, an acknowledgment in the product documentation would be
nathanheisey@5931
    16
     appreciated but is not required.
nathanheisey@5931
    17
  2. Altered source versions must be plainly marked as such, and must not be
nathanheisey@5931
    18
     misrepresented as being the original software.
nathanheisey@5931
    19
  3. This notice may not be removed or altered from any source distribution.
nathanheisey@5931
    20
*/
nathanheisey@5931
    21
nathanheisey@5931
    22
#include "SDL_bframebuffer.h"
nathanheisey@5931
    23
nathanheisey@5931
    24
#include <AppKit.h>
nathanheisey@5931
    25
#include <InterfaceKit.h>
nathanheisey@5931
    26
#include "SDL_bmodes.h"
nathanheisey@5931
    27
#include "SDL_BWin.h"
nathanheisey@5931
    28
nathanheisey@5931
    29
#include "../../main/beos/SDL_BApp.h"
nathanheisey@5931
    30
nathanheisey@5931
    31
#ifdef __cplusplus
nathanheisey@5931
    32
extern "C" {
nathanheisey@5931
    33
#endif
nathanheisey@5931
    34
nathanheisey@5936
    35
int32 BE_UpdateOnce(SDL_Window *window);
nathanheisey@5936
    36
nathanheisey@5931
    37
static inline SDL_BWin *_ToBeWin(SDL_Window *window) {
nathanheisey@5931
    38
	return ((SDL_BWin*)(window->driverdata));
nathanheisey@5931
    39
}
nathanheisey@5931
    40
nathanheisey@5931
    41
static inline SDL_BApp *_GetBeApp() {
nathanheisey@5931
    42
	return ((SDL_BApp*)be_app);
nathanheisey@5931
    43
}
nathanheisey@5931
    44
nathanheisey@5931
    45
int BE_CreateWindowFramebuffer(_THIS, SDL_Window * window,
nathanheisey@5931
    46
                                       Uint32 * format,
nathanheisey@5931
    47
                                       void ** pixels, int *pitch) {
nathanheisey@5931
    48
	SDL_BWin *bwin = _ToBeWin(window);
nathanheisey@5931
    49
	BScreen bscreen;
nathanheisey@5931
    50
	if(!bscreen.IsValid()) {
nathanheisey@5931
    51
		return -1;
nathanheisey@5931
    52
	}
nathanheisey@5931
    53
nathanheisey@5931
    54
	while(!bwin->Connected()) { snooze(100); }
nathanheisey@5931
    55
	
nathanheisey@5931
    56
	/* Make sure we have exclusive access to frame buffer data */
nathanheisey@5931
    57
	bwin->LockBuffer();
nathanheisey@5932
    58
nathanheisey@5931
    59
	/* format */
nathanheisey@5931
    60
	display_mode bmode;
nathanheisey@5931
    61
	bscreen.GetMode(&bmode);
nathanheisey@5931
    62
	int32 bpp = ColorSpaceToBitsPerPixel(bmode.space);
nathanheisey@5931
    63
	*format = BPPToSDLPxFormat(bpp);
nathanheisey@5931
    64
nathanheisey@5933
    65
	/* Create the new bitmap object */
nathanheisey@5933
    66
	BBitmap *bitmap = bwin->GetBitmap();
nathanheisey@5934
    67
nathanheisey@5933
    68
	if(bitmap) {
nathanheisey@5933
    69
		delete bitmap;
nathanheisey@5933
    70
	}
nathanheisey@5933
    71
	bitmap = new BBitmap(bwin->Bounds(), (color_space)bmode.space,
nathanheisey@5933
    72
			false,	/* Views not accepted */
nathanheisey@5933
    73
			true);	/* Contiguous memory required */
nathanheisey@5934
    74
			
nathanheisey@5934
    75
	if(bitmap->InitCheck() != B_OK) {
nathanheisey@5934
    76
		SDL_SetError("Could not initialize back buffer!\n");
nathanheisey@5934
    77
		return -1;
nathanheisey@5934
    78
	}
nathanheisey@5934
    79
nathanheisey@5934
    80
nathanheisey@5933
    81
	bwin->SetBitmap(bitmap);
nathanheisey@5933
    82
	
nathanheisey@5933
    83
	/* Set the pixel pointer */
nathanheisey@5933
    84
	*pixels = bitmap->Bits();
nathanheisey@5931
    85
nathanheisey@5933
    86
	/* pitch = width of window, in bytes */
nathanheisey@5933
    87
	*pitch = bitmap->BytesPerRow();
nathanheisey@5931
    88
nathanheisey@5931
    89
	bwin->SetBufferExists(true);
nathanheisey@5931
    90
	bwin->SetTrashBuffer(false);
nathanheisey@5931
    91
	bwin->UnlockBuffer();
nathanheisey@5931
    92
	return 0;
nathanheisey@5931
    93
}
nathanheisey@5931
    94
nathanheisey@5931
    95
nathanheisey@5931
    96
nathanheisey@5931
    97
int BE_UpdateWindowFramebuffer(_THIS, SDL_Window * window,
nathanheisey@5931
    98
                                      SDL_Rect * rects, int numrects) {
nathanheisey@5931
    99
	if(!window)
nathanheisey@5931
   100
		return 0;
nathanheisey@5931
   101
nathanheisey@5931
   102
	SDL_BWin *bwin = _ToBeWin(window);
nathanheisey@5936
   103
nathanheisey@5936
   104
#ifdef DRAWTHREAD	
nathanheisey@5931
   105
	bwin->LockBuffer();
nathanheisey@5931
   106
	bwin->SetBufferDirty(true);
nathanheisey@5931
   107
	bwin->UnlockBuffer();
nathanheisey@5936
   108
#else
nathanheisey@5936
   109
	bwin->SetBufferDirty(true);
nathanheisey@5936
   110
	BE_UpdateOnce(window);
nathanheisey@5936
   111
#endif
nathanheisey@5931
   112
nathanheisey@5931
   113
	return 0;
nathanheisey@5931
   114
}
nathanheisey@5931
   115
nathanheisey@5931
   116
int32 BE_DrawThread(void *data) {
nathanheisey@5931
   117
	SDL_BWin *bwin = (SDL_BWin*)data;
nathanheisey@5931
   118
	SDL_Window *window = _GetBeApp()->GetSDLWindow(bwin->GetID());
nathanheisey@5931
   119
	
nathanheisey@5931
   120
	BScreen bscreen;
nathanheisey@5931
   121
	if(!bscreen.IsValid()) {
nathanheisey@5931
   122
		return -1;
nathanheisey@5931
   123
	}
nathanheisey@5931
   124
nathanheisey@5931
   125
	while(bwin->ConnectionEnabled()) {
nathanheisey@5931
   126
		if( bwin->Connected() && bwin->BufferExists() && bwin->BufferIsDirty() ) {
nathanheisey@5931
   127
			bwin->LockBuffer();
nathanheisey@5933
   128
			BBitmap *bitmap = bwin->GetBitmap();
nathanheisey@5933
   129
			int32 windowPitch = bitmap->BytesPerRow();
nathanheisey@5931
   130
			int32 bufferPitch = bwin->GetRowBytes();
nathanheisey@5931
   131
			uint8 *windowpx;
nathanheisey@5931
   132
			uint8 *bufferpx;
nathanheisey@5931
   133
nathanheisey@5931
   134
			int32 BPP = bwin->GetBytesPerPx();
nathanheisey@5931
   135
			int32 windowSub = bwin->GetFbX() * BPP +
nathanheisey@5931
   136
						  bwin->GetFbY() * windowPitch;
nathanheisey@5931
   137
			clipping_rect *clips = bwin->GetClips();
nathanheisey@5931
   138
			int32 numClips = bwin->GetNumClips();
nathanheisey@5931
   139
			int i, y;
nathanheisey@5931
   140
nathanheisey@5931
   141
			/* Blit each clipping rectangle */
nathanheisey@5931
   142
			bscreen.WaitForRetrace();
nathanheisey@5931
   143
			for(i = 0; i < numClips; ++i) {
nathanheisey@5931
   144
				clipping_rect rc = clips[i];
nathanheisey@5931
   145
				/* Get addresses of the start of each clipping rectangle */
nathanheisey@5931
   146
				int32 width = clips[i].right - clips[i].left + 1;
nathanheisey@5931
   147
				int32 height = clips[i].bottom - clips[i].top + 1;
nathanheisey@5931
   148
				bufferpx = bwin->GetBufferPx() + 
nathanheisey@5931
   149
					clips[i].top * bufferPitch + clips[i].left * BPP;
nathanheisey@5934
   150
				windowpx = (uint8*)bitmap->Bits() + 
nathanheisey@5933
   151
					clips[i].top * windowPitch + clips[i].left * BPP -
nathanheisey@5933
   152
					windowSub;
nathanheisey@5931
   153
nathanheisey@5931
   154
				/* Copy each row of pixels from the window buffer into the frame
nathanheisey@5931
   155
				   buffer */
nathanheisey@5931
   156
				for(y = 0; y < height; ++y)
nathanheisey@5931
   157
				{
nathanheisey@5936
   158
nathanheisey@5931
   159
					if(bwin->CanTrashWindowBuffer()) {
nathanheisey@5931
   160
						goto escape;	/* Break out before the buffer is killed */
nathanheisey@5931
   161
					}
nathanheisey@5936
   162
nathanheisey@5934
   163
//					printf("memcpy(0x%x, 0x%x, %i) ", bufferpx, windowpx, width * BPP);
nathanheisey@5931
   164
					memcpy(bufferpx, windowpx, width * BPP);
nathanheisey@5931
   165
					bufferpx += bufferPitch;
nathanheisey@5931
   166
					windowpx += windowPitch;
nathanheisey@5931
   167
				}
nathanheisey@5934
   168
//				printf("\t-\t");
nathanheisey@5931
   169
			}
nathanheisey@5934
   170
//			printf("\n");
nathanheisey@5931
   171
			bwin->SetBufferDirty(false);
nathanheisey@5931
   172
escape:
nathanheisey@5931
   173
			bwin->UnlockBuffer();
nathanheisey@5931
   174
		} else {
nathanheisey@5931
   175
			snooze(16000);
nathanheisey@5931
   176
		}
nathanheisey@5931
   177
	}
nathanheisey@5931
   178
	
nathanheisey@5931
   179
	return B_OK;
nathanheisey@5931
   180
}
nathanheisey@5931
   181
nathanheisey@5931
   182
void BE_DestroyWindowFramebuffer(_THIS, SDL_Window * window) {
nathanheisey@5931
   183
	SDL_BWin *bwin = _ToBeWin(window);
nathanheisey@5931
   184
	
nathanheisey@5931
   185
	bwin->LockBuffer();
nathanheisey@5931
   186
	
nathanheisey@5931
   187
	/* Free and clear the window buffer */
nathanheisey@5933
   188
	BBitmap *bitmap = bwin->GetBitmap();
nathanheisey@5933
   189
	delete bitmap;
nathanheisey@5933
   190
	bwin->SetBitmap(NULL);
nathanheisey@5931
   191
	bwin->SetBufferExists(false);
nathanheisey@5931
   192
	bwin->UnlockBuffer();
nathanheisey@5931
   193
}
nathanheisey@5931
   194
nathanheisey@5936
   195
nathanheisey@5936
   196
int32 BE_UpdateOnce(SDL_Window *window) {
nathanheisey@5936
   197
	SDL_BWin *bwin = _ToBeWin(window);
nathanheisey@5936
   198
	BScreen bscreen;
nathanheisey@5936
   199
	if(!bscreen.IsValid()) {
nathanheisey@5936
   200
		return -1;
nathanheisey@5936
   201
	}
nathanheisey@5936
   202
nathanheisey@5936
   203
	if(bwin->ConnectionEnabled() && bwin->Connected()) {
nathanheisey@5936
   204
		bwin->LockBuffer();
nathanheisey@5936
   205
		int32 windowPitch = window->surface->pitch;
nathanheisey@5936
   206
		int32 bufferPitch = bwin->GetRowBytes();
nathanheisey@5936
   207
		uint8 *windowpx;
nathanheisey@5936
   208
		uint8 *bufferpx;
nathanheisey@5936
   209
nathanheisey@5936
   210
		int32 BPP = bwin->GetBytesPerPx();
nathanheisey@5936
   211
		uint8 *windowBaseAddress = (uint8*)window->surface->pixels;
nathanheisey@5936
   212
		int32 windowSub = bwin->GetFbX() * BPP +
nathanheisey@5936
   213
						  bwin->GetFbY() * windowPitch;
nathanheisey@5936
   214
		clipping_rect *clips = bwin->GetClips();
nathanheisey@5936
   215
		int32 numClips = bwin->GetNumClips();
nathanheisey@5936
   216
		int i, y;
nathanheisey@5936
   217
nathanheisey@5936
   218
		/* Blit each clipping rectangle */
nathanheisey@5936
   219
		bscreen.WaitForRetrace();
nathanheisey@5936
   220
		for(i = 0; i < numClips; ++i) {
nathanheisey@5936
   221
			clipping_rect rc = clips[i];
nathanheisey@5936
   222
			/* Get addresses of the start of each clipping rectangle */
nathanheisey@5936
   223
			int32 width = clips[i].right - clips[i].left + 1;
nathanheisey@5936
   224
			int32 height = clips[i].bottom - clips[i].top + 1;
nathanheisey@5936
   225
			bufferpx = bwin->GetBufferPx() + 
nathanheisey@5936
   226
				clips[i].top * bufferPitch + clips[i].left * BPP;
nathanheisey@5936
   227
			windowpx = windowBaseAddress + 
nathanheisey@5936
   228
				clips[i].top * windowPitch + clips[i].left * BPP - windowSub;
nathanheisey@5936
   229
nathanheisey@5936
   230
			/* Copy each row of pixels from the window buffer into the frame
nathanheisey@5936
   231
			   buffer */
nathanheisey@5936
   232
			for(y = 0; y < height; ++y)
nathanheisey@5936
   233
			{
nathanheisey@5936
   234
				memcpy(bufferpx, windowpx, width * BPP);
nathanheisey@5936
   235
				bufferpx += bufferPitch;
nathanheisey@5936
   236
				windowpx += windowPitch;
nathanheisey@5936
   237
			}
nathanheisey@5936
   238
		}
nathanheisey@5936
   239
		bwin->UnlockBuffer();
nathanheisey@5936
   240
	}
nathanheisey@5936
   241
	return 0;
nathanheisey@5936
   242
}
nathanheisey@5936
   243
nathanheisey@5931
   244
#ifdef __cplusplus
nathanheisey@5931
   245
}
nathanheisey@5931
   246
#endif