/*
 * Mesa 3-D graphics library
 * Version:  6.5.2
 * 
 * Copyright (C) 1999-2006  Brian Paul   All Rights Reserved.
 * 
 * Permission is hereby granted, free of charge, to any person obtaining a
 * copy of this software and associated documentation files (the "Software"),
 * to deal in the Software without restriction, including without limitation
 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
 * and/or sell copies of the Software, and to permit persons to whom the
 * Software is furnished to do so, subject to the following conditions:
 * 
 * The above copyright notice and this permission notice shall be included
 * in all copies or substantial portions of the Software.
 * 
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
 * BRIAN PAUL BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
 * AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 */


/*
 * This program opens two GLX windows, renders into one and uses
 * glCopyPixels to copy the image from the first window into the
 * second by means of the GLX 1.3 function glxMakeContextCurrent().
 * This function works just like the glXMakeCurrentReadSGI() function
 * in the GLX_SGI_make_current_read extension.
 */


#define GL_GLEXT_PROTOTYPES
#define GLX_GLXEXT_PROTOTYPES
#include <GL/gl.h>
#include <GL/glx.h>
#include <X11/keysym.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>


#ifdef GLX_VERSION_1_3


static Display *Dpy;
static int ScrNum;
static GLXContext Context;
static Window Win[2];  /* Win[0] = source,  Win[1] = dest */
static GLint Width[2], Height[2];
static GLboolean TestClipping = GL_FALSE;
static GLfloat Angle = 0.0;

static GLboolean DrawFront = GL_FALSE;

PFNGLXMAKECURRENTREADSGIPROC make_context_current = NULL;

static Window
CreateWindow(Display *dpy, int scrnum, XVisualInfo *visinfo,
             int xpos, int ypos, int width, int height,
             const char *name)
{
   Window win;
   XSetWindowAttributes attr;
   unsigned long mask;
   Window root;

   root = RootWindow(dpy, scrnum);

   /* window attributes */
   attr.background_pixel = 0;
   attr.border_pixel = 0;
   attr.colormap = XCreateColormap(dpy, root, visinfo->visual, AllocNone);
   attr.event_mask = StructureNotifyMask | ExposureMask | KeyPressMask;
   mask = CWBackPixel | CWBorderPixel | CWColormap | CWEventMask;

   win = XCreateWindow(dpy, root, xpos, ypos, width, height,
		        0, visinfo->depth, InputOutput,
		        visinfo->visual, mask, &attr);
   if (win) {
      XSizeHints sizehints;
      sizehints.x = xpos;
      sizehints.y = ypos;
      sizehints.width  = width;
      sizehints.height = height;
      sizehints.flags = USSize | USPosition;
      XSetNormalHints(dpy, win, &sizehints);
      XSetStandardProperties(dpy, win, name, name,
                              None, (char **)NULL, 0, &sizehints);

      XMapWindow(dpy, win);
   }
   return win;
}


static void
Redraw(void)
{
   /* make the first window the current one */
   if (! (*make_context_current)(Dpy, Win[0], Win[0], Context)) {
      printf("glXMakeContextCurrent failed in Redraw()\n");
      return;
   }

   Angle += 1.0;

   if (DrawFront) {
      glDrawBuffer(GL_FRONT);
      glReadBuffer(GL_FRONT);
   }
   else {
      glDrawBuffer(GL_BACK);
      glReadBuffer(GL_BACK);
   }

   glViewport(0, 0, Width[0], Height[0]);
   glMatrixMode(GL_PROJECTION);
   glLoadIdentity();
   glOrtho(-1.0, 1.0, -1.0, 1.0, -1.0, 1.0);
   glMatrixMode(GL_MODELVIEW);

   glShadeModel(GL_FLAT);
   glClearColor(0.5, 0.5, 0.5, 0.0);
   glClear(GL_COLOR_BUFFER_BIT);

   /* draw blue quad */
   glColor3f(0.3, 0.3, 1.0);
   glPushMatrix();
   glRotatef(Angle, 0, 0, 1);
   glBegin(GL_POLYGON);
   glVertex2f(-0.5, -0.25);
   glVertex2f( 0.5, -0.25);
   glVertex2f( 0.5, 0.25);
   glVertex2f(-0.5, 0.25);
   glEnd();
   glPopMatrix();

   if (DrawFront)
      glFinish();
   else
      glXSwapBuffers(Dpy, Win[0]);


   /* copy image from window 0 to window 1 */
   if (!(*make_context_current)(Dpy, Win[1], Win[0], Context)) {
      printf("glXMakeContextCurrent failed in Redraw()\n");
      return;
   }

   /* copy the image between windows */
   glClearColor(0.0, 0.0, 0.0, 0.0);
   glClear(GL_COLOR_BUFFER_BIT);

   if (TestClipping) {
      glWindowPos2iARB(-2, -2);
      glCopyPixels(-2, -2, Width[0] + 4, Height[0] + 4, GL_COLOR);
   }
   else {
      glWindowPos2iARB(0, 0);
      glCopyPixels(0, 0, Width[0], Height[0], GL_COLOR);
   }

   if (DrawFront)
      glFinish();
   else
      glXSwapBuffers(Dpy, Win[1]);
}



