2 * engine.c - GL representation of a 4 stroke engine
6 * Copyright (C) 2001 Ben Buxton (bb@cactii.net)
7 * modified by Ed Beroset (beroset@mindspring.com)
8 * new to 2.0 version is:
9 * - command line argument to specify number of cylinders
10 * - command line argument to specify included angle of engine
11 * - removed broken command line argument to specify rotation speed
12 * - included crankshaft shapes and firing orders for real engines
13 * verified using the Bosch _Automotive Handbook_, 5th edition, pp 402,403
15 * Permission to use, copy, modify, distribute, and sell this software and its
16 * documentation for any purpose is hereby granted without fee, provided that
17 * the above copyright notice appear in all copies and that both that
18 * copyright notice and this permission notice appear in supporting
19 * documentation. No representations are made about the suitability of this
20 * software for any purpose. It is provided "as is" without express or
25 #define DEFAULTS "*delay: 30000 \n" \
26 "*showFPS: False \n" \
27 "*titleFont: -*-helvetica-medium-r-normal-*-180-*\n" \
29 # define refresh_engine 0
30 # include "xlockmore.h" /* from the xscreensaver distribution */
31 #else /* !STANDALONE */
32 # include "xlock.h" /* from the xlockmore distribution */
33 #endif /* !STANDALONE */
37 #include "gltrackball.h"
39 /* lifted from lament.c */
40 #define RAND(n) ((long) ((random() & 0x7fffffff) % ((long) (n))))
41 #define RANDSIGN() ((random() & 1) ? 1 : -1)
46 #define DEF_ENGINE "(none)"
47 #define DEF_TITLES "False"
48 #define DEF_SPIN "True"
49 #define DEF_MOVE "True"
52 #define countof(x) (sizeof((x))/sizeof((*x)))
54 static char *which_engine;
57 static Bool do_titles;
59 static XrmOptionDescRec opts[] = {
60 {"-engine", ".engine.engine", XrmoptionSepArg, DEF_ENGINE },
61 {"-move", ".engine.move", XrmoptionNoArg, "True" },
62 {"+move", ".engine.move", XrmoptionNoArg, "False" },
63 {"-spin", ".engine.spin", XrmoptionNoArg, "True" },
64 {"+spin", ".engine.spin", XrmoptionNoArg, "False" },
65 { "-titles", ".engine.titles", XrmoptionNoArg, "True" },
66 { "+titles", ".engine.titles", XrmoptionNoArg, "False" },
69 static argtype vars[] = {
70 {&which_engine, "engine", "Engine", DEF_ENGINE, t_String},
71 {&move, "move", "Move", DEF_MOVE, t_Bool},
72 {&spin, "spin", "Spin", DEF_SPIN, t_Bool},
73 {&do_titles, "titles", "Titles", DEF_TITLES, t_Bool},
76 ENTRYPOINT ModeSpecOpt engine_opts = {countof(opts), opts, countof(vars), vars, NULL};
79 ModStruct engine_description =
80 {"engine", "init_engine", "draw_engine", "release_engine",
81 "draw_engine", "init_engine", NULL, &engine_opts,
82 1000, 1, 2, 1, 4, 1.0, "",
83 "A four stroke engine", 0, NULL};
87 /* these defines are used to provide symbolic means
88 * by which to refer to various portions or multiples
89 * of a cyle in degrees
95 #define MOVE_MULT 0.05
97 #define RAND_RANGE(min, max) ((min) + (max - min) * f_rand())
101 GLXContext *glx_context;
103 GLfloat x, y, z; /* position */
104 GLfloat dx, dy, dz; /* position */
105 GLfloat an1, an2, an3; /* internal angle */
106 GLfloat nx, ny, nz; /* spin vector */
107 GLfloat a; /* spin angle */
108 GLfloat da; /* spin speed */
110 trackball_state *trackball;
123 float sin_table[TWOREV];
124 float cos_table[TWOREV];
125 float tan_table[TWOREV];
128 GLfloat boom_lpos[4];
129 GLfloat boom_d, boom_wd;
132 GLfloat viewer[3], lookat[3];
135 GLfloat ln[730], yp[730], ang[730];
139 GLuint shaft_list, piston_list;
140 int shaft_polys, piston_polys;
144 static Engine *engine = NULL;
146 static const GLfloat lightpos[] = {7.0, 7.0, 12, 1.0};
147 static const GLfloat light_sp[] = {0.8, 0.8, 0.8, 0.5};
148 static const GLfloat red[] = {1.0, 0, 0, 1.0};
149 static const GLfloat green[] = {0.0, 1, 0, 1.0};
150 static const GLfloat blue[] = {0, 0, 1, 1.0};
151 static const GLfloat white[] = {1.0, 1, 1, 1.0};
152 static const GLfloat yellow_t[] = {1, 1, 0, 0.4};
154 static GLvoid normal(GLfloat [], GLfloat [], GLfloat [],
155 GLfloat *, GLfloat *, GLfloat *);
158 * this table represents both the firing order and included angle of engine.
159 * To simplify things, we always number from 0 starting at the flywheel and
160 * moving down the crankshaft toward the back of the engine. This doesn't
161 * always match manufacturer's schemes. For example, the Porsche 911 engine
162 * is a flat six with the following configuration (Porsche's numbering):
165 * |= firing order is 1-6-2-4-3-5 in this diagram
168 * We renumber these using our scheme but preserve the effective firing order:
171 * |= firing order is 4-1-2-5-0-3 in this diagram
174 * To avoid going completely insane, we also reorder these so the newly
175 * renumbered cylinder 0 is always first: 0-3-4-1-2-5
177 * For a flat 6, the included angle is 180 degrees (0 would be a inline
178 * engine). Because these are all four-stroke engines, each piston goes
179 * through 720 degrees of rotation for each time the spark plug sparks,
180 * so in this case, we would use the following angles:
182 * cylinder firing order angle
183 * -------- ------------ -----
197 int pistonAngle[12]; /* twelve cylinders should suffice... */
198 int speed; /* step size in degrees for engine speed */
199 const char *engineName; /* currently unused */
202 static const engine_type engines[] = {
203 { 3, 0, { 0, 240, 480, 0, 0, 0,
204 0, 0, 0, 0, 0, 0 }, 12,
206 { 4, 0, { 0, 180, 540, 360, 0, 0,
207 0, 0, 0, 0, 0, 0 }, 12,
209 { 4, 180, { 0, 360, 180, 540, 0, 0,
210 0, 0, 0, 0, 0, 0 }, 12,
212 { 5, 0, { 0, 576, 144, 432, 288, 0,
213 0, 0, 0, 0, 0, 0 }, 12,
215 { 6, 0, { 0, 240, 480, 120, 600, 360,
216 0, 0, 0, 0, 0, 0 }, 12,
218 { 6, 90, { 0, 360, 480, 120, 240, 600,
219 0, 0, 0, 0, 0, 0 }, 12,
221 { 6, 180, { 0, 360, 240, 600, 480, 120,
222 0, 0, 0, 0, 0, 0 }, 12,
224 { 8, 90, { 0, 450, 90, 180, 270, 360,
225 540, 630, 0, 0, 0, 0 }, 15,
227 {10, 90, { 0, 72, 432, 504, 288, 360,
228 144, 216, 576, 648, 0, 0 }, 12,
230 {12, 60, { 0, 300, 240, 540, 480, 60,
231 120, 420, 600, 180, 360, 660 }, 12,
235 /* this define is just a little shorter way of referring to members of the
238 #define ENG engines[e->engineType]
240 /* given a number of cylinders and an included angle, finds matching engine */
242 find_engine(char *name)
247 if (!name || !*name || !strcasecmp (name, "(none)"))
248 return (random() % countof(engines));
250 for (s = name; *s; s++)
251 if (*s == '-' || *s == '_') *s = ' ';
253 for (i = 0; i < countof(engines); i++) {
254 if (!strcasecmp(name, engines[i].engineName))
258 fprintf (stderr, "%s: unknown engine type \"%s\"\n", progname, name);
259 fprintf (stderr, "%s: available models are:\n", progname);
260 for (i = 0; i < countof(engines); i++) {
261 fprintf (stderr, "\t %-13s (%d cylinders",
262 engines[i].engineName, engines[i].cylinders);
263 if (engines[i].includedAngle == 0)
264 fprintf (stderr, ")\n");
265 else if (engines[i].includedAngle == 180)
266 fprintf (stderr, ", flat)\n");
268 fprintf (stderr, ", V)\n");
273 /* we use trig tables to speed things up - 200 calls to sin()
274 in one frame can be a bit harsh..
277 static void make_tables(Engine *e)
282 f = ONEREV / (M_PI * 2);
283 for (i = 0 ; i <= TWOREV ; i++) {
284 e->sin_table[i] = sin(i/f);
286 for (i = 0 ; i <= TWOREV ; i++) {
287 e->cos_table[i] = cos(i/f);
289 for (i = 0 ; i <= TWOREV ; i++) {
290 e->tan_table[i] = tan(i/f);
294 /* if inner and outer are the same, we draw a cylinder, not a tube */
295 /* for a tube, endcaps is 0 (none), 1 (left), 2 (right) or 3(both) */
296 /* angle is how far around the axis to go (up to 360) */
298 static int cylinder (Engine *e, GLfloat x, GLfloat y, GLfloat z,
299 float length, float outer, float inner, int endcaps, int sang, int eang)
302 int a; /* current angle around cylinder */
303 int b = 0; /* previous */
304 int angle, norm, step, sangle;
305 float z1, y1, z2, y2, ex=0;
307 float Z1, Y1, Z2, Y2, xl, Y3, Z3;
308 GLfloat y2c[TWOREV], z2c[TWOREV];
309 GLfloat ony, onz; /* previous normals */
313 nsegs = outer*(MAX(e->win_w, e->win_h)/200);
314 nsegs = MAX(nsegs, 6);
315 nsegs = MAX(nsegs, 40);
321 z1 = e->cos_table[sangle]*outer+z; y1 = e->sin_table[sangle] * outer+y;
322 Z1 = e->cos_table[sangle] * inner+z; Y1 = e->sin_table[sangle]*inner+y ;
326 if (inner < outer && endcaps < 3) tube = 1;
330 for (a = sangle ; a <= angle || b <= angle ; a+= step) {
331 y2=outer*(float)e->sin_table[a]+y;
332 z2=outer*(float)e->cos_table[a]+z;
333 y3=outer*(float)e->sin_table[a+step]+y;
334 z3=outer*(float)e->cos_table[a+step]+z;
336 y2c[a] = y2; z2c[a] = z2; /* cache for later */
338 Y2=inner*(float)e->sin_table[a]+y;
339 Z2=inner*(float)e->cos_table[a]+z;
340 Y3=inner*(float)e->sin_table[a+step]+y;
341 Z3=inner*(float)e->cos_table[a+step]+z;
343 glNormal3f(0, y1, z1);
345 glVertex3f(xl,y1,z1);
346 glNormal3f(0, y2, z2);
347 glVertex3f(xl,y2,z2);
350 if (a == sangle && angle - sangle < ONEREV) {
352 glVertex3f(x, Y1, Z1);
355 glVertex3f(x, y1, z1);
356 glVertex3f(xl, y1, z1);
358 glVertex3f(xl, Z1, Z1);
360 glVertex3f(xl, y, z);
365 glNormal3f(-1, 0, 0); /* left end */
366 glVertex3f(x, y1, z1);
367 glVertex3f(x, y2, z2);
368 glVertex3f(x, Y2, Z2);
369 glVertex3f(x, Y1, Z1);
373 glNormal3f(0, -Y1, -Z1); /* inner surface */
374 glVertex3f(x, Y1, Z1);
375 glVertex3f(xl, Y1, Z1);
376 glNormal3f(0, -Y2, -Z2);
377 glVertex3f(xl, Y2, Z2);
378 glVertex3f(x, Y2, Z2);
382 glNormal3f(1, 0, 0); /* right end */
383 glVertex3f(xl, y1, z1);
384 glVertex3f(xl, y2, z2);
385 glVertex3f(xl, Y2, Z2);
386 glVertex3f(xl, Y1, Z1);
397 if (angle - sangle < ONEREV) {
399 GLfloat v1[3], v2[3], v3[3];
400 v1[0] = x; v1[1] = y; v1[2] = z;
401 v2[0] = x; v2[1] = y1; v2[2] = z1;
402 v3[0] = xl; v3[1] = y1; v3[2] = z1;
403 normal(&v2[0], &v1[0], &v3[0], &nx, &ny, &nz);
405 glNormal3f(nx, ny, nz);
407 glVertex3f(x, y1, z1);
408 glVertex3f(xl, y1, z1);
409 glVertex3f(xl, y, z);
419 } else if (endcaps == 2) {
420 start = end = length+0.01;
422 end = length+0.02; start = -0.01;
424 norm = (ex == length+0.01) ? -1 : 1;
431 for(ex = start ; ex <= end ; ex += length) {
432 z1 = outer*e->cos_table[sangle]+z;
433 y1 = y+e->sin_table[sangle]*outer;
435 glBegin(GL_TRIANGLES);
437 for (a = sangle ; a <= angle || b <= angle; a+= step) {
438 glNormal3f(norm, 0, 0);
439 glVertex3f(x+ex,y, z);
440 glVertex3f(x+ex,y1,z1);
441 glVertex3f(x+ex,y2c[a],z2c[a]);
443 y1 = y2c[a]; z1 = z2c[a];
454 /* this is just a convenience function to make a solid rod */
455 static int rod (Engine *e, GLfloat x, GLfloat y, GLfloat z, float length, float diameter)
457 return cylinder(e, x, y, z, length, diameter, diameter, 3, 0, ONEREV);
460 static GLvoid normal(GLfloat v1[], GLfloat v2[], GLfloat v3[],
461 GLfloat *nx, GLfloat *ny, GLfloat *nz)
463 GLfloat x, y, z, X, Y, Z;
480 static int Rect(GLfloat x, GLfloat y, GLfloat z, GLfloat w, GLfloat h,
488 yh = y+h; xw = x+w; zt = z - t;
490 glBegin(GL_QUADS); /* front */
493 glVertex3f(x, yh, z);
494 glVertex3f(xw, yh, z);
495 glVertex3f(xw, y, z);
498 glNormal3f(0, 0, -1);
499 glVertex3f(x, y, zt);
500 glVertex3f(x, yh, zt);
501 glVertex3f(xw, yh, zt);
502 glVertex3f(xw, y, zt);
506 glVertex3f(x, yh, z);
507 glVertex3f(x, yh, zt);
508 glVertex3f(xw, yh, zt);
509 glVertex3f(xw, yh, z);
512 glNormal3f(0, -1, 0);
514 glVertex3f(x, y, zt);
515 glVertex3f(xw, y, zt);
516 glVertex3f(xw, y, z);
519 glNormal3f(-1, 0, 0);
521 glVertex3f(x, y, zt);
522 glVertex3f(x, yh, zt);
523 glVertex3f(x, yh, z);
527 glVertex3f(xw, y, z);
528 glVertex3f(xw, y, zt);
529 glVertex3f(xw, yh, zt);
530 glVertex3f(xw, yh, z);
536 static int makepiston(Engine *e)
539 GLfloat colour[] = {0.6, 0.6, 0.6, 1.0};
541 e->piston_list = glGenLists(1);
542 glNewList(e->piston_list, GL_COMPILE);
543 glRotatef(90, 0, 0, 1);
544 glMaterialfv(GL_FRONT, GL_AMBIENT_AND_DIFFUSE, colour);
545 glMaterialfv(GL_FRONT, GL_SPECULAR, colour);
546 glMateriali(GL_FRONT, GL_SHININESS, 20);
547 polys += cylinder(e, 0, 0, 0, 2, 1, 0.7, 2, 0, ONEREV); /* body */
548 colour[0] = colour[1] = colour[2] = 0.2;
549 glMaterialfv(GL_FRONT, GL_AMBIENT_AND_DIFFUSE, colour);
550 polys += cylinder(e, 1.6, 0, 0, 0.1, 1.05, 1.05, 0, 0, ONEREV); /* ring */
551 polys += cylinder(e, 1.8, 0, 0, 0.1, 1.05, 1.05, 0, 0, ONEREV); /* ring */
556 static int CrankBit(Engine *e, GLfloat x)
559 polys += Rect(x, -1.4, 0.5, 0.2, 1.8, 1);
560 polys += cylinder(e, x, -0.5, 0, 0.2, 2, 2, 1, 60, 120);
564 static int boom(Engine *e, GLfloat x, GLfloat y, int s)
567 int flameOut = 720/ENG.speed/ENG.cylinders;
569 if (e->boom_time == 0 && s) {
570 e->boom_red[0] = e->boom_red[1] = 0;
574 } else if (e->boom_time == 0 && !s) {
576 } else if (e->boom_time >= 8 && e->boom_time < flameOut && !s) {
578 e->boom_red[0] -= 0.2; e->boom_red[1] -= 0.1;
580 } else if (e->boom_time >= flameOut) {
582 glDisable(GL_LIGHT1);
585 e->boom_red[0] += 0.2; e->boom_red[1] += 0.1;
589 e->boom_lpos[0] = x-e->boom_d; e->boom_lpos[1] = y;
590 glLightfv(GL_LIGHT1, GL_POSITION, e->boom_lpos);
591 glLightfv(GL_LIGHT1, GL_DIFFUSE, e->boom_red);
592 glLightfv(GL_LIGHT1, GL_SPECULAR, e->boom_red);
593 glLighti(GL_LIGHT1, GL_LINEAR_ATTENUATION, 1.3);
594 glLighti(GL_LIGHT1, GL_CONSTANT_ATTENUATION, 0);
596 glMaterialfv(GL_FRONT, GL_AMBIENT_AND_DIFFUSE, e->boom_red);
597 e->boom_wd = e->boom_d*3;
598 if (e->boom_wd > 0.7) e->boom_wd = 0.7;
600 glDepthMask(GL_FALSE);
601 glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
602 polys += rod(e, x, y, 0, e->boom_d, e->boom_wd);
603 glDepthMask(GL_TRUE);
608 static int display(Engine *e)
617 glEnable(GL_LIGHTING);
618 glClear(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT);
620 gluLookAt(e->viewer[0], e->viewer[1], e->viewer[2],
621 e->lookat[0], e->lookat[1], e->lookat[2],
624 glLightfv(GL_LIGHT0, GL_POSITION, lightpos);
625 glLightfv(GL_LIGHT0, GL_SPECULAR, light_sp);
626 glLightfv(GL_LIGHT0, GL_DIFFUSE, light_sp);
630 get_position (e->rot, &x, &y, &z, !e->button_down_p);
631 glTranslatef(x*16-9, y*14-7, z*16-10);
635 gltrackball_rotate (e->trackball);
636 get_rotation(e->rot, &x, &y, &z, !e->button_down_p);
637 glRotatef(x*ONEREV, 1.0, 0.0, 0.0);
638 glRotatef(y*ONEREV, 0.0, 1.0, 0.0);
639 glRotatef(x*ONEREV, 0.0, 0.0, 1.0);
642 /* So the rotation appears around the centre of the engine */
643 glTranslatef(-5, 0, 0);
647 glRotatef(e->display_a, 1, 0, 0);
648 glCallList(e->shaft_list);
649 polys += e->shaft_polys;
652 /* init the ln[] matrix for speed */
653 if (e->ln_init == 0) {
654 for (e->ln_init = 0 ; e->ln_init < 730 ; e->ln_init++) {
655 zb = e->sin_table[e->ln_init];
656 yb = e->cos_table[e->ln_init];
657 /* y ordinate of piston */
658 e->yp[e->ln_init] = yb + sqrt(25 - (zb*zb));
660 e->ln[e->ln_init] = sqrt(zb*zb + (yb-e->yp[e->ln_init])*(yb-e->yp[e->ln_init]));
661 /* angle of connecting rod */
662 e->ang[e->ln_init] = asin(zb/5)*57;
663 e->ang[e->ln_init] *= -1;
668 sides = (ENG.includedAngle == 0) ? 1 : 2;
669 for (half = 0; half < sides; half++, glRotatef(ENG.includedAngle,1,0,0))
672 /* glMaterialfv(GL_FRONT, GL_AMBIENT_AND_DIFFUSE, white); */
673 for (j = 0; j < ENG.cylinders; j += sides)
675 b = (e->display_a + ENG.pistonAngle[j+half]) % ONEREV;
677 glTranslatef(e->crankWidth/2 + e->crankOffset*(j+half), e->yp[b]-0.3, 0);
678 glCallList(e->piston_list);
679 polys += e->piston_polys;
684 glRotatef(90, 0, 0, 1);
685 glMaterialfv(GL_FRONT, GL_AMBIENT_AND_DIFFUSE, red);
686 for (j = 0; j < ENG.cylinders; j += sides)
688 polys += cylinder(e, 8.5, -e->crankWidth/2-e->crankOffset*(j+half), 0,
689 0.5, 0.4, 0.3, 1, 0, ONEREV);
691 glMaterialfv(GL_FRONT, GL_AMBIENT_AND_DIFFUSE, white);
692 for (j = 0; j < ENG.cylinders; j += sides)
694 polys += rod(e, 8, -e->crankWidth/2-e->crankOffset*(j+half), 0, 0.5, 0.2);
695 polys += rod(e, 9, -e->crankWidth/2-e->crankOffset*(j+half), 0, 1, 0.15);
699 glMaterialfv(GL_FRONT, GL_AMBIENT_AND_DIFFUSE, blue);
700 for (j = 0; j < ENG.cylinders; j += sides)
702 b = (e->display_a+HALFREV+ENG.pistonAngle[j+half]) % TWOREV;
704 glRotatef(e->ang[b], 0, 1, 0);
707 -e->crankWidth/2-e->crankOffset*(j+half),
715 glMaterialfv(GL_FRONT, GL_AMBIENT_AND_DIFFUSE, yellow_t);
717 glDepthMask(GL_FALSE);
718 glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
719 rightSide = (sides > 1) ? 0 : 1.6;
721 polys += Rect(-e->crankWidth/2, -0.5, 1, 0.2, 9, 2);
723 polys += Rect(0.3+e->crankOffset*ENG.cylinders-rightSide, -0.5, 1, 0.2, 9, 2);
725 polys += Rect(-e->crankWidth/2+0.2, 8.3, 1,
726 e->crankWidth/2+0.1+e->crankOffset*ENG.cylinders-rightSide, 0.2, 2);
728 polys += Rect(-e->crankWidth/2+0.2, 3, 1,
729 e->crankWidth/2+0.1+e->crankOffset*ENG.cylinders-rightSide, 0.2, 0.2);
731 polys += Rect(-e->crankWidth/2+0.2, 3, -1+0.2,
732 e->crankWidth/2+0.1+e->crankOffset*ENG.cylinders-rightSide, 0.2, 0.2);
733 /* plates between cylinders */
734 for (j=0; j < ENG.cylinders - (sides == 1); j += sides)
735 polys += Rect(0.4+e->crankWidth+e->crankOffset*(j-half), 3, 1, 1, 5.3, 2);
736 glDepthMask(GL_TRUE);
740 /* see which of our plugs should fire now, if any */
741 for (j = 0; j < ENG.cylinders; j++)
743 if (0 == ((e->display_a + ENG.pistonAngle[j]) % TWOREV))
747 glRotatef(ENG.includedAngle,1,0,0);
748 glRotatef(90, 0, 0, 1);
749 polys += boom(e, 8, -e->crankWidth/2-e->crankOffset*j, 1);
755 if (e->lastPlug != j)
757 /* this code causes the last plug explosion to dim gradually */
759 glRotatef(ENG.includedAngle, 1, 0, 0);
760 glRotatef(90, 0, 0, 1);
761 polys += boom(e, 8, -e->crankWidth/2-e->crankOffset*e->lastPlug, 0);
765 e->display_a += ENG.speed;
766 if (e->display_a >= TWOREV)
773 static int makeshaft (Engine *e)
777 float crankThick = 0.2;
778 float crankDiam = 0.3;
780 e->shaft_list = glGenLists(1);
781 glNewList(e->shaft_list, GL_COMPILE);
783 glMaterialfv(GL_FRONT, GL_AMBIENT_AND_DIFFUSE, blue);
784 /* draw the flywheel */
785 polys += cylinder(e, -2.5, 0, 0, 1, 3, 2.5, 0, 0, ONEREV);
786 polys += Rect(-2, -0.3, 2.8, 0.5, 0.6, 5.6);
787 polys += Rect(-2, -2.8, 0.3, 0.5, 5.6, 0.6);
789 /* now make each of the shaft bits between the cranks,
790 * starting from the flywheel end which is at X-coord 0.
791 * the first cranskhaft bit is always 2 units long
793 polys += rod(e, -2, 0, 0, 2, crankDiam);
795 /* Each crank is crankWidth units wide and the total width of a
796 * cylinder assembly is 3.3 units. For inline engines, there is just
797 * a single crank per cylinder width. For other engine
798 * configurations, there is a crank between each pair of adjacent
799 * cylinders on one side of the engine, so the crankOffset length is
802 e->crankOffset = 3.3;
803 if (ENG.includedAngle != 0)
805 for (j = 0; j < ENG.cylinders - 1; j++)
807 e->crankWidth - crankThick + e->crankOffset*j, 0, 0,
808 e->crankOffset - e->crankWidth + 2 * crankThick, crankDiam);
809 /* the last bit connects to the engine wall on the non-flywheel end */
810 polys += rod(e, e->crankWidth - crankThick + e->crankOffset*j, 0, 0, 0.9, crankDiam);
813 for (j = 0; j < ENG.cylinders; j++)
817 glRotatef(HALFREV+ENG.pistonAngle[j]+ENG.includedAngle,1,0,0);
819 glRotatef(HALFREV+ENG.pistonAngle[j],1,0,0);
821 glMaterialfv(GL_FRONT, GL_AMBIENT_AND_DIFFUSE, blue);
822 polys += rod(e, e->crankOffset*j, -1.0, 0.0, e->crankWidth, crankDiam);
823 glMaterialfv(GL_FRONT, GL_AMBIENT_AND_DIFFUSE, green);
824 /* draw right part of crank */
825 polys += CrankBit(e, e->crankOffset*j);
826 /* draw left part of crank */
827 polys += CrankBit(e, e->crankWidth-crankThick+e->crankOffset*j);
835 ENTRYPOINT void reshape_engine(ModeInfo *mi, int width, int height)
837 Engine *e = &engine[MI_SCREEN(mi)];
838 glViewport(0,0,(GLint)width, (GLint) height);
839 glMatrixMode(GL_PROJECTION);
841 glFrustum(-1.0,1.0,-1.0,1.0,1.5,70.0);
842 glMatrixMode(GL_MODELVIEW);
848 ENTRYPOINT void init_engine(ModeInfo *mi)
850 int screen = MI_SCREEN(mi);
853 if (engine == NULL) {
854 if ((engine = (Engine *) calloc(MI_NUM_SCREENS(mi),
855 sizeof(Engine))) == NULL)
859 e->window = MI_WINDOW(mi);
861 e->x = e->y = e->z = e->a = e->an1 = e->nx = e->ny = e->nz =
862 e->dx = e->dy = e->dz = e->da = 0;
865 e->dx = (float)(random() % 1000)/30000;
866 e->dy = (float)(random() % 1000)/30000;
867 e->dz = (float)(random() % 1000)/30000;
869 e->viewer[0] = 0; e->viewer[1] = 2; e->viewer[2] = 18;
870 e->lookat[0] = 0; e->lookat[1] = 0; e->lookat[2] = 0;
874 e->da = (float)(random() % 1000)/125 - 4;
875 e->nx = (float)(random() % 100) / 100;
876 e->ny = (float)(random() % 100) / 100;
877 e->nz = (float)(random() % 100) / 100;
881 double spin_speed = 0.5;
882 double wander_speed = 0.01;
885 e->boom_red[3] = 0.9;
890 e->rot = make_rotator (spin ? spin_speed : 0,
891 spin ? spin_speed : 0,
892 spin ? spin_speed : 0,
894 move ? wander_speed : 0,
897 e->trackball = gltrackball_init ();
900 if ((e->glx_context = init_GL(mi)) != NULL) {
901 reshape_engine(mi, MI_WIDTH(mi), MI_HEIGHT(mi));
905 glClearColor(0.0,0.0,0.0,0.0);
906 glShadeModel(GL_SMOOTH);
907 glPolygonMode(GL_FRONT_AND_BACK,GL_FILL);
908 glEnable(GL_DEPTH_TEST);
909 glEnable(GL_LIGHTING);
911 glEnable(GL_NORMALIZE);
913 e->engineType = find_engine(which_engine);
915 e->engine_name = malloc(200);
916 sprintf (e->engine_name,
918 engines[e->engineType].engineName,
919 (engines[e->engineType].includedAngle == 0 ? "" :
920 engines[e->engineType].includedAngle == 180 ? "Flat " : "V"),
921 engines[e->engineType].cylinders,
922 (engines[e->engineType].includedAngle == 0 ? " Cylinder" : "")
925 e->shaft_polys = makeshaft(e);
926 e->piston_polys = makepiston(e);
927 load_font (mi->dpy, "titleFont", &e->xfont, &e->font_dlist);
931 engine_handle_event (ModeInfo *mi, XEvent *event)
933 Engine *e = &engine[MI_SCREEN(mi)];
935 if (event->xany.type == ButtonPress &&
936 event->xbutton.button == Button1)
938 e->button_down_p = True;
939 gltrackball_start (e->trackball,
940 event->xbutton.x, event->xbutton.y,
941 MI_WIDTH (mi), MI_HEIGHT (mi));
945 else if (event->xany.type == ButtonRelease &&
946 event->xbutton.button == Button1) {
947 e->button_down_p = False;
951 else if (event->xany.type == ButtonPress &&
952 (event->xbutton.button == Button4 ||
953 event->xbutton.button == Button5 ||
954 event->xbutton.button == Button6 ||
955 event->xbutton.button == Button7))
957 gltrackball_mousewheel (e->trackball, event->xbutton.button, 10,
958 !!event->xbutton.state);
961 else if (event->xany.type == MotionNotify &&
963 gltrackball_track (e->trackball,
964 event->xmotion.x, event->xmotion.y,
965 MI_WIDTH (mi), MI_HEIGHT (mi));
971 ENTRYPOINT void draw_engine(ModeInfo *mi)
973 Engine *e = &engine[MI_SCREEN(mi)];
974 Window w = MI_WINDOW(mi);
975 Display *disp = MI_DISPLAY(mi);
980 glXMakeCurrent(disp, w, *(e->glx_context));
983 mi->polygon_count = display(e);
986 print_gl_string (mi->dpy, e->xfont, e->font_dlist,
987 mi->xgwa.width, mi->xgwa.height,
988 10, mi->xgwa.height - 10,
989 e->engine_name, False);
991 if(mi->fps_p) do_fps(mi);
993 glXSwapBuffers(disp, w);
997 release_engine(ModeInfo *mi)
999 if (engine != NULL) {
1000 (void) free((void *) engine);
1006 XSCREENSAVER_MODULE ("Engine", engine)