|
|
1.1 root 1:
2: static char sccsid[] = " quiz.c 4.2 85/01/09 ";
3:
4: #include <stdio.h>
5: #include <signal.h>
6: #define NF 10
7: #define NL 300
8: #define NC 200
9: #define SL 100
10: #define NA 10
11:
12: int tflag;
13: int xx[NL];
14: char score[NL];
15: int rights;
16: int wrongs;
17: int guesses;
18: FILE *input;
19: int nl = 0;
20: int na = NA;
21: int inc;
22: int ptr = 0;
23: int nc = 0;
24: char line[150];
25: char response[100];
26: char *tmp[NF];
27: int select[NF];
28:
29: readline()
30: {
31: char *t;
32: loop:
33: for(t=line;(*t=getc(input))!=-1;t++) {
34: nc++;
35: if(*t==' '&&(t==line||t[-1]==' '))
36: t--;
37: if(*t=='\n') {
38: if(t[-1]=='\\') /*inexact test*/
39: continue;
40: while(t>line&&t[-1]==' ')
41: *--t = '\n';
42: *++t = 0;
43: return(1);
44: }
45: if(t-line>=NC) {
46: printf("Too hard for me\n");
47: do {
48: *line = getc(input);
49: if(*line==0377)
50: return(0);
51: } while(*line!='\n');
52: goto loop;
53: }
54: }
55: return(0);
56: }
57:
58: char *eu;
59: char *ev;
60: cmp(u,v)
61: char *u,*v;
62: {
63: int x;
64: eu = u;
65: ev = v;
66: x = disj(1);
67: if(x!=1)
68: return(x);
69: return(eat(1,0));
70: }
71:
72: disj(s)
73: {
74: int t, x;
75: char *u;
76: u = eu;
77: t = 0;
78: for(;;) {
79: x = string(s);
80: if(x>1)
81: return(x);
82: switch(*ev) {
83: case 0:
84: case ']':
85: case '}':
86: return(t|x&s);
87: case '|':
88: ev++;
89: t |= s;
90: s = 0;
91: continue;
92: }
93: if(s) eu = u;
94: if(string(0)>1)
95: return(2);
96: switch(*ev) {
97: case 0:
98: case ']':
99: return(0);
100: case '}':
101: return(1);
102: case '|':
103: ev++;
104: continue;
105: default:
106: return(2);
107: }
108: }
109: }
110:
111: string(s)
112: {
113: int x;
114: for(;;) {
115: switch(*ev) {
116: case 0:
117: case '|':
118: case ']':
119: case '}':
120: return(1);
121: case '\\':
122: ev++;
123: if(*ev==0)
124: return(2);
125: if(*ev=='\n') {
126: ev++;
127: continue;
128: }
129: default:
130: if(eat(s,*ev)==1)
131: continue;
132: return(0);
133: case '[':
134: ev++;
135: x = disj(s);
136: if(*ev!=']' || x>1)
137: return(2);
138: ev++;
139: if(s==0)
140: continue;
141: if(x==0)
142: return(0);
143: continue;
144: case '{':
145: ev++;
146: x = disj(s);
147: if(*ev!='}'||x>1)
148: return(2);
149: ev++;
150: continue;
151: }
152: }
153: }
154:
155: eat(s,c)
156: char c;
157: {
158: if(*ev!=c)
159: return(2);
160: if(s==0) {
161: ev++;
162: return(1);
163: }
164: if(fold(*eu)!=fold(c))
165: return(0);
166: eu++;
167: ev++;
168: return(1);
169: }
170:
171: fold(c)
172: char c;
173: {
174: if(c<'A'||c>'Z')
175: return(c);
176: return(c|040);
177: }
178:
179: publish(t)
180: char *t;
181: {
182: ev = t;
183: pub1(1);
184: }
185:
186: pub1(s)
187: {
188: for(;;ev++){
189: switch(*ev) {
190: case '|':
191: s = 0;
192: ev;
193: continue;
194: case ']':
195: case '}':
196: case 0:
197: return;
198: case '[':
199: case '{':
200: ev++;
201: pub1(s);
202: ev;
203: continue;
204: case '\\':
205: if(*++ev=='\n')
206: continue;
207: default:
208: if(s)
209: putchar(*ev);
210: }
211: }
212: }
213:
214: segment(u,w)
215: char *u, *w[];
216: {
217: char *s;
218: int i;
219: char *t;
220: s = u;
221: for(i=0;i<NF;i++) {
222: u = s;
223: t = w[i];
224: while(*s!=':'&&*s!='\n'&&s-u<SL) {
225: if(*s=='\\') {
226: if(s[1] == '\n') {
227: s += 2;
228: continue;
229: }
230: *t++ = *s++;
231: }
232: *t++ = *s++;
233: }
234:
235: while(*s!=':'&&*s!='\n')
236: s++;
237: *t = 0;
238: if(*s++=='\n') {
239: return(i+1);
240: }
241: }
242: printf("Too many facts about one thing\n");
243: }
244:
245: perm(u,m,v,n,p)
246: int p[];
247: char *u[], *v[];
248: {
249: int i, j;
250: int x;
251: for(i=0;i<m;i++) {
252: for(j=0;j<n;j++) {
253: x = cmp(u[i],v[j]);
254: if(x>1) badinfo();
255: if(x==0)
256: continue;
257: p[i] = j;
258: goto uloop;
259: }
260: return(0);
261: uloop: ;
262: }
263: return(1);
264: }
265:
266: find(u,m)
267: char *u[];
268: {
269: int n;
270: while(readline()){
271: n = segment(line,tmp);
272: if(perm(u,m,tmp+1,n-1,select))
273: return(1);
274: }
275: return(0);
276: }
277:
278: readindex()
279: {
280: xx[0] = nc = 0;
281: while(readline()) {
282: xx[++nl] = nc;
283: if(nl>=NL) {
284: printf("I've forgotten some of it;\n");
285: printf("I remember %d items.\n", nl);
286: break;
287: }
288: }
289: }
290:
291: talloc()
292: {
293: int i;
294: char *malloc();
295:
296: for(i=0;i<NF;i++)
297: tmp[i] = malloc(SL);
298: }
299:
300: main(argc,argv)
301: char *argv[];
302: {
303: register j;
304: int i;
305: int x;
306: int z;
307: char *info;
308: int tvec[2];
309: char *t;
310: extern done();
311: int count;
312: info = "/usr/games/lib/quiz.k/index";
313: time(tvec);
314: inc = tvec[1]&077774|01;
315: loop:
316: if(argc>1&&*argv[1]=='-') {
317: switch(argv[1][1]) {
318: case 'i':
319: if(argc>2)
320: info = argv[2];
321: argc -= 2;
322: argv += 2;
323: goto loop;
324: case 't':
325: tflag = 1;
326: argc--;
327: argv++;
328: goto loop;
329: }
330: }
331: input = fopen(info,"r");
332: if(input==NULL) {
333: printf("No info\n");
334: exit(0);
335: }
336: talloc();
337: if(argc<=2)
338: instruct(info);
339: signal(SIGINT,done);
340: argv[argc] = 0;
341: if(find(&argv[1],argc-1)==0)
342: dunno();
343: fclose(input);
344: input = fopen(tmp[0],"r");
345: if(input==NULL)
346: dunno();
347: readindex();
348: if(!tflag || na>nl)
349: na = nl;
350: stdout->_flag |= _IONBF;
351: for(;;) {
352: i = next();
353: fseek(input,xx[i]+0L,0);
354: z = xx[i+1]-xx[i];
355: for(j=0;j<z;j++)
356: line[j] = getc(input);
357: segment(line,tmp);
358: if(*tmp[select[0]] == '\0' || *tmp[select[1]] == '\0') {
359: score[i] = 1;
360: continue;
361: }
362: publish(tmp[select[0]]);
363: printf("\n");
364: for(count=0;;count++) {
365: if(query(response)==0) {
366: publish(tmp[select[1]]);
367: printf("\n");
368: if(count==0) wrongs++;
369: score[i] = tflag?-1:1;
370: break;
371: }
372: x = cmp(response,tmp[select[1]]);
373: if(x>1) badinfo();
374: if(x==1) {
375: printf("Right!\n");
376: if(count==0) rights++;
377: if(++score[i]>=1 && na<nl)
378: na++;
379: break;
380: }
381: printf("What?\n");
382: if(count==0) wrongs++;
383: score[i] = tflag?-1:1;
384: }
385: guesses += count;
386: }
387: }
388:
389: query(r)
390: char *r;
391: {
392: char *t;
393: for(t=r;;t++) {
394: if(read(0,t,1)==0)
395: done();
396: if(*t==' '&&(t==r||t[-1]==' '))
397: t--;
398: if(*t=='\n') {
399: while(t>r&&t[-1]==' ')
400: *--t = '\n';
401: break;
402: }
403: }
404: *t = 0;
405: return(t-r);
406: }
407:
408: next()
409: {
410: int flag;
411: inc = inc*3125&077777;
412: ptr = (inc>>2)%na;
413: flag = 0;
414: while(score[ptr]>0)
415: if(++ptr>=na) {
416: ptr = 0;
417: if(flag) done();
418: flag = 1;
419: }
420: return(ptr);
421: }
422:
423: done()
424: {
425: printf("\nRights %d, wrongs %d, ", rights, wrongs);
426: if(guesses)
427: printf("extra guesses %d, ", guesses);
428: printf("score %d%%\n",100*rights/(rights+wrongs));
429: exit(0);
430: }
431: instruct(info)
432: {
433: char *t;
434: int i, n;
435: printf("Subjects:\n\n");
436: while(readline()) {
437: printf("-");
438: n = segment(line,tmp);
439: for(i=1;i<n;i++) {
440: printf(" ");
441: publish(tmp[i]);
442: }
443: printf("\n");
444: }
445: printf("\n");
446: input = fopen(info,"r");
447: if(input==NULL)
448: abort();
449: readline();
450: segment(line,tmp);
451: printf("For example,\n");
452: printf(" quiz ");
453: publish(tmp[1]);
454: printf(" ");
455: publish(tmp[2]);
456: printf("\nasks you a ");
457: publish(tmp[1]);
458: printf(" and you answer the ");
459: publish(tmp[2]);
460: printf("\n quiz ");
461: publish(tmp[2]);
462: printf(" ");
463: publish(tmp[1]);
464: printf("\nworks the other way around\n");
465: printf("\nType empty line to get correct answer.\n");
466: exit(0);
467: }
468:
469: badinfo(){
470: printf("Bad info %s\n",line);
471: }
472:
473: dunno()
474: {
475: printf("I don't know about that\n");
476: exit(0);
477: }
This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.