static void
Resize(Window win, unsigned int width, unsigned int height)
{
   int i;
   if (win == Win[0]) {
      i = 0;
   }
   else {
      i = 1;
   }
   Width[i] = width;
   Height[i] = height;
   if (!glXMakeCurrent(Dpy, Win[i], Context)) {
      printf("glXMakeCurrent failed in Resize()\n");
      return;
   }
}



static void
EventLoop(void)
{
   XEvent event;
   while (1) {
      if (XPending(Dpy) > 0) {
         XNextEvent( Dpy, &event );
         switch (event.type) {
            case Expose:
               Redraw();
               break;
            case ConfigureNotify:
               Resize(event.xany.window, event.xconfigure.width, event.xconfigure.height);
               break;
            case KeyPress:
               {
                  char buf[100];
                  KeySym keySym;
                  XComposeStatus stat;
                  XLookupString(&event.xkey, buf, sizeof(buf), &keySym, &stat);
                  if (keySym == XK_Escape) {
                        /* exit */
                        return;
                  }
                  else if (buf[0] == 'f') {
                     DrawFront = !DrawFront;
                     printf("Drawing to %s buffer\n",
                            DrawFront ? "GL_FRONT" : "GL_BACK");
                  }
               }
               break;
            default:
               /*no-op*/ ;
         }
      }
      else {
         /* animate */
         Redraw();
      }
   }
}


static void
Init(void)
{
   XVisualInfo *visinfo;
   int attrib[] = { GLX_RGBA,
		    GLX_RED_SIZE, 1,
		    GLX_GREEN_SIZE, 1,
		    GLX_BLUE_SIZE, 1,
		    GLX_DOUBLEBUFFER,
		    None };
   int major, minor;

   Dpy = XOpenDisplay(NULL);
   if (!Dpy) {
      printf("Couldn't open default display!\n");
      exit(1);
   }

   ScrNum = DefaultScreen(Dpy);

   glXQueryVersion(Dpy, &major, &minor);

   if (major * 100 + minor >= 103) {
      make_context_current = (PFNGLXMAKECURRENTREADSGIPROC)
	  glXGetProcAddressARB( (GLubyte *) "glXMakeContextCurrent" );
   }
   else {
      const char * const glxExtensions = glXQueryExtensionsString(Dpy, ScrNum);
      const char * ext = strstr( glxExtensions, "GLX_SGI_make_current_read" );
      const size_t len = strlen( "GLX_SGI_make_current_read" );
      
      if ( (ext != NULL) 
	   && ((ext[len] == ' ') || (ext[len] == '\0')) ) {
	 make_context_current = (PFNGLXMAKECURRENTREADSGIPROC) 
	     glXGetProcAddressARB( (GLubyte *) "glXMakeCurrentReadSGI" );
      }
   }

   if (make_context_current == NULL) {
      fprintf(stderr, "Sorry, this program requires either GLX 1.3 "
	      "or GLX_SGI_make_current_read.\n");
      exit(1);
   }

   visinfo = glXChooseVisual(Dpy, ScrNum, attrib);
   if (!visinfo) {
      printf("Unable to find RGB, double-buffered visual\n");
      exit(1);
   }

   Context = glXCreateContext(Dpy, visinfo, NULL, True);
   if (!Context) {
      printf("Couldn't create GLX context\n");
      exit(1);
   }


   Win[0] = CreateWindow(Dpy, ScrNum, visinfo,
                         0, 0, 300, 300, "source window");

   Win[1] = CreateWindow(Dpy, ScrNum, visinfo,
                         350, 0, 300, 300, "dest window");

   printf("Press Esc to exit\n");
   printf("Press 'f' to toggle front/back buffer drawing\n");
}


int
main(int argc, char *argv[])
{
   if (argc > 1 && strcmp(argv[1], "-clip") == 0)
      TestClipping = GL_TRUE;
   Init();
   EventLoop();
   return 0;
}


#else


int
main(int argc, char *argv[])
{
   printf("This program requires GLX 1.3!\n");
   return 0;
}


#endif /* GLX_VERSION_1_3 */