blob: b07754966c39460406f85ad6cc7b2aedd029a8c6 [file] [log] [blame]
/*******************************************************************************************************************************************
@File rad_LinuxX11.cpp
@Title Radiance HelloAPI Tutorial
@Version
@Copyright Copyright (c) Imagination Technologies Limited.
@Platform
@Description Basic Tutorial that shows step-by-step how to initialize Radiance, use it for drawing a triangle and terminate it.
Entry Point: main
*******************************************************************************************************************************************/
/*******************************************************************************************************************************************
Include Files
*******************************************************************************************************************************************/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "X11/Xlib.h"
#include "X11/Xutil.h"
#include <EGL/egl.h>
#include "RAD/rad.h"
/*******************************************************************************************************************************************
Defines
*******************************************************************************************************************************************/
// Name of the application
#define APPLICATION_NAME "HelloAPI"
// Width and height of the window
#define WINDOW_WIDTH 500
#define WINDOW_HEIGHT 500
// Index to bind the attributes to vertex shaders
#define VERTEX_ARRAY 0
/*******************************************************************************************************************************************
Helper Functions
*******************************************************************************************************************************************/
/*!*****************************************************************************************************************************************
@Function TestEGLError
@Input functionLastCalled Function which triggered the error
@Return True if no EGL error was detected
@Description Tests for an EGL error and prints it.
*******************************************************************************************************************************************/
bool TestEGLError(const char* functionLastCalled)
{
/* eglGetError returns the last error that occurred using EGL, not necessarily the status of the last called function. The user has to
check after every single EGL call or at least once every frame. Usually this would be for debugging only, but for this example
it is enabled always.
*/
EGLint lastError = eglGetError();
if (lastError != EGL_SUCCESS)
{
printf("%s failed (%x).\n", functionLastCalled, lastError);
return false;
}
return true;
}
/*!*****************************************************************************************************************************************
@Function HandleX11Errors
@Input nativeDisplay Handle to the display
@Input error The error event to handle
@Return Result code to send to the X window system
@Description Processes event messages for the main window
*******************************************************************************************************************************************/
int HandleX11Errors(Display *nativeDisplay, XErrorEvent *error)
{
// Get the X Error
char errorStringBuffer[256];
XGetErrorText(nativeDisplay, error->error_code, errorStringBuffer, 256);
// Print the error
printf("%s", errorStringBuffer);
// Exit the application
exit(-1);
return 0;
}
/*******************************************************************************************************************************************
Application Functions
*******************************************************************************************************************************************/
/*!*****************************************************************************************************************************************
@Function CreateNativeDisplay
@Output nativeDisplay Native display to create
@Return Whether the function succeeded or not.
@Description Creates a native isplay for the application to render into.
*******************************************************************************************************************************************/
bool CreateNativeDisplay(Display** nativeDisplay)
{
// Check for a valid display
if (!nativeDisplay)
{
return false;
}
// Open the display
*nativeDisplay = XOpenDisplay( 0 );
if (!*nativeDisplay)
{
printf("Error: Unable to open X display\n");
return false;
}
return true;
}
/*!*****************************************************************************************************************************************
@Function CreateNativeWindow
@Input nativeDisplay Native display used by the application
@Output nativeWindow Native window type to create
@Return Whether the function succeeded or not.
@Description Creates a native window for the application to render into.
*******************************************************************************************************************************************/
bool CreateNativeWindow(Display* nativeDisplay, Window* nativeWindow)
{
// Get the default screen for the display
int defaultScreen = XDefaultScreen(nativeDisplay);
// Get the default depth of the display
int defaultDepth = DefaultDepth(nativeDisplay, defaultScreen);
// Select a visual info
XVisualInfo* visualInfo = new XVisualInfo;
XMatchVisualInfo( nativeDisplay, defaultScreen, defaultDepth, TrueColor, visualInfo);
if (!visualInfo)
{
printf("Error: Unable to acquire visual\n");
return false;
}
// Get the root window for the display and default screen
Window rootWindow = RootWindow(nativeDisplay, defaultScreen);
// Create a colour map from the display, root window and visual info
Colormap colourMap = XCreateColormap(nativeDisplay, rootWindow, visualInfo->visual, AllocNone);
// Now setup the final window by specifying some attributes
XSetWindowAttributes windowAttributes;
// Set the colour map that was just created
windowAttributes.colormap = colourMap;
// Set events that will be handled by the app, add to these for other events.
windowAttributes.event_mask = StructureNotifyMask | ExposureMask | ButtonPressMask;
// Create the window
*nativeWindow =XCreateWindow(nativeDisplay, // The display used to create the window
rootWindow, // The parent (root) window - the desktop
0, // The horizontal (x) origin of the window
0, // The vertical (y) origin of the window
WINDOW_WIDTH, // The width of the window
WINDOW_HEIGHT, // The height of the window
0, // Border size - set it to zero
visualInfo->depth, // Depth from the visual info
InputOutput, // Window type - this specifies InputOutput.
visualInfo->visual, // Visual to use
CWEventMask | CWColormap, // Mask specifying these have been defined in the window attributes
&windowAttributes); // Pointer to the window attribute structure
// Make the window viewable by mapping it to the display
XMapWindow(nativeDisplay, *nativeWindow);
// Set the window title
XStoreName(nativeDisplay, *nativeWindow, APPLICATION_NAME);
// Setup the window manager protocols to handle window deletion events
Atom windowManagerDelete = XInternAtom(nativeDisplay, "WM_DELETE_WINDOW", True);
XSetWMProtocols(nativeDisplay, *nativeWindow, &windowManagerDelete , 1);
// Delete the visual info
delete visualInfo;
return true;
}
/*!*****************************************************************************************************************************************
@Function CreateEGLDisplay
@Input nativeDisplay The native display used by the application
@Output eglDisplay EGLDisplay created from nativeDisplay
@Return Whether the function succeeded or not.
@Description Creates an EGLDisplay from a native native display, and initialises it.
*******************************************************************************************************************************************/
bool CreateEGLDisplay( Display* nativeDisplay, EGLDisplay &eglDisplay )
{
/* Get an EGL display.
EGL uses the concept of a "display" which in most environments corresponds to a single physical screen. After creating a native
display for a given windowing system, EGL can use this handle to get a corresponding EGLDisplay handle to it for use in rendering.
Should this fail, EGL is usually able to provide access to a default display.
*/
eglDisplay = eglGetDisplay((EGLNativeDisplayType)nativeDisplay);
// If a display couldn't be obtained, return an error.
if (eglDisplay == EGL_NO_DISPLAY)
{
printf("Failed to get an EGLDisplay");
return false;
}
/* Initialize EGL.
EGL has to be initialized with the display obtained in the previous step. All EGL functions other than eglGetDisplay
and eglGetError need an initialised EGLDisplay.
If an application is not interested in the EGL version number it can just pass NULL for the second and third parameters, but they
are queried here for illustration purposes.
*/
EGLint eglMajorVersion, eglMinorVersion;
if (!eglInitialize(eglDisplay, &eglMajorVersion, &eglMinorVersion))
{
printf("Failed to initialise the EGLDisplay");
return false;
}
return true;
}
/*!*****************************************************************************************************************************************
@Function ChooseEGLConfig
@Input eglDisplay The EGLDisplay used by the application
@Output eglConfig The EGLConfig chosen by the function
@Return Whether the function succeeded or not.
@Description Chooses an appropriate EGLConfig and return it.
*******************************************************************************************************************************************/
bool ChooseEGLConfig( EGLDisplay eglDisplay, EGLConfig& eglConfig )
{
/* Specify the required configuration attributes.
An EGL "configuration" describes the capabilities an application requires and the type of surfaces that can be used for drawing.
Each implementation exposes a number of different configurations, and an application needs to describe to EGL what capabilities it
requires so that an appropriate one can be chosen. The first step in doing this is to create an attribute list, which is an array
of key/value pairs which describe particular capabilities requested. In this application nothing special is required so we can query
the minimum of needing it to render to a window, and being OpenGL ES 2.0 capable.
*/
const EGLint configurationAttributes[] =
{
EGL_SURFACE_TYPE, EGL_WINDOW_BIT,
EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT,
EGL_NONE
};
/* Find a suitable EGLConfig
eglChooseConfig is provided by EGL to provide an easy way to select an appropriate configuration. It takes in the capabilities
specified in the attribute list, and returns a list of available configurations that match or exceed the capabilities requested.
Details of all the possible attributes and how they are selected for by this function are available in the EGL reference pages here:
http://www.khronos.org/registry/egl/sdk/docs/man/xhtml/eglChooseConfig.html
It is also possible to simply get the entire list of configurations and use a custom algorithm to choose a suitable one, as many
advanced applications choose to do. For this application however, taking the first EGLConfig that the function returns suits
its needs perfectly, so we limit it to returning a single EGLConfig.
*/
EGLint configsReturned;
if (!eglChooseConfig(eglDisplay, configurationAttributes, &eglConfig, 1, &configsReturned) || (configsReturned != 1))
{
printf("Failed to choose a suitable config.");
return false;
}
return true;
}
/*!*****************************************************************************************************************************************
@Function CreateEGLSurface
@Input nativeWindow A native window that's been created
@Input eglDisplay The EGLDisplay used by the application
@Input eglConfig An EGLConfig chosen by the application
@Output eglSurface The EGLSurface created from the native window.
@Return Whether the function succeeds or not.
@Description Creates an EGLSurface from a native window
*******************************************************************************************************************************************/
bool CreateEGLSurface( Window nativeWindow, EGLDisplay eglDisplay, EGLConfig eglConfig, EGLSurface& eglSurface)
{
/* Create an EGLSurface for rendering.
Using a native window created earlier and a suitable eglConfig, a surface is created that can be used to render OpenGL ES calls to.
There are three main surface types in EGL, which can all be used in the same way once created but work slightly differently:
- Window Surfaces - These are created from a native window and are drawn to the screen.
- Pixmap Surfaces - These are created from a native windowing system as well, but are offscreen and are not displayed to the user.
- PBuffer Surfaces - These are created directly within EGL, and like Pixmap Surfaces are offscreen and thus not displayed.
The offscreen surfaces are useful for non-rendering contexts and in certain other scenarios, but for most applications the main
surface used will be a window surface as performed below.
*/
eglSurface = eglCreateWindowSurface(eglDisplay, eglConfig, (EGLNativeWindowType)nativeWindow, NULL);
if (!TestEGLError("eglCreateWindowSurface"))
{
return false;
}
return true;
}
/*!*****************************************************************************************************************************************
@Function SetupEGLContext
@Input eglDisplay The EGLDisplay used by the application
@Input eglConfig An EGLConfig chosen by the application
@Input eglSurface The EGLSurface created from the native window.
@Output eglContext The EGLContext created by this function
@Input nativeWindow A native window, used to display error messages
@Return Whether the function succeeds or not.
@Description Sets up the EGLContext, creating it and then installing it to the current thread.
*******************************************************************************************************************************************/
bool SetupEGLContext( EGLDisplay eglDisplay, EGLConfig eglConfig, EGLSurface eglSurface, EGLContext& eglContext )
{
/* Create a context.
EGL has to create what is known as a context for OpenGL ES. The concept of a context is OpenGL ES's way of encapsulating any
resources and state. What appear to be "global" functions in OpenGL actually only operate on the current context. A context
is required for any operations in OpenGL ES.
Similar to an EGLConfig, a context takes in a list of attributes specifying some of its capabilities. However in most cases this
is limited to just requiring the version of the OpenGL ES context required - In this case, OpenGL ES 2.0.
*/
EGLint contextAttributes[] =
{
EGL_CONTEXT_CLIENT_VERSION, 2,
EGL_NONE
};
// Create the context with the context attributes supplied
eglContext = eglCreateContext(eglDisplay, eglConfig, NULL, contextAttributes);
if (!TestEGLError("eglCreateContext"))
{
return false;
}
/* Make OpenGL ES the current API.
After creating the context, EGL needs a way to know that any subsequent EGL calls are going to be affecting OpenGL ES,
rather than any other API (such as OpenVG).
*/
eglBindAPI(EGL_OPENGL_ES_API);
if (!TestEGLError("eglBindAPI"))
{
return false;
}
/* Bind the context to the current thread.
Due to the way OpenGL uses global functions, contexts need to be made current so that any function call can operate on the correct
context. Specifically, make current will bind the context to the thread it's called from, and unbind it from any others. To use
multiple contexts at the same time, users should use multiple threads and synchronise between them.
*/
eglMakeCurrent(eglDisplay, eglSurface, eglSurface, eglContext);
if (!TestEGLError("eglMakeCurrent"))
{
return false;
}
return true;
}
/*!*****************************************************************************************************************************************
@Function RenderScene
@Input nativeDisplay The native display used by the application
@Return Whether the function succeeds or not.
@Description Renders the scene to the framebuffer. Usually called within a loop.
*******************************************************************************************************************************************/
bool RenderScene( Display* nativeDisplay )
{
void TestRAD();
TestRAD();
// Check for messages from the windowing system.
int numberOfMessages = XPending(nativeDisplay);
for( int i = 0; i < numberOfMessages; i++ )
{
XEvent event;
XNextEvent(nativeDisplay, &event);
switch( event.type )
{
// Exit on window close
case ClientMessage:
// Exit on mouse click
case ButtonPress:
case DestroyNotify:
return false;
default:
break;
}
}
return true;
}
/*!*****************************************************************************************************************************************
@Function ReleaseEGLState
@Input eglDisplay The EGLDisplay used by the application
@Description Releases all resources allocated by EGL
*******************************************************************************************************************************************/
void ReleaseEGLState(EGLDisplay eglDisplay)
{
if(eglDisplay != NULL)
{
// To release the resources in the context, first the context has to be released from its binding with the current thread.
eglMakeCurrent(eglDisplay, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT);
// Terminate the display, and any resources associated with it (including the EGLContext)
eglTerminate(eglDisplay);
}
}
/*!*****************************************************************************************************************************************
@Function ReleaseWindowAndDisplay
@Input nativeDisplay The native display to release
@Input nativeWindow The native window to destroy
@Description Releases all resources allocated by the windowing system
*******************************************************************************************************************************************/
void ReleaseNativeResources(Display* nativeDisplay, Window nativeWindow)
{
// Destroy the window
if (nativeWindow)
{
XDestroyWindow(nativeDisplay, nativeWindow);
}
// Release the display.
if (nativeDisplay)
{
XCloseDisplay(nativeDisplay);
}
}
/*!*****************************************************************************************************************************************
@Function main
@Input argc Number of arguments passed to the application, ignored.
@Input argv Command line strings passed to the application, ignored.
@Return Result code to send to the Operating System
@Description Main function of the program, executes other functions.
*******************************************************************************************************************************************/
int main(int /*argc*/, char **/*argv*/)
{
// X11 variables
Display* nativeDisplay = NULL;
Window nativeWindow = 0;
// EGL variables
EGLDisplay eglDisplay = NULL;
EGLConfig eglConfig = NULL;
EGLSurface eglSurface = NULL;
EGLContext eglContext = NULL;
// Get access to a native display
if (!CreateNativeDisplay(&nativeDisplay))
{
goto cleanup;
}
// Setup the windowing system, create a window
if (!CreateNativeWindow(nativeDisplay, &nativeWindow))
{
goto cleanup;
}
// Create and Initialise an EGLDisplay from the native display
if (!CreateEGLDisplay(nativeDisplay, eglDisplay))
{
goto cleanup;
}
// Choose an EGLConfig for the application, used when setting up the rendering surface and EGLContext
if (!ChooseEGLConfig(eglDisplay, eglConfig))
{
goto cleanup;
}
// Create an EGLSurface for rendering from the native window
if (!CreateEGLSurface(nativeWindow, eglDisplay, eglConfig, eglSurface))
{
goto cleanup;
}
// Setup the EGL Context from the other EGL constructs created so far, so that the application is ready to submit OpenGL ES commands
if (!SetupEGLContext(eglDisplay, eglConfig, eglSurface, eglContext))
{
goto cleanup;
}
// Initialise the fragment and vertex shaders used in the application
void InitRAD();
InitRAD();
// Renders a triangle for 800 frames using the state setup in the previous function
for (int i = 0; i < 800; ++i)
{
if (!RenderScene(nativeDisplay))
{
break;
}
}
cleanup:
void CleanRAD();
CleanRAD();
// Release the EGL State
ReleaseEGLState(eglDisplay);
// Release the windowing system resources
ReleaseNativeResources(nativeDisplay, nativeWindow);
// Destroy the eglWindow
return 0;
}
/*******************************************************************************************************************************************
End of file (rad_LinuxX11.cpp)
*******************************************************************************************************************************************/