Skip to content

Commit 5ffbcd0

Browse files
committed
Distinguish between owned and borrowed references to a SDL_Surface
In many SDL APIs that return a SDL_Surface *, the surface is considered to be owned by the caller, and must be freed by the caller. However, SDL_SetVideoMode and presumably SDL_GetVideoSurface return a pointer to SDL's internal video surface, which will be freed by SDL if necessary, and must not be freed by library users. Incorrectly freeing this surface can lead to a use-after-free crash, manifesting as a test failure in t/core_video.t. Resolves: libsdl-org/sdl12-compat#305 Signed-off-by: Simon McVittie <[email protected]>
1 parent 3a84fb7 commit 5ffbcd0

File tree

3 files changed

+31
-5
lines changed

3 files changed

+31
-5
lines changed

src/Core/Video.xs

+4-2
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,8 @@
1010

1111
#include <SDL.h>
1212

13+
typedef SDL_Surface SDL_Surface_borrowed;
14+
1315
void _uinta_free(Uint16* av, int len_from_av_len)
1416
{
1517
if( av != NULL)
@@ -56,7 +58,7 @@ See: L<http:/*www.libsdl.org/cgi/docwiki.cgi/SDL_API#head-813f033ec44914f267f321
5658

5759
=cut
5860

59-
SDL_Surface *
61+
SDL_Surface_borrowed *
6062
video_get_video_surface()
6163
PREINIT:
6264
char* CLASS = "SDL::Surface";
@@ -125,7 +127,7 @@ video_video_mode_ok ( width, height, bpp, flags )
125127
RETVAL
126128

127129

128-
SDL_Surface *
130+
SDL_Surface_borrowed *
129131
video_set_video_mode ( width, height, bpp, flags )
130132
int width
131133
int height

src/helper.h

+4-3
Original file line numberDiff line numberDiff line change
@@ -58,12 +58,13 @@ void objDESTROY(SV *bag, void (* callback)(void *object))
5858
Uint32 *threadid = (Uint32*)(pointers[2]);
5959

6060
if(PERL_GET_CONTEXT == pointers[1]
61-
&& *threadid == SDL_ThreadID())
61+
&& (threadid == NULL || *threadid == SDL_ThreadID()))
6262
{
6363
pointers[0] = NULL;
64-
if(object)
64+
if(object && threadid != NULL)
6565
callback(object);
66-
safefree(threadid);
66+
if (threadid != NULL)
67+
safefree(threadid);
6768
safefree(pointers);
6869
}
6970
}

typemap

+23
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,7 @@ SDL_UserEvent * O_OBJECT
3434
SDL_QuitEvent * O_OBJECT
3535
SDL_keysym * O_OBJECT
3636
SDL_Surface * O_OBJECT
37+
SDL_Surface_borrowed * O_BORROWED
3738
SDL_SysWMmsg * T_PTR
3839
SDL_CD * O_OBJECT
3940
SDL_CDtrack * O_OBJECT
@@ -122,6 +123,17 @@ O_OBJECT
122123
XSRETURN_UNDEF;
123124
}
124125

126+
O_BORROWED
127+
if ($var) {
128+
void** pointers = malloc(3 * sizeof(void*));
129+
pointers[0] = (void*)$var;
130+
pointers[1] = (void*)PERL_GET_CONTEXT;
131+
pointers[2] = NULL;
132+
sv_setref_pv( $arg, CLASS, (void*)pointers );
133+
} else {
134+
XSRETURN_UNDEF;
135+
}
136+
125137
INPUT
126138

127139
O_OBJECT_NPGC
@@ -136,3 +148,14 @@ O_OBJECT
136148
} else {
137149
XSRETURN_UNDEF;
138150
}
151+
152+
O_BORROWED
153+
/* Same as O_OBJECT */
154+
if( sv_isobject($arg) && (SvTYPE(SvRV($arg)) == SVt_PVMG) ) {
155+
void** pointers = (void**)INT2PTR(void *, SvIV((SV *)SvRV( $arg )));
156+
$var = ($type)(pointers[0]);
157+
} else if ($arg == 0) {
158+
XSRETURN(0);
159+
} else {
160+
XSRETURN_UNDEF;
161+
}

0 commit comments

Comments
 (0)