src/video/haiku/SDL_bframebuffer.cc
author Sam Lantinga <slouken@libsdl.org>
Fri, 09 Dec 2016 01:47:43 -0800
changeset 10690 23a825f341e6
parent 9998 f67cf37e9cd4
child 10737 3406a0f8b041
permissions -rw-r--r--
Fixed bug 3513 - SDL_GL_SwapWindow does not return error status

Return an error code from SDL_GL_SwapWindow(), like the other SDL APIs.
nathanheisey@5931
     1
/*
nathanheisey@5931
     2
  Simple DirectMedia Layer
slouken@9998
     3
  Copyright (C) 1997-2016 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
*/
icculus@8093
    21
#include "../../SDL_internal.h"
slouken@6044
    22
icculus@7981
    23
#if SDL_VIDEO_DRIVER_HAIKU
nathanheisey@5931
    24
nathanheisey@5931
    25
#include "SDL_bframebuffer.h"
nathanheisey@5931
    26
nathanheisey@5931
    27
#include <AppKit.h>
nathanheisey@5931
    28
#include <InterfaceKit.h>
nathanheisey@5931
    29
#include "SDL_bmodes.h"
nathanheisey@5931
    30
#include "SDL_BWin.h"
nathanheisey@5931
    31
icculus@7981
    32
#include "../../main/haiku/SDL_BApp.h"
nathanheisey@5931
    33
nathanheisey@5931
    34
#ifdef __cplusplus
nathanheisey@5931
    35
