KGL
KGL
KGL
2002
c Paul Boese
1 Overview 5
2 KGL API 6
2.1 KGL DC-specific APIs . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 6
2.1.1 int glKosInit() . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 6
2.1.2 void glKosShutdown() . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 6
2.1.3 void glKosGetScreenSize(GLfloat *x, GLfloat *y) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 6
2.1.4 void glKosBeginFrame(); . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 6
2.1.5 void glKosFinishFrame() . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 6
2.1.6 void glKosFinishList() . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 8
2.1.7 void glKosMatrixIdent() (DR) . . . . . . . . . . . . . . . . . . . . 8
2.1.8 void glKosMatrixDirty() (DR) . . . . . . . . . . . . . . . . . . . . 8
2.1.9 void glKosPolyHdrDirty() (DR) . . . . . . . . . . . . . . . . . . . 8
2.1.10 void glKosMatrixApply(GLenum mode) (DR) . . . . . . . . . . . . . . . 8
2.1.11 void glKosSendPolyHdr() (DR) . . . . . . . . . . . . . . . . . . . . 8
2.2 Per Fragment Operations . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 8
2.2.1 void glBlendFunc(GLenum sfactor, GLenum dfactor) . . . . . . . . . . . . . . . . . . . . . . . . . . . . 8
2.2.2 void glClearDepth(GLclampd depth) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 9
2.2.3 void glDepthFunc(GLenum func) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 9
2.2.4 void glDepthMask(GLboolean flag) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 9
2.2.5 glScissor(GLint x, GLint y, GLsizei width, GLsizei height) . . . . . . . . . . . . . . . . . . . . . . . . . 9
2.3 Display lists . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 10
2.4 Drawing functions . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 10
2.4.1 void glBegin(GLenum mode) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 10
2.4.2 void glColor3f(GLfloat red, GLfloat green, GLfloat blue) . . . . . . . . . . . . . . . . . . . . . . . . . . 10
2.4.3 void glColor3fv(GLfloat *v) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 10
2.4.4 void glColor4f(GLfloat red, GLfloat green, GLfloat blue, GLfloat alpha) . . . . . . . . . . . . . . . . . 10
2.4.5 void glColor4fv(GLfloat *v) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 10
2.4.6 void glColor4ub(GLubyte red, GLubyte green, GLubyte blue, GLubyte alpha) . . . . . . . . . . . . . 10
2.4.7 void glEnd(void) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 10
2.4.8 void glNormal3f(GLfloat nx, GLfloat ny, GLfloat nz) . . . . . . . . . . . . . . . . . . . . . . . . . . . . 10
2.4.9 void glPointSize(GLfloat size) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 11
2.4.10 void glTexCoord2f(GLfloat s, GLfloat t) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 11
2.4.11 void glTexCoord2fv(GLfloat *v) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 11
1
2.4.12 void glVertex3f(GLfloat x, GLfloat y, GLfloat z) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 11
2.4.13 void glVertex3fv(GLfloat *v) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 11
2.5 Fog . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 11
2.5.1 void glFogi(GLenum pname, GLint param) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 11
2.5.2 void glFogiv(GLenum pname, const GLint *params) . . . . . . . . . . . . . . . . . . . . . . . . . . . . 11
2.5.3 void glFogf(GLenum pname, GLfloat param) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 11
2.5.4 void glFogfv(GLenum pname, const GLfloat *params) . . . . . . . . . . . . . . . . . . . . . . . . . . . 11
2.6 Lighting . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 12
2.6.1 void glShadeModel(GLenum mode) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 12
2.7 Miscellaneous . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 12
2.7.1 void glClear(GLbitfield mask) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 12
2.7.2 void glClearColor(GLclampf red, GLclampf green, GLclampf blue, GLclampf alpha) . . . . . . . . . . 12
2.7.3 void glCullFace(GLenum mode) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 12
2.7.4 void glDisable(GLenum cap) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 12
2.7.5 void glEnable(GLenum cap) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 12
2.7.6 void glFrontFace(GLenum mode) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 13
2.7.7 void glFlush() . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 13
2.7.8 void glHint(GLenum target, GLenum mode) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 13
2.8 Query functions . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 13
2.8.1 const GLubyte *glGetString(GLenum name) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 13
2.9 Texture functions . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 13
2.9.1 void glBindTexture(GLenum target, GLuint texture) . . . . . . . . . . . . . . . . . . . . . . . . . . . . 13
2.9.2 void glDeleteTextures(GLsizei n, const GLuint *textures) . . . . . . . . . . . . . . . . . . . . . . . . . 14
2.9.3 void glGenTextures(GLsizei n, GLuint *textures) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 14
2.9.4 void glKosTex2D(GLint internal fmt, GLsizei width, GLsizei height, pvr ptr t txr address, ) . . . . . . 14
2.9.5 void glTexEnvi(GLenum target, GLenum pname, GLint param) . . . . . . . . . . . . . . . . . . . . . . 14
2.9.6 void glTexImage2D(GLenum target, GLint level, GLint internalFormat, GLsizei width, GLsizei height,
GLint border, GLenum format, GLenum type, const GLvoid *pixels) . . . . . . . . . . . . . . . . . . . 14
2.9.7 void glTexParameteri(GLenum target, GLenum pname, GLint param) . . . . . . . . . . . . . . . . . . 14
2.10 Transformation functions . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 15
2.10.1 void glDepthRange(GLclampf n, GLclampf f) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 15
2.10.2 void glFrustum(GLfloat left, GLfloat right, GLfloat bottom, GLfloat top, GLfloat znear, GLfloat zfar) 15
2.10.3 void glLoadIdentity(void) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 15
2.10.4 void glLoadMatrixf(const GLfloat *m) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 15
2.10.5 void glLoadTransposeMatrixf(const GLfloat *m) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 15
2.10.6 void glMatrixMode(GLenum mode) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 15
2.10.7 void glMultMatrixf(const GLfloat *m) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 15
2.10.8 void glMultTransposeMatrixf(const GLfloat *m) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 15
2.10.9 void glOrtho(GLfloat left, GLfloat right, GLfloat bottom, GLfloat top, GLfloat znear, GLfloat zfar) . 15
2.10.10 void glPopMatrix(void) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 15
2.10.11 void glPushMatrix(void) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 15
2.10.12 void glRotatef(GLfloat angle, GLfloat x, GLfloat y, GLfloat z) . . . . . . . . . . . . . . . . . . . . . . 15
2.10.13 void glScalef(GLfloat x, GLfloat y, GLfloat z) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 16
2.10.14 void glTranslatef(GLfloat x, GLfloat y, GLfloat z) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 16
2
2.10.15 void glViewport(GLint x, GLint y, GLsizei width, GLsizei height) . . . . . . . . . . . . . . . . . . . . 16
2.10.16 void gluLookAt(GLfloat eyex, GLfloat eyey, GLfloat eyez, GLfloat centerx, GLfloat centery, GLfloat
centerz, GLfloat upx, GLfloat upy, GLfloat upz) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 16
2.10.17 void gluPerspective(GLfloat fovy, GLfloat aspect, GLfloat zNear, GLfloat zFar) . . . . . . . . . . . . . 16
3 Direct Render 17
3.1 Direct Render sample code . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 19
3.2 Direct Render API . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 19
3
Legal
“Sega” and “Dreamcast” are registered trademarks of Sega Enterprises, Ltd. This package has no association with Sega or
any of its affiliates, besides working on their hardware.
“Gameboy Advance” is a registered trademark of Nintendo, Inc. This package has no association with Nintendo or any
of its affiliates, besides working on some of their hardware.
“OpenGL” is a registered trademark owned by Silicon Graphics, Inc. This package has no association with Silicon
Graphics or any of its affiliates. This product is based on the published OpenGL
r API, but is not an implementation which
is certified or licensed by Silicon Graphics, Inc. under the OpenGL
r API.
All other trademarks are owned by their respective trademark holders.
KallistiOS
2000-2002
c Dan Potter. Other portions
c their individual authors; please contact the authors for information
on using and/or distriburing their code.
KallistiGL
2000-2002
c Dan Potter, Benoit Miller, and others. Other portions
c their individual authors; please contact
the authors for information on using and/or distriburing their code.
Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following
conditions are met:
1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following
disclaimer in the documentation and/or other materials provided with the distribution.
3. Neither the name of Cryptic Allusion nor the names of its contributors may be used to endorse or promote products
derived from this software without specific prior written permission.
THIS SOFTWARE IS PROVIDED BY THE AUTHORS AND CONTRIBUTORS “AS IS” AND ANY EXPRESS OR IM-
PLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABIL-
ITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHORS OR
CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CON-
SEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLI-
GENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
OF THE POSSIBILITY OF SUCH DAMAGE.
4
Chapter 1
Overview
The KallistiGL (KGL) name is not intended to infringe on the OpenGL
r trademark owned by Silicon Graphics Inc. It
stands for KallistiOS Graphics Library. Please note the following quote (in case you didn’t read the legal stuff) from the
trademark.pdf published at the SGI website
http://www.sgi.com/software/opengl/license.html
“This product is based on the published OpenGL
r API, but is not an implementation which is certified or
licensed by Silicon Graphics, Inc. under the OpenGL
r API.”
We are not making this a library that is going to be 100% OpenGL
r complient, nor are we planning get it conformance
tested and licensed. It is our goal to make a library that is fast and provides enough features that it will be possible to port
some OpenGL
r games, and also make it easier for homebrew developers to create new and original works for the Dreamcast.
We are striving for a certain level of OpenGL
r conformance but do not intend to add features that we believe are either
incompatible with an embedded system or don’t map easily to the hardware. At the time of this writing KGL is a Dreamcast
only API.
5
Chapter 2
KGL API
This is not an OpenGL
r tutorial or complete reference. Those are available online or at a bookstore. This section describes
the KGL API currently implemented, the GL XXX enums supported, and any known non-conformance issues, or DC features
for each API function. The API includes non-working stubs for some functions to ease porting. These stubs will be clearly
noted. If you think KGL is doing something weird or wrong please refer to this document to determine if the functions in
question are supported.
Some of the glKos functions were created to support the Direct Render method of sending polygons to the hardware.
Those functions are marker with “(DR).”
6
Algorithm 1 DC specific API exmaple
pvr init params t params = {
7
2.1.6 void glKosFinishList()
Finish with the current list. See code example 1, page 7. XXX - describe lists.
mode
GL KOS SCREENVIEW
GL MODELVIEW
GL PROJECTION
8
Constant sfactor dfactor Computed Blend Factor
GL ZERO X X (0, 0, 0, 0)
GL ONE X X (1, 1, 1, 1)
GL DST COLOR X (Rd , Gd , Bd , Ad )
GL SRC COLOR X (Rs , Gs , Bs , As )
GL ONE MINUS DST COLOR X (1, 1, 1, 1) - (Rd , Gd , Bd , Ad )
GL ONE MINUS SRC COLOR X (1, 1, 1, 1) - (Rs , Gs , Bs , As )
GL SRC ALPHA X X (As , As , As , As )
GL ONE MINUS SRC ALPHA X X (1, 1, 1, 1) - (As , As , As , As )
GL DST ALPHA X X (Ad , Ad , Ad , Ad )
GL ONE MINUS DST ALPHA X X (1, 1, 1, 1) - (Ad , Ad , Ad , Ad )
GL SRC ALPHA SATURATE not available
GL CONSTANT COLOR not available
GL ONE MINUS CONSTANT COLOR not available
GL CONSTANT ALPHA not available
GL ONE MINUS CONSTANT ALPHA not available
9
The specified scissor box will always have a size that is some multiple of 32. glScissor(0, 0, 32, 32) allows only the ’tile’
in the lower left hand corner of the screen to be modified and glScissor(0, 0, 0, 0) disallows modification to all ’tiles’ on the
screen. Enabling GL SCISSOR TEST or GL KOS USERCLIP OUTSIDE will cause rendered objects to be clipped to the
inside or outside of the scissor box respectively.
Mode Implemented
GL POINTS X
GL LINES
GL LINE LOOP
GL LINE STRIP
GL TRIANGLES X
GL TRIANGLE STRIP X
GL TRIANGLE FAN
GL QUADS X
GL QUAD STRIP
GL POLYGON
10
2.4.9 void glPointSize(GLfloat size)
GL POINTS are emulated using quads. Points are therefore rendered square. Default size is 1.0 which is 2 pixels square on
the DC. Maximum size is 100.0.
2.5 Fog
Uses the PVR2 hardware fog table. Care must be taken when changing fog values because there is only one fog table and
any changes made while the PVR is rendering a scene may result in display artifacts. Most noteably if the changes made are
large. When vertex fog is implemented this will not be an issue, however vertex fog will take more CPU time.
11
2.6 Lighting
This is a decidedly lackluster section. Hopefully lighting will be supported at some point in the future.
2.7 Miscellaneous
2.7.1 void glClear(GLbitfield mask)
NOP: there’s nothing to clear on the PVR2. Provided to ease porting.
2.7.2 void glClearColor(GLclampf red, GLclampf green, GLclampf blue, GLclampf alpha)
Sets the background color. Alpha parameter has no effect.
12
2.7.6 void glFrontFace(GLenum mode)
Set polygon front face. Valid modes are GL CW and GL CCW. Default mode is GL CCW. Note: In KGL prior to KOS-1.1.7
the default was GL CW and could not be changed.
String name
GL VENDOR
GL RENDERER
GL VERSION
GL EXTENSIONS
13
2.9.2 void glDeleteTextures(GLsizei n, const GLuint *textures)
2.9.3 void glGenTextures(GLsizei n, GLuint *textures)
2.9.4 void glKosTex2D(GLint internal fmt, GLsizei width, GLsizei height, pvr ptr t txr address,
)
If you have already allocated and loaded a texture you can use this shortcut function to make it available to KGL. The
internal fmt varies from from the OpenGL spec. The table shows the internal formats supported by the PVR2.
internal fmt
GL ARGB1555
GL RGB565
GL ARGB4444
GL YUV422
GL BUMP
GL ARGB1555 TWID
GL RGB565 TWID
GL ARGB4444 TWID
GL YUV422 TWID
GL BUMP TWID
2.9.6 void glTexImage2D(GLenum target, GLint level, GLint internalFormat, GLsizei width,
GLsizei height, GLint border, GLenum format, GLenum type, const GLvoid *pixels)
Defines a two-dimensional texture. The only supported target is GL TEXTURE 2D. Border, level, and pixel conversion
(type) are not supported. The internalFormat and format arguments must be the same. See the table in section 2.9.4, on
page 14.
14
target pname param
GL FILTER NONE (default)
GL TEXTURE 2D GL TEXTURE FILTER
GL FILTER BILINEAR
GL REPEAT (default)
GL TEXTURE WRAP S
GL CLAMP
GL REPEAT (default)
GL TEXTURE WRAP T
GL CLAMP
2.10.2 void glFrustum(GLfloat left, GLfloat right, GLfloat bottom, GLfloat top, GLfloat
znear, GLfloat zfar)
2.10.3 void glLoadIdentity(void)
2.10.4 void glLoadMatrixf(const GLfloat *m)
Load an arbitrary matrix into the current matrix.
15
2.10.13 void glScalef(GLfloat x, GLfloat y, GLfloat z)
2.10.14 void glTranslatef(GLfloat x, GLfloat y, GLfloat z)
2.10.15 void glViewport(GLint x, GLint y, GLsizei width, GLsizei height)
The default values are automatically set during glKosInit( ) and are determined by the current video mode. You can of
course choose your own values and KGL will render to the defined rectangle. The defined area will not be clipped. To clip
the defined rectangle you will need to use glScissors( ).
2.10.16 void gluLookAt(GLfloat eyex, GLfloat eyey, GLfloat eyez, GLfloat centerx, GLfloat
centery, GLfloat centerz, GLfloat upx, GLfloat upy, GLfloat upz)
2.10.17 void gluPerspective(GLfloat fovy, GLfloat aspect, GLfloat zNear, GLfloat zFar)
16
Chapter 3
Direct Render
When standard KGL just isn’t fast enough for your needs it’s time to summon the forces of Direct Render. Direct Render,
from this point forward referred to as DR, lets you mix the convenience features of KGL with a number of macros and glKOS
functions. When used properly you can expect get 3 to 4 times more polys per frame than KGL alone.
The peformace impact of DR methods are so great that the techniques have been applied to the core KGL drawing functions
for a nice performance boost. Some benchmarks taken before and after the conversion are posted below. GL TRIANGLES
were used since they are easy to draw and are the least efficient way draw objects, so any improvements in terms of polygons
per second (pps) are pretty impressive. The raw PVR benckmarks are even more impressive.
The raw PVR benchmarks for triangles are even more impressive. These benchmarks are from the pvrmark and pvrmark strip
test programs.
17
Average Frame Rate: ˜59.890300 fps (387251 pps)
Entering PHASE FINAL
Average Frame Rate: ˜61.187100 fps (395635 pps)
Average Frame Rate: ˜59.875900 fps (387158 pps)
Average Frame Rate: ˜59.873900 fps (387145 pps)
Average Frame Rate: ˜60.927500 fps (393957 pps)
Average Frame Rate: ˜59.414400 fps (384173 pps)
The DCPVR2 can draw long triangles strips even more efficiently.
You’re probably sick of benchmarks but I have to show you the results of the pure PVR API in action. This is the output
of the pvrmark strips direct program:
These PVR benchmark test programs were used in the development of the Direct Render methods, macros, and glKos
functions described in this chapter. We owe this all the Dan Potters’ desire to put 6 transparent spheres on the screen at
once. DR can do far more then that. In the words of the wizard himself in response to a question about the GL TRIANGLE
benchmarks:
1) Every extra amount of time that you spend transmitting polygons to the TA for rendering is less time you
have available for doing things like playing background music, updating AI, etc. So the faster we can make this,
the better. That’s why I started the whole DR thing anyway – I could have made my graphics stuff fast enough
without it, but wouldn’t have had any time left over for anything else. Plus if you have extra polys left over then
you can go about improving the visual quality of the effects.
2) It’s a lot easier to rack up tons of polygons than you might think. For example, if you have a spaceship
with 100 quads in it (not very many, let me tell you! =) and you have 10 of them on screen, that’s about a fifth
of the available time there if you did nothing else.
18
3.1 Direct Render sample code
Rather than enumerate all the DR macros and functions right away we’ll toss out a couple pieces of sample code so you can
see the differences between plain old KGL and KGL using DR. Note these are not complete functional programs, for those I
strongly urge you to review the bubbles demo under the Dreamcast KGL examples.
XXX KGL sphere quads
XXX KGL/DR sphere DR
19
Chapter 4
This document was written in a stock LYX 1.1.6fix4 distribution, using no fancy add-ons (so it ought to load ok for you if
you want to try it). The PostScript, PDF, HTML, and text versions were generated using the export options in LYX.
20