#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <GL/glut.h>
GLenum doubleBuffer;
float rotX = 0.0, rotY = 0.0;
int teaList;
long patchData[][16] = {
{102,103,104,105,4,5,6,7,8,9,10,11,12,13,14,15},
{12,13,14,15,16,17,18,19,20,21,22,23,24,25,26,27},
{24,25,26,27,29,30,31,32,33,34,35,36,37,38,39,40},
{96,96,96,96,97,98,99,100,101,101,101,101,0,1,2,3,},
{0,1,2,3,106,107,108,109,110,111,112,113,114,115,116,117},
{118,118,118,118,124,122,119,121,123,126,125,120,40,39,38,37},
{41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56},
{53,54,55,56,57,58,59,60,61,62,63,64,28,65,66,67},
{68,69,70,71,72,73,74,75,76,77,78,79,80,81,82,83},
{80,81,82,83,84,85,86,87,88,89,90,91,92,93,94,95}
};
float cpData[][3] = {
{0.2,0,2.7},{0.2,-0.112,2.7},{0.112,-0.2,2.7},{0,-0.2,2.7},
{1.3375,0,2.53125},{1.3375,-0.749,2.53125},{0.749,-1.3375,2.53125},
{0,-1.3375,2.53125},{1.4375,0,2.53125},{1.4375,-0.805,2.53125},
{0.805,-1.4375,2.53125},{0,-1.4375,2.53125},{1.5,0,2.4},{1.5,-0.84,2.4},
{0.84,-1.5,2.4},{0,-1.5,2.4},{1.75,0,1.875},{1.75,-0.98,1.875},
{0.98,-1.75,1.875},{0,-1.75,1.875},{2,0,1.35},{2,-1.12,1.35},
{1.12,-2,1.35},{0,-2,1.35},{2,0,0.9},{2,-1.12,0.9},{1.12,-2,0.9},
{0,-2,0.9},{-2,0,0.9},{2,0,0.45},{2,-1.12,0.45},{1.12,-2,0.45},
{0,-2,0.45},{1.5,0,0.225},{1.5,-0.84,0.225},{0.84,-1.5,0.225},
{0,-1.5,0.225},{1.5,0,0.15},{1.5,-0.84,0.15},{0.84,-1.5,0.15},
{0,-1.5,0.15},{-1.6,0,2.025},{-1.6,-0.3,2.025},{-1.5,-0.3,2.25},
{-1.5,0,2.25},{-2.3,0,2.025},{-2.3,-0.3,2.025},{-2.5,-0.3,2.25},
{-2.5,0,2.25},{-2.7,0,2.025},{-2.7,-0.3,2.025},{-3,-0.3,2.25},
{-3,0,2.25},{-2.7,0,1.8},{-2.7,-0.3,1.8},{-3,-0.3,1.8},{-3,0,1.8},
{-2.7,0,1.575},{-2.7,-0.3,1.575},{-3,-0.3,1.35},{-3,0,1.35},
{-2.5,0,1.125},{-2.5,-0.3,1.125},{-2.65,-0.3,0.9375},{-2.65,0,0.9375},
{-2,-0.3,0.9},{-1.9,-0.3,0.6},{-1.9,0,0.6},{1.7,0,1.425},
{1.7,-0.66,1.425},{1.7,-0.66,0.6},{1.7,0,0.6},{2.6,0,1.425},
{2.6,-0.66,1.425},{3.1,-0.66,0.825},{3.1,0,0.825},{2.3,0,2.1},
{2.3,-0.25,2.1},{2.4,-0.25,2.025},{2.4,0,2.025},{2.7,0,2.4},
{2.7,-0.25,2.4},{3.3,-0.25,2.4},{3.3,0,2.4},{2.8,0,2.475},
{2.8,-0.25,2.475},{3.525,-0.25,2.49375},{3.525,0,2.49375},
{2.9,0,2.475},{2.9,-0.15,2.475},{3.45,-0.15,2.5125},{3.45,0,2.5125},
{2.8,0,2.4},{2.8,-0.15,2.4},{3.2,-0.15,2.4},{3.2,0,2.4},{0,0,3.15},
{0.8,0,3.15},{0.8,-0.45,3.15},{0.45,-0.8,3.15},{0,-0.8,3.15},
{0,0,2.85},{1.4,0,2.4},{1.4,-0.784,2.4},{0.784,-1.4,2.4},{0,-1.4,2.4},
{0.4,0,2.55},{0.4,-0.224,2.55},{0.224,-0.4,2.55},{0,-0.4,2.55},
{1.3,0,2.55},{1.3,-0.728,2.55},{0.728,-1.3,2.55},{0,-1.3,2.55},
{1.3,0,2.4},{1.3,-0.728,2.4},{0.728,-1.3,2.4},{0,-1.3,2.4},{0,0,0},
{1.425,-0.798,0},{1.5,0,0.075},{1.425,0,0},{0.798,-1.425,0},
{0,-1.5,0.075},{0,-1.425,0},{1.5,-0.84,0.075},{0.84,-1.5,0.075}
};
void Teacup(long grid)
{
float p[4][4][3], q[4][4][3], r[4][4][3], s[4][4][3];
long i, j, k, l;
teaList = 1;
glNewList(teaList, GL_COMPILE);
glPushMatrix();
glRotatef(270.0, 1.0, 0.0, 0.0);
for (i = 0; i < 10; i++) {
for (j = 0; j < 4; j++) {
for (k = 0; k < 4; k++) {
for (l = 0; l < 3; l++) {
p[j][k][l] = cpData[patchData[i][j*4+k]][l];
q[j][k][l] = cpData[patchData[i][j*4+(3-k)]][l];
if (l == 1) {
q[j][k][l] *= -1.0;
}
if (i < 6) {
r[j][k][l] = cpData[patchData[i][j*4+(3-k)]][l];
if (l == 0) {
r[j][k][l] *= -1.0;
}
s[j][k][l] = cpData[patchData[i][j*4+k]][l];
if (l == 0) {
s[j][k][l] *= -1.0;
}
if (l == 1) {
s[j][k][l] *= -1.0;
}
}
}
}
}
glMap2f(GL_MAP2_VERTEX_3, 0, 1, 3, 4, 0, 1, 12, 4, &p[0][0][0]);
glEnable(GL_MAP2_VERTEX_3);
glMapGrid2f(grid, 0.0, 1.0, grid, 0.0, 1.0);
glEvalMesh2(GL_FILL, 0, grid, 0, grid);
glMap2f(GL_MAP2_VERTEX_3, 0, 1, 3, 4, 0, 1, 12, 4, &q[0][0][0]);
glEvalMesh2(GL_FILL, 0, grid, 0, grid);
if (i < 6) {
glMap2f(GL_MAP2_VERTEX_3, 0, 1, 3, 4, 0, 1, 12, 4, &r[0][0][0]);
glEvalMesh2(GL_FILL, 0, grid, 0, grid);
glMap2f(GL_MAP2_VERTEX_3, 0, 1, 3, 4, 0, 1, 12, 4, &s[0][0][0]);
glEvalMesh2(GL_FILL, 0, grid, 0, grid);
}
}
glDisable(GL_MAP2_VERTEX_3);
glPopMatrix();
glEndList();
}
static void Init(void)
{
float position[] = {0.0, 3.0, 3.0, 0.0};
float local_view[] = {0.0};
float ambient[] = {0.1745, 0.01175, 0.01175};
float diffuse[] = {0.61424, 0.04136, 0.04136};
float specular[] = {0.727811, 0.626959, 0.626959};
glEnable(GL_DEPTH_TEST);
glDepthFunc(GL_LESS);
glLightfv(GL_LIGHT0, GL_POSITION, position);
glLightModelfv(GL_LIGHT_MODEL_LOCAL_VIEWER, local_view);
glFrontFace(GL_CW);
glEnable(GL_LIGHTING);
glEnable(GL_LIGHT0);
glEnable(GL_AUTO_NORMAL);
glEnable(GL_NORMALIZE);
glMaterialfv(GL_FRONT, GL_AMBIENT, ambient);
glMaterialfv(GL_FRONT, GL_DIFFUSE, diffuse);
glMaterialfv(GL_FRONT, GL_SPECULAR, specular);
glMaterialf(GL_FRONT, GL_SHININESS, 0.6*128.0);
glClearColor(0.5, 0.5, 0.5, 1.0);
glColor3f(1.0, 1.0, 1.0);
Teacup(14);
}
static void Reshape(int w, int h)
{
glViewport(0, 0, (GLint)w, (GLint)h);
glMatrixMode(GL_PROJECTION);
glLoadIdentity();
glOrtho(-6.0, 6.0, -6.0, 6.0, -1.0, 10.0);
glMatrixMode(GL_MODELVIEW);
glLoadIdentity();
}
static void Key(unsigned char key, int x, int y)
{
switch (key) {
case 27:
exit(0);
}
}
static void SpecialKey(int key, int x, int y)
{
switch (key) {
case GLUT_KEY_UP:
rotX -= 20.0;
glutPostRedisplay();
break;
case GLUT_KEY_DOWN:
rotX += 20.0;
glutPostRedisplay();
break;
case GLUT_KEY_LEFT:
rotY -= 20.0;
glutPostRedisplay();
break;
case GLUT_KEY_RIGHT:
rotY += 20.0;
glutPostRedisplay();
break;
}
}
static void Draw(void)
{
glClear(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT);
glPushMatrix();
glTranslatef(0.0, 0.0, -5.0);
glRotatef(rotY, 0.0,1.0,0.0);
glRotatef(rotX, 1.0,0.0,0.0);
glCallList(teaList);
glPopMatrix();
if (doubleBuffer) {
glutSwapBuffers();
} else {
glFlush();
}
}
static void Args(int argc, char **argv)
{
GLint i;
doubleBuffer = GL_FALSE;
for (i = 1; i < argc; i++) {
if (strcmp(argv[i], "-sb") == 0) {
doubleBuffer = GL_FALSE;
} else if (strcmp(argv[i], "-db") == 0) {
doubleBuffer = GL_TRUE;
}
}
}
int main(int argc, char **argv)
{
GLenum type;
glutInit(&argc, argv);
Args(argc, argv);
type = GLUT_RGB | GLUT_DEPTH;
type |= (doubleBuffer) ? GLUT_DOUBLE : GLUT_SINGLE;
glutInitDisplayMode(type);
glutInitWindowSize(300, 300);
glutCreateWindow("TeaPot");
Init();
glutReshapeFunc(Reshape);
glutKeyboardFunc(Key);
glutSpecialFunc(SpecialKey);
glutDisplayFunc(Draw);
glutMainLoop();
}
I2luY2x1ZGUgPHN0ZGlvLmg+CiNpbmNsdWRlIDxzdHJpbmcuaD4KI2luY2x1ZGUgPHN0ZGxpYi5oPgojaW5jbHVkZSA8R0wvZ2x1dC5oPgoKCkdMZW51bSBkb3VibGVCdWZmZXI7CgpmbG9hdCByb3RYID0gMC4wLCByb3RZID0gMC4wOwppbnQgdGVhTGlzdDsKCmxvbmcgcGF0Y2hEYXRhW11bMTZdID0gewogICAgezEwMiwxMDMsMTA0LDEwNSw0LDUsNiw3LDgsOSwxMCwxMSwxMiwxMywxNCwxNX0sCiAgICB7MTIsMTMsMTQsMTUsMTYsMTcsMTgsMTksMjAsMjEsMjIsMjMsMjQsMjUsMjYsMjd9LAogICAgezI0LDI1LDI2LDI3LDI5LDMwLDMxLDMyLDMzLDM0LDM1LDM2LDM3LDM4LDM5LDQwfSwKICAgIHs5Niw5Niw5Niw5Niw5Nyw5OCw5OSwxMDAsMTAxLDEwMSwxMDEsMTAxLDAsMSwyLDMsfSwKICAgIHswLDEsMiwzLDEwNiwxMDcsMTA4LDEwOSwxMTAsMTExLDExMiwxMTMsMTE0LDExNSwxMTYsMTE3fSwKICAgIHsxMTgsMTE4LDExOCwxMTgsMTI0LDEyMiwxMTksMTIxLDEyMywxMjYsMTI1LDEyMCw0MCwzOSwzOCwzN30sCiAgICB7NDEsNDIsNDMsNDQsNDUsNDYsNDcsNDgsNDksNTAsNTEsNTIsNTMsNTQsNTUsNTZ9LAogICAgezUzLDU0LDU1LDU2LDU3LDU4LDU5LDYwLDYxLDYyLDYzLDY0LDI4LDY1LDY2LDY3fSwKICAgIHs2OCw2OSw3MCw3MSw3Miw3Myw3NCw3NSw3Niw3Nyw3OCw3OSw4MCw4MSw4Miw4M30sCiAgICB7ODAsODEsODIsODMsODQsODUsODYsODcsODgsODksOTAsOTEsOTIsOTMsOTQsOTV9Cn07CgpmbG9hdCBjcERhdGFbXVszXSA9IHsKICAgIHswLjIsMCwyLjd9LHswLjIsLTAuMTEyLDIuN30sezAuMTEyLC0wLjIsMi43fSx7MCwtMC4yLDIuN30sCiAgICB7MS4zMzc1LDAsMi41MzEyNX0sezEuMzM3NSwtMC43NDksMi41MzEyNX0sezAuNzQ5LC0xLjMzNzUsMi41MzEyNX0sCiAgICB7MCwtMS4zMzc1LDIuNTMxMjV9LHsxLjQzNzUsMCwyLjUzMTI1fSx7MS40Mzc1LC0wLjgwNSwyLjUzMTI1fSwKICAgIHswLjgwNSwtMS40Mzc1LDIuNTMxMjV9LHswLC0xLjQzNzUsMi41MzEyNX0sezEuNSwwLDIuNH0sezEuNSwtMC44NCwyLjR9LAogICAgezAuODQsLTEuNSwyLjR9LHswLC0xLjUsMi40fSx7MS43NSwwLDEuODc1fSx7MS43NSwtMC45OCwxLjg3NX0sCiAgICB7MC45OCwtMS43NSwxLjg3NX0sezAsLTEuNzUsMS44NzV9LHsyLDAsMS4zNX0sezIsLTEuMTIsMS4zNX0sCiAgICB7MS4xMiwtMiwxLjM1fSx7MCwtMiwxLjM1fSx7MiwwLDAuOX0sezIsLTEuMTIsMC45fSx7MS4xMiwtMiwwLjl9LAogICAgezAsLTIsMC45fSx7LTIsMCwwLjl9LHsyLDAsMC40NX0sezIsLTEuMTIsMC40NX0sezEuMTIsLTIsMC40NX0sCiAgICB7MCwtMiwwLjQ1fSx7MS41LDAsMC4yMjV9LHsxLjUsLTAuODQsMC4yMjV9LHswLjg0LC0xLjUsMC4yMjV9LAogICAgezAsLTEuNSwwLjIyNX0sezEuNSwwLDAuMTV9LHsxLjUsLTAuODQsMC4xNX0sezAuODQsLTEuNSwwLjE1fSwKICAgIHswLC0xLjUsMC4xNX0sey0xLjYsMCwyLjAyNX0sey0xLjYsLTAuMywyLjAyNX0sey0xLjUsLTAuMywyLjI1fSwKICAgIHstMS41LDAsMi4yNX0sey0yLjMsMCwyLjAyNX0sey0yLjMsLTAuMywyLjAyNX0sey0yLjUsLTAuMywyLjI1fSwKICAgIHstMi41LDAsMi4yNX0sey0yLjcsMCwyLjAyNX0sey0yLjcsLTAuMywyLjAyNX0sey0zLC0wLjMsMi4yNX0sCiAgICB7LTMsMCwyLjI1fSx7LTIuNywwLDEuOH0sey0yLjcsLTAuMywxLjh9LHstMywtMC4zLDEuOH0sey0zLDAsMS44fSwKICAgIHstMi43LDAsMS41NzV9LHstMi43LC0wLjMsMS41NzV9LHstMywtMC4zLDEuMzV9LHstMywwLDEuMzV9LAogICAgey0yLjUsMCwxLjEyNX0sey0yLjUsLTAuMywxLjEyNX0sey0yLjY1LC0wLjMsMC45Mzc1fSx7LTIuNjUsMCwwLjkzNzV9LAogICAgey0yLC0wLjMsMC45fSx7LTEuOSwtMC4zLDAuNn0sey0xLjksMCwwLjZ9LHsxLjcsMCwxLjQyNX0sCiAgICB7MS43LC0wLjY2LDEuNDI1fSx7MS43LC0wLjY2LDAuNn0sezEuNywwLDAuNn0sezIuNiwwLDEuNDI1fSwKICAgIHsyLjYsLTAuNjYsMS40MjV9LHszLjEsLTAuNjYsMC44MjV9LHszLjEsMCwwLjgyNX0sezIuMywwLDIuMX0sCiAgICB7Mi4zLC0wLjI1LDIuMX0sezIuNCwtMC4yNSwyLjAyNX0sezIuNCwwLDIuMDI1fSx7Mi43LDAsMi40fSwKICAgIHsyLjcsLTAuMjUsMi40fSx7My4zLC0wLjI1LDIuNH0sezMuMywwLDIuNH0sezIuOCwwLDIuNDc1fSwKICAgIHsyLjgsLTAuMjUsMi40NzV9LHszLjUyNSwtMC4yNSwyLjQ5Mzc1fSx7My41MjUsMCwyLjQ5Mzc1fSwKICAgIHsyLjksMCwyLjQ3NX0sezIuOSwtMC4xNSwyLjQ3NX0sezMuNDUsLTAuMTUsMi41MTI1fSx7My40NSwwLDIuNTEyNX0sCiAgICB7Mi44LDAsMi40fSx7Mi44LC0wLjE1LDIuNH0sezMuMiwtMC4xNSwyLjR9LHszLjIsMCwyLjR9LHswLDAsMy4xNX0sCiAgICB7MC44LDAsMy4xNX0sezAuOCwtMC40NSwzLjE1fSx7MC40NSwtMC44LDMuMTV9LHswLC0wLjgsMy4xNX0sCiAgICB7MCwwLDIuODV9LHsxLjQsMCwyLjR9LHsxLjQsLTAuNzg0LDIuNH0sezAuNzg0LC0xLjQsMi40fSx7MCwtMS40LDIuNH0sCiAgICB7MC40LDAsMi41NX0sezAuNCwtMC4yMjQsMi41NX0sezAuMjI0LC0wLjQsMi41NX0sezAsLTAuNCwyLjU1fSwKICAgIHsxLjMsMCwyLjU1fSx7MS4zLC0wLjcyOCwyLjU1fSx7MC43MjgsLTEuMywyLjU1fSx7MCwtMS4zLDIuNTV9LAogICAgezEuMywwLDIuNH0sezEuMywtMC43MjgsMi40fSx7MC43MjgsLTEuMywyLjR9LHswLC0xLjMsMi40fSx7MCwwLDB9LAogICAgezEuNDI1LC0wLjc5OCwwfSx7MS41LDAsMC4wNzV9LHsxLjQyNSwwLDB9LHswLjc5OCwtMS40MjUsMH0sCiAgICB7MCwtMS41LDAuMDc1fSx7MCwtMS40MjUsMH0sezEuNSwtMC44NCwwLjA3NX0sezAuODQsLTEuNSwwLjA3NX0KfTsKCgp2b2lkIFRlYWN1cChsb25nIGdyaWQpCnsKICAgIGZsb2F0IHBbNF1bNF1bM10sIHFbNF1bNF1bM10sIHJbNF1bNF1bM10sIHNbNF1bNF1bM107CiAgICBsb25nIGksIGosIGssIGw7CgogICAgdGVhTGlzdCA9IDE7CiAgICBnbE5ld0xpc3QodGVhTGlzdCwgR0xfQ09NUElMRSk7CiAgICBnbFB1c2hNYXRyaXgoKTsKICAgIGdsUm90YXRlZigyNzAuMCwgMS4wLCAwLjAsIDAuMCk7CiAgICBmb3IgKGkgPSAwOyBpIDwgMTA7IGkrKykgewoJZm9yIChqID0gMDsgaiA8IDQ7IGorKykgewoJICAgIGZvciAoayA9IDA7IGsgPCA0OyBrKyspIHsKCQlmb3IgKGwgPSAwOyBsIDwgMzsgbCsrKSB7CgkJICAgIHBbal1ba11bbF0gPSBjcERhdGFbcGF0Y2hEYXRhW2ldW2oqNCtrXV1bbF07CgkJICAgIHFbal1ba11bbF0gPSBjcERhdGFbcGF0Y2hEYXRhW2ldW2oqNCsoMy1rKV1dW2xdOwoJCSAgICBpZiAobCA9PSAxKSB7CgkJCXFbal1ba11bbF0gKj0gLTEuMDsKCQkgICAgfQoJCSAgICBpZiAoaSA8IDYpIHsKCQkJcltqXVtrXVtsXSA9IGNwRGF0YVtwYXRjaERhdGFbaV1baio0KygzLWspXV1bbF07CgkJCWlmIChsID09IDApIHsKCQkJICAgIHJbal1ba11bbF0gKj0gLTEuMDsKCQkJfQoJCQlzW2pdW2tdW2xdID0gY3BEYXRhW3BhdGNoRGF0YVtpXVtqKjQra11dW2xdOwoJCQlpZiAobCA9PSAwKSB7CgkJCSAgICBzW2pdW2tdW2xdICo9IC0xLjA7CgkJCX0KCQkJaWYgKGwgPT0gMSkgewoJCQkgICAgc1tqXVtrXVtsXSAqPSAtMS4wOwoJCQl9CgkJICAgIH0KCQl9CgkgICAgfQoJfQoJZ2xNYXAyZihHTF9NQVAyX1ZFUlRFWF8zLCAwLCAxLCAzLCA0LCAwLCAxLCAxMiwgNCwgJnBbMF1bMF1bMF0pOwoJZ2xFbmFibGUoR0xfTUFQMl9WRVJURVhfMyk7CglnbE1hcEdyaWQyZihncmlkLCAwLjAsIDEuMCwgZ3JpZCwgMC4wLCAxLjApOwoJZ2xFdmFsTWVzaDIoR0xfRklMTCwgMCwgZ3JpZCwgMCwgZ3JpZCk7CglnbE1hcDJmKEdMX01BUDJfVkVSVEVYXzMsIDAsIDEsIDMsIDQsIDAsIDEsIDEyLCA0LCAmcVswXVswXVswXSk7CglnbEV2YWxNZXNoMihHTF9GSUxMLCAwLCBncmlkLCAwLCBncmlkKTsKCWlmIChpIDwgNikgewoJICAgIGdsTWFwMmYoR0xfTUFQMl9WRVJURVhfMywgMCwgMSwgMywgNCwgMCwgMSwgMTIsIDQsICZyWzBdWzBdWzBdKTsKCSAgICBnbEV2YWxNZXNoMihHTF9GSUxMLCAwLCBncmlkLCAwLCBncmlkKTsKCSAgICBnbE1hcDJmKEdMX01BUDJfVkVSVEVYXzMsIDAsIDEsIDMsIDQsIDAsIDEsIDEyLCA0LCAmc1swXVswXVswXSk7CgkgICAgZ2xFdmFsTWVzaDIoR0xfRklMTCwgMCwgZ3JpZCwgMCwgZ3JpZCk7Cgl9CiAgICB9CiAgICBnbERpc2FibGUoR0xfTUFQMl9WRVJURVhfMyk7CiAgICBnbFBvcE1hdHJpeCgpOwogICAgZ2xFbmRMaXN0KCk7Cn0KCnN0YXRpYyB2b2lkIEluaXQodm9pZCkKewogICAgZmxvYXQgcG9zaXRpb25bXSA9IHswLjAsIDMuMCwgMy4wLCAwLjB9OwogICAgZmxvYXQgbG9jYWxfdmlld1tdID0gezAuMH07CiAgICBmbG9hdCBhbWJpZW50W10gPSB7MC4xNzQ1LCAwLjAxMTc1LCAwLjAxMTc1fTsKICAgIGZsb2F0IGRpZmZ1c2VbXSA9IHswLjYxNDI0LCAwLjA0MTM2LCAwLjA0MTM2fTsKICAgIGZsb2F0IHNwZWN1bGFyW10gPSB7MC43Mjc4MTEsIDAuNjI2OTU5LCAwLjYyNjk1OX07CgogICAgZ2xFbmFibGUoR0xfREVQVEhfVEVTVCk7CiAgICBnbERlcHRoRnVuYyhHTF9MRVNTKTsKCiAgICBnbExpZ2h0ZnYoR0xfTElHSFQwLCBHTF9QT1NJVElPTiwgcG9zaXRpb24pOwogICAgZ2xMaWdodE1vZGVsZnYoR0xfTElHSFRfTU9ERUxfTE9DQUxfVklFV0VSLCBsb2NhbF92aWV3KTsKCiAgICBnbEZyb250RmFjZShHTF9DVyk7CiAgICBnbEVuYWJsZShHTF9MSUdIVElORyk7CiAgICBnbEVuYWJsZShHTF9MSUdIVDApOwogICAgZ2xFbmFibGUoR0xfQVVUT19OT1JNQUwpOwogICAgZ2xFbmFibGUoR0xfTk9STUFMSVpFKTsKCiAgICBnbE1hdGVyaWFsZnYoR0xfRlJPTlQsIEdMX0FNQklFTlQsIGFtYmllbnQpOwogICAgZ2xNYXRlcmlhbGZ2KEdMX0ZST05ULCBHTF9ESUZGVVNFLCBkaWZmdXNlKTsKICAgIGdsTWF0ZXJpYWxmdihHTF9GUk9OVCwgR0xfU1BFQ1VMQVIsIHNwZWN1bGFyKTsKICAgIGdsTWF0ZXJpYWxmKEdMX0ZST05ULCBHTF9TSElOSU5FU1MsIDAuNioxMjguMCk7CgogICAgZ2xDbGVhckNvbG9yKDAuNSwgMC41LCAwLjUsIDEuMCk7CiAgICBnbENvbG9yM2YoMS4wLCAxLjAsIDEuMCk7CgogICAgVGVhY3VwKDE0KTsKfQoKc3RhdGljIHZvaWQgUmVzaGFwZShpbnQgdywgaW50IGgpCnsKCiAgICBnbFZpZXdwb3J0KDAsIDAsIChHTGludCl3LCAoR0xpbnQpaCk7CgogICAgZ2xNYXRyaXhNb2RlKEdMX1BST0pFQ1RJT04pOwogICAgZ2xMb2FkSWRlbnRpdHkoKTsKICAgIGdsT3J0aG8oLTYuMCwgNi4wLCAtNi4wLCA2LjAsIC0xLjAsIDEwLjApOwogICAgZ2xNYXRyaXhNb2RlKEdMX01PREVMVklFVyk7CiAgICBnbExvYWRJZGVudGl0eSgpOwp9CgpzdGF0aWMgdm9pZCBLZXkodW5zaWduZWQgY2hhciBrZXksIGludCB4LCBpbnQgeSkKewoKICAgIHN3aXRjaCAoa2V5KSB7CiAgICAgIGNhc2UgMjc6CglleGl0KDApOwogICAgfQp9CgpzdGF0aWMgdm9pZCBTcGVjaWFsS2V5KGludCBrZXksIGludCB4LCBpbnQgeSkKewoKICAgIHN3aXRjaCAoa2V5KSB7CiAgICAgIGNhc2UgR0xVVF9LRVlfVVA6Cglyb3RYIC09IDIwLjA7CglnbHV0UG9zdFJlZGlzcGxheSgpOwoJYnJlYWs7CiAgICAgIGNhc2UgR0xVVF9LRVlfRE9XTjoKCXJvdFggKz0gMjAuMDsKCWdsdXRQb3N0UmVkaXNwbGF5KCk7CglicmVhazsKICAgICAgY2FzZSBHTFVUX0tFWV9MRUZUOgoJcm90WSAtPSAyMC4wOwoJZ2x1dFBvc3RSZWRpc3BsYXkoKTsKCWJyZWFrOwogICAgICBjYXNlIEdMVVRfS0VZX1JJR0hUOgoJcm90WSArPSAyMC4wOwoJZ2x1dFBvc3RSZWRpc3BsYXkoKTsKCWJyZWFrOwogICAgfQp9CgpzdGF0aWMgdm9pZCBEcmF3KHZvaWQpCnsKCiAgICBnbENsZWFyKEdMX0NPTE9SX0JVRkZFUl9CSVR8R0xfREVQVEhfQlVGRkVSX0JJVCk7CgogICAgZ2xQdXNoTWF0cml4KCk7CgogICAgZ2xUcmFuc2xhdGVmKDAuMCwgMC4wLCAtNS4wKTsKICAgIGdsUm90YXRlZihyb3RZLCAwLjAsMS4wLDAuMCk7CiAgICBnbFJvdGF0ZWYocm90WCwgMS4wLDAuMCwwLjApOwoKICAgIGdsQ2FsbExpc3QodGVhTGlzdCk7CgogICAgZ2xQb3BNYXRyaXgoKTsKCiAgICBpZiAoZG91YmxlQnVmZmVyKSB7CglnbHV0U3dhcEJ1ZmZlcnMoKTsKICAgIH0gZWxzZSB7CglnbEZsdXNoKCk7CiAgICB9Cn0KCnN0YXRpYyB2b2lkIEFyZ3MoaW50IGFyZ2MsIGNoYXIgKiphcmd2KQp7CiAgICBHTGludCBpOwoKICAgIGRvdWJsZUJ1ZmZlciA9IEdMX0ZBTFNFOwoKICAgIGZvciAoaSA9IDE7IGkgPCBhcmdjOyBpKyspIHsKCWlmIChzdHJjbXAoYXJndltpXSwgIi1zYiIpID09IDApIHsKCSAgICBkb3VibGVCdWZmZXIgPSBHTF9GQUxTRTsKCX0gZWxzZSBpZiAoc3RyY21wKGFyZ3ZbaV0sICItZGIiKSA9PSAwKSB7CgkgICAgZG91YmxlQnVmZmVyID0gR0xfVFJVRTsKCX0KICAgIH0KfQoKaW50IG1haW4oaW50IGFyZ2MsIGNoYXIgKiphcmd2KQp7CiAgICBHTGVudW0gdHlwZTsKCiAgICBnbHV0SW5pdCgmYXJnYywgYXJndik7CiAgICBBcmdzKGFyZ2MsIGFyZ3YpOwoKICAgIHR5cGUgPSBHTFVUX1JHQiB8IEdMVVRfREVQVEg7CiAgICB0eXBlIHw9IChkb3VibGVCdWZmZXIpID8gR0xVVF9ET1VCTEUgOiBHTFVUX1NJTkdMRTsKICAgIGdsdXRJbml0RGlzcGxheU1vZGUodHlwZSk7CiAgICBnbHV0SW5pdFdpbmRvd1NpemUoMzAwLCAzMDApOwogICAgZ2x1dENyZWF0ZVdpbmRvdygiVGVhUG90Iik7CgogICAgSW5pdCgpOwoKICAgIGdsdXRSZXNoYXBlRnVuYyhSZXNoYXBlKTsKICAgIGdsdXRLZXlib2FyZEZ1bmMoS2V5KTsKICAgIGdsdXRTcGVjaWFsRnVuYyhTcGVjaWFsS2V5KTsKICAgIGdsdXREaXNwbGF5RnVuYyhEcmF3KTsKICAgIGdsdXRNYWluTG9vcCgpOwp9Cg==