CC/GCC 但不是 G++ (C/SDL2/Linux) 的奇数分段错误
Odd segmentation fault with CC/GCC but not G++ (C/SDL2/Linux)
发布的代码是直接从流行的 SDL2 教程的示例中复制的,以确保不是我犯了一些愚蠢的错误。我对这个例子所做的就是改变有问题的图像文件的路径,我将 bool 类型更改为 int,将 false 更改为 0,将 true 更改为 1。据我所知,不应保留任何特定于 C++ 的内容。
无论我做什么,一切似乎都正常,但是当用 CC/GCC 编译时(我想这真的是一样的)我最后遇到了一个分段错误,我怀疑是在 close() 中,它我一直无法确定。用 G++ 编译以某种方式防止分段错误。
解决办法当然很简单,用G++就可以了,但是我很想知道问题出在哪里。
main.c:
//Using SDL and standard IO
#include <SDL2/SDL.h>
#include <stdio.h>
//Screen dimension constants
const int SCREEN_WIDTH = 640;
const int SCREEN_HEIGHT = 480;
//Starts up SDL and creates window
int init();
//Loads media
int loadMedia();
//Frees media and shuts down SDL
void close();
//The window we'll be rendering to
SDL_Window* gWindow = NULL;
//The surface contained by the window
SDL_Surface* gScreenSurface = NULL;
//The image we will load and show on the screen
SDL_Surface* gHelloWorld = NULL;
int init()
{
//Initialization flag
int success = 1;
//Initialize SDL
if( SDL_Init( SDL_INIT_VIDEO ) < 0 )
{
printf( "SDL could not initialize! SDL_Error: %s\n", SDL_GetError() );
success = 0;
}
else
{
//Create window
gWindow = SDL_CreateWindow( "SDL Tutorial", SDL_WINDOWPOS_UNDEFINED,
SDL_WINDOWPOS_UNDEFINED, SCREEN_WIDTH,
SCREEN_HEIGHT, SDL_WINDOW_SHOWN );
if( gWindow == NULL )
{
printf( "Window could not be created! SDL_Error: %s\n",
SDL_GetError() );
success = 0;
}
else
{
//Get window surface
gScreenSurface = SDL_GetWindowSurface( gWindow );
}
}
return success;
}
int loadMedia()
{
//Loading success flag
int success = 1;
//Load splash image
gHelloWorld = SDL_LoadBMP( "hello_world.bmp" );
if( gHelloWorld == NULL )
{
printf( "Unable to load image %s! SDL Error: %s\n", "hello_world.bmp",
SDL_GetError() );
success = 0;
}
return success;
}
void close()
{
//Deallocate surface
SDL_FreeSurface( gHelloWorld );
gHelloWorld = NULL;
//Destroy window
SDL_DestroyWindow( gWindow );
gWindow = NULL;
//Quit SDL subsystems
SDL_Quit();
}
int main( int argc, char* args[] )
{
//Start up SDL and create window
if( !init() )
{
printf( "Failed to initialize!\n" );
}
else
{
//Load media
if( !loadMedia() )
{
printf( "Failed to load media!\n" );
}
else
{
//Apply the image
SDL_BlitSurface( gHelloWorld, NULL, gScreenSurface, NULL );
//Update the surface
SDL_UpdateWindowSurface( gWindow );
//Wait two seconds
SDL_Delay( 2000 );
}
}
//Free resources and close SDL
close();
return 0;
}
我看不出它有什么相关性,但为了安全起见,这是我正在使用的标准 Makefile
#OBJS specifies which files to compile as part of the project
OBJS = main.c
#CC specifies which compiler we're using
CC = cc
#COMPILER_FLAGS specifies the additional compilation options we're using
# -w suppresses all warnings
COMPILER_FLAGS = -Wall -Wextra -pedantic
#LINKER_FLAGS specifies the libraries we're linking against
LINKER_FLAGS = -lSDL2
#OBJ_NAME specifies the name of our exectuable
OBJ_NAME = test
#This is the target that compiles our executable
all : $(OBJS)
$(CC) $(OBJS) $(COMPILER_FLAGS) $(LINKER_FLAGS) -o $(OBJ_NAME)
我没有收到任何错误或警告,但未使用 argv 和 argc。
此时我已无路可走,所以我在这里问。
此致。
注意:
我应该提到的是,无论是什么问题,它绝对不是硬件问题,各种搜索结果都表明,因为我在两个完全不同的硬件平台上得到了完全相同的结果和问题。
如果重命名函数 close()
,段错误就会消失,看起来 init()
调用 SDL,调用 X11,调用 X11,调用驱动程序,驱动程序调用 close()
,但不是调用右边的 close()
它会调用你的。在 C++ 中,函数的名称会被改成其他名称,所以这不是问题。
发布的代码是直接从流行的 SDL2 教程的示例中复制的,以确保不是我犯了一些愚蠢的错误。我对这个例子所做的就是改变有问题的图像文件的路径,我将 bool 类型更改为 int,将 false 更改为 0,将 true 更改为 1。据我所知,不应保留任何特定于 C++ 的内容。
无论我做什么,一切似乎都正常,但是当用 CC/GCC 编译时(我想这真的是一样的)我最后遇到了一个分段错误,我怀疑是在 close() 中,它我一直无法确定。用 G++ 编译以某种方式防止分段错误。
解决办法当然很简单,用G++就可以了,但是我很想知道问题出在哪里。
main.c:
//Using SDL and standard IO
#include <SDL2/SDL.h>
#include <stdio.h>
//Screen dimension constants
const int SCREEN_WIDTH = 640;
const int SCREEN_HEIGHT = 480;
//Starts up SDL and creates window
int init();
//Loads media
int loadMedia();
//Frees media and shuts down SDL
void close();
//The window we'll be rendering to
SDL_Window* gWindow = NULL;
//The surface contained by the window
SDL_Surface* gScreenSurface = NULL;
//The image we will load and show on the screen
SDL_Surface* gHelloWorld = NULL;
int init()
{
//Initialization flag
int success = 1;
//Initialize SDL
if( SDL_Init( SDL_INIT_VIDEO ) < 0 )
{
printf( "SDL could not initialize! SDL_Error: %s\n", SDL_GetError() );
success = 0;
}
else
{
//Create window
gWindow = SDL_CreateWindow( "SDL Tutorial", SDL_WINDOWPOS_UNDEFINED,
SDL_WINDOWPOS_UNDEFINED, SCREEN_WIDTH,
SCREEN_HEIGHT, SDL_WINDOW_SHOWN );
if( gWindow == NULL )
{
printf( "Window could not be created! SDL_Error: %s\n",
SDL_GetError() );
success = 0;
}
else
{
//Get window surface
gScreenSurface = SDL_GetWindowSurface( gWindow );
}
}
return success;
}
int loadMedia()
{
//Loading success flag
int success = 1;
//Load splash image
gHelloWorld = SDL_LoadBMP( "hello_world.bmp" );
if( gHelloWorld == NULL )
{
printf( "Unable to load image %s! SDL Error: %s\n", "hello_world.bmp",
SDL_GetError() );
success = 0;
}
return success;
}
void close()
{
//Deallocate surface
SDL_FreeSurface( gHelloWorld );
gHelloWorld = NULL;
//Destroy window
SDL_DestroyWindow( gWindow );
gWindow = NULL;
//Quit SDL subsystems
SDL_Quit();
}
int main( int argc, char* args[] )
{
//Start up SDL and create window
if( !init() )
{
printf( "Failed to initialize!\n" );
}
else
{
//Load media
if( !loadMedia() )
{
printf( "Failed to load media!\n" );
}
else
{
//Apply the image
SDL_BlitSurface( gHelloWorld, NULL, gScreenSurface, NULL );
//Update the surface
SDL_UpdateWindowSurface( gWindow );
//Wait two seconds
SDL_Delay( 2000 );
}
}
//Free resources and close SDL
close();
return 0;
}
我看不出它有什么相关性,但为了安全起见,这是我正在使用的标准 Makefile
#OBJS specifies which files to compile as part of the project
OBJS = main.c
#CC specifies which compiler we're using
CC = cc
#COMPILER_FLAGS specifies the additional compilation options we're using
# -w suppresses all warnings
COMPILER_FLAGS = -Wall -Wextra -pedantic
#LINKER_FLAGS specifies the libraries we're linking against
LINKER_FLAGS = -lSDL2
#OBJ_NAME specifies the name of our exectuable
OBJ_NAME = test
#This is the target that compiles our executable
all : $(OBJS)
$(CC) $(OBJS) $(COMPILER_FLAGS) $(LINKER_FLAGS) -o $(OBJ_NAME)
我没有收到任何错误或警告,但未使用 argv 和 argc。
此时我已无路可走,所以我在这里问。
此致。
注意: 我应该提到的是,无论是什么问题,它绝对不是硬件问题,各种搜索结果都表明,因为我在两个完全不同的硬件平台上得到了完全相同的结果和问题。
如果重命名函数 close()
,段错误就会消失,看起来 init()
调用 SDL,调用 X11,调用 X11,调用驱动程序,驱动程序调用 close()
,但不是调用右边的 close()
它会调用你的。在 C++ 中,函数的名称会被改成其他名称,所以这不是问题。