extern "C" {
nathanheisey@5931
    36
#endif
nathanheisey@5931
    37
nathanheisey@5936
    38
int32 BE_UpdateOnce(SDL_Window *window);
nathanheisey@5936
    39
slouken@7860
    40
static SDL_INLINE SDL_BWin *_ToBeWin(SDL_Window *window) {
nathanheisey@5931
    41
	return ((SDL_BWin*)(window->driverdata));
nathanheisey@5931
    42
}
nathanheisey@5931
    43
slouken@7860
    44
static SDL_INLINE SDL_BApp *_GetBeApp() {
nathanheisey@5931
    45
	return ((SDL_BApp*)be_app);
nathanheisey@5931
    46
}
nathanheisey@5931
    47
nathanheisey@5931
    48
int BE_CreateWindowFramebuffer(_THIS, SDL_Window * window,
nathanheisey@5931
    49
                                       Uint32 * format,
nathanheisey@5931
    50
                                       void ** pixels, int *pitch) {
nathanheisey@5931
    51
	SDL_BWin *bwin = _ToBeWin(window);
nathanheisey@5931
    52
	BScreen bscreen;
nathanheisey@5931
    53
	if(!bscreen.IsValid()) {
nathanheisey@5931
    54
		return -1;
nathanheisey@5931
    55
	}
nathanheisey@5931
    56
nathanheisey@5931
    57
	while(!bwin->Connected()) { snooze(100); }
nathanheisey@5931
    58
	
nathanheisey@5931
    59
	/* Make sure we have exclusive access to frame buffer data */
nathanheisey@5931
    60
	bwin->LockBuffer();
nathanheisey@5932
    61
nathanheisey@5931
    62
	/* format */
nathanheisey@5931
    63
	display_mode bmode;
nathanheisey@5931
    64
	bscreen.GetMode(&bmode);
nathanheisey@5940
    65
	int32 bpp = BE_ColorSpaceToBitsPerPixel(bmode.space);
nathanheisey@5940
    66
	*format = BE_BPPToSDLPxFormat(bpp);
nathanheisey@5931
    67
nathanheisey@5933
    68
	/* Create the new bitmap object */
nathanheisey@5933
    69
	BBitmap *bitmap = bwin->GetBitmap();
nathanheisey@5934
    70
nathanheisey@5933
    71
	if(bitmap) {
nathanheisey@5933
    72
		delete bitmap;
nathanheisey@5933
    73
	}
nathanheisey@5933
    74
	bitmap = new BBitmap(bwin->Bounds(), (color_space)bmode.space,
nathanheisey@5933
    75
			false,	/* Views not accepted */
nathanheisey@5933
    76
			true);	/* Contiguous memory required */
nathanheisey@5934
    77
			
nathanheisey@5934
    78
	if(bitmap->InitCheck() != B_OK) {
icculus@7037
    79
		return SDL_SetError("Could not initialize back buffer!\n");
nathanheisey@5934
    80
	}
nathanheisey@5934
    81
nathanheisey@5934
    82
nathanheisey@5933
    83
	bwin->SetBitmap(bitmap);
nathanheisey@5933
    84
	
nathanheisey@5933
    85
	/* Set the pixel pointer */
nathanheisey@5933
    86
	*pixels = bitmap->Bits();
nathanheisey@5931
    87
nathanheisey@5933
    88
	/* pitch = width of window, in bytes */
nathanheisey@5933
    89
	*pitch = bitmap->BytesPerRow();
nathanheisey@5931
    90
nathanheisey@5931
    91
	bwin->SetBufferExists(true);
nathanheisey@5931
    92
	bwin->SetTrashBuffer(false);
nathanheisey@5931
    93
	bwin->UnlockBuffer();
nathanheisey@5931
    94
	return 0;
nathanheisey@5931
    95
}
nathanheisey@5931
    96
nathanheisey@5931
    97
nathanheisey@5931
    98
nathanheisey@5931
    99
int BE_UpdateWindowFramebuffer(_THIS, SDL_Window * window,
slouken@7014
   100
                                      const SDL_Rect * rects, int numrects) {
nathanheisey@5931
   101
	if(!window)
nathanheisey@5931
   102
		return 0;
nathanheisey@5931
   103
nathanheisey@5931
   104
	SDL_BWin *bwin = _ToBeWin(window);
nathanheisey@5936
   105
nathanheisey@5936
   106
#ifdef DRAWTHREAD	
nathanheisey@5931
   107
	bwin->LockBuffer();
nathanheisey@5931
   108
	bwin->SetBufferDirty(true);
nathanheisey@5931
   109
	bwin->UnlockBuffer();
nathanheisey@5936
   110
#else
nathanheisey@5936
   111
	bwin->SetBufferDirty(true);
nathanheisey@5936
   112
	BE_UpdateOnce(window);
nathanheisey@5936
   113
#endif
nathanheisey@5931
   114
nathanheisey@5931
   115
	return 0;
nathanheisey@5931
   116
}
nathanheisey@5931
   117
nathanheisey@5931
   118
int32 BE_DrawThread(void *data) {
nathanheisey@5931
   119
	SDL_BWin *bwin = (SDL_BWin*)data;
nathanheisey@5931
   120
	
nathanheisey@5931
   121
	BScreen bscreen;
nathanheisey@5931
   122
	if(!bscreen.IsValid()) {
nathanheisey@5931
   123
		return -1;
nathanheisey@5931
   124
	}
nathanheisey@5931
   125
nathanheisey@5931
   126
	while(bwin->ConnectionEnabled()) {
nathanheisey@5931
   127
		if( bwin->Connected() && bwin->BufferExists() && bwin->BufferIsDirty() ) {
nathanheisey@5931
   128
			bwin->LockBuffer();
nathanheisey@5938
   129
			BBitmap *bitmap = NULL;
nathanheisey@5939
   130
			bitmap = bwin->GetBitmap();
nathanheisey@5933
   131
			int32 windowPitch = bitmap->BytesPerRow();
nathanheisey@5931
   132
			int32 bufferPitch = bwin->GetRowBytes();
nathanheisey@5931
   133
			uint8 *windowpx;
nathanheisey@5931
   134
			uint8 *bufferpx;
nathanheisey@5931
   135
nathanheisey@5931
   136
			int32 BPP = bwin->GetBytesPerPx();
nathanheisey@5931
   137
			int32 windowSub = bwin->GetFbX() * BPP +
nathanheisey@5931
   138
						  bwin->GetFbY() * windowPitch;
nathanheisey@5931
   139
			clipping_rect *clips = bwin->GetClips();
nathanheisey@5931
   140
			int32 numClips = bwin->GetNumClips();
nathanheisey@5931
   141
			int i, y;
nathanheisey@5931
   142
nathanheisey@5931
   143
			/* Blit each clipping rectangle */
nathanheisey@5931
   144
			bscreen.WaitForRetrace();
nathanheisey@5931
   145
			for(i = 0; i < numClips; ++i) {
nathanheisey@5931
   146
				clipping_rect rc = clips[i];
nathanheisey@5931
   147
				/* Get addresses of the start of each clipping rectangle */
nathanheisey@5931
   148
				int32 width = clips[i].right - clips[i].left + 1;
nathanheisey@5931
   149
				int32 height = clips[i].bottom - clips[i].top + 1;
nathanheisey@5931
   150
				bufferpx = bwin->GetBufferPx() + 
nathanheisey@5931
   151
					clips[i].top * bufferPitch + clips[i].left * BPP;
nathanheisey@5934
   152
				windowpx = (uint8*)bitmap->Bits() + 
nathanheisey@5933
   153
					clips[i].top * windowPitch + clips[i].left * BPP -
nathanheisey@5933
   154
					windowSub;
nathanheisey@5931
   155
nathanheisey@5931
   156
				/* Copy each row of pixels from the window buffer into the frame
nathanheisey@5931
   157
				   buffer */
nathanheisey@5931
   158
				for(y = 0; y < height; ++y)
nathanheisey@5931
   159
				{
nathanheisey@5936
   160
nathanheisey@5931
   161
					if(bwin->CanTrashWindowBuffer()) {
nathanheisey@5931
   162
						goto escape;	/* Break out before the buffer is killed */
nathanheisey@5931
   163
					}
nathanheisey@5936
   164
nathanheisey@5931
   165
					memcpy(bufferpx, windowpx, width * BPP);
nathanheisey@5931
   166
					bufferpx += bufferPitch;
nathanheisey@5931
   167
					windowpx += windowPitch;
nathanheisey@5931
   168
				}
nathanheisey@5931
   169
			}
nathanheisey@5939
   170
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@5938
   196
/*
nathanheisey@5938
   197
 * TODO:
nathanheisey@5938
   198
 * This was written to test if certain errors were caused by threading issues.
nathanheisey@5938
   199
 * The specific issues have since become rare enough that they may have been
nathanheisey@5938
   200
 * solved, but I doubt it- they were pretty sporadic before now.
nathanheisey@5938
   201
 */
nathanheisey@5936
   202
int32 BE_UpdateOnce(SDL_Window *window) {
nathanheisey@5936
   203
	SDL_BWin *bwin = _ToBeWin(window);
nathanheisey@5936
   204
	BScreen bscreen;
nathanheisey@5936
   205
	if(!bscreen.IsValid()) {
nathanheisey@5936
   206
		return -1;
nathanheisey@5936
   207
	}
nathanheisey@5936
   208
nathanheisey@5936
   209
	if(bwin->ConnectionEnabled() && bwin->Connected()) {
nathanheisey@5936
   210
		bwin->LockBuffer();
nathanheisey@5936
   211
		int32 windowPitch = window->surface->pitch;
nathanheisey@5936
   212
		int32 bufferPitch = bwin->GetRowBytes();
nathanheisey@5936
   213
		uint8 *windowpx;
nathanheisey@5936
   214
		uint8 *bufferpx;
nathanheisey@5936
   215
nathanheisey@5936
   216
		int32 BPP = bwin->GetBytesPerPx();
nathanheisey@5936
   217
		uint8 *windowBaseAddress = (uint8*)window->surface->pixels;
nathanheisey@5936
   218
		int32 windowSub = bwin->GetFbX() * BPP +
nathanheisey@5936
   219
						  bwin->GetFbY() * windowPitch;
nathanheisey@5936
   220
		clipping_rect *clips = bwin->GetClips();
nathanheisey@5936
   221
		int32 numClips = bwin->GetNumClips();
nathanheisey@5936
   222
		int i, y;
nathanheisey@5936
   223
nathanheisey@5936
   224
		/* Blit each clipping rectangle */
nathanheisey@5936
   225
		bscreen.WaitForRetrace();
nathanheisey@5936
   226
		for(i = 0; i < numClips; ++i) {
nathanheisey@5936
   227
			clipping_rect rc = clips[i];
nathanheisey@5936
   228
			/* Get addresses of the start of each clipping rectangle */
nathanheisey@5936
   229
			int32 width = clips[i].right - clips[i].left + 1;
nathanheisey@5936
   230
			int32 height = clips[i].bottom - clips[i].top + 1;
nathanheisey@5936
   231
			bufferpx = bwin->GetBufferPx() + 
nathanheisey@5936
   232
				clips[i].top * bufferPitch + clips[i].left * BPP;
nathanheisey@5936
   233
			windowpx = windowBaseAddress + 
nathanheisey@5936
   234
				clips[i].top * windowPitch + clips[i].left * BPP - windowSub;
nathanheisey@5936
   235
nathanheisey@5936
   236
			/* Copy each row of pixels from the window buffer into the frame
nathanheisey@5936
   237
			   buffer */
nathanheisey@5936
   238
			for(y = 0; y < height; ++y)
nathanheisey@5936
   239
			{
nathanheisey@5936
   240
				memcpy(bufferpx, windowpx, width * BPP);
nathanheisey@5936
   241
				bufferpx += bufferPitch;
nathanheisey@5936
   242
				windowpx += windowPitch;
nathanheisey@5936
   243
			}
nathanheisey@5936
   244
		}
nathanheisey@5936
   245
		bwin->UnlockBuffer();
nathanheisey@5936
   246
	}
nathanheisey@5936
   247
	return 0;
nathanheisey@5936
   248
}
nathanheisey@5936
   249
nathanheisey@5931
   250
#ifdef __cplusplus
nathanheisey@5931
   251
}
nathanheisey@5931
   252
#endif
slouken@6044
   253
icculus@7981
   254
#endif /* SDL_VIDEO_DRIVER_HAIKU */
slouken@10690
   255
slouken@10690
   256
/* vi: set ts=4 sw=4 expandtab: */