2 * Copyright (c) 1991, 1993
3 * The Regents of the University of California. All rights reserved.
5 * This code is derived from software contributed to Berkeley by
6 * Jim R. Oldroyd at The Instruction Set and Keith Gabryelski at
7 * Commodore Business Machines.
9 * Redistribution and use in source and binary forms, with or without
10 * modification, are permitted provided that the following conditions
12 * 1. Redistributions of source code must retain the above copyright
13 * notice, this list of conditions and the following disclaimer.
14 * 2. Redistributions in binary form must reproduce the above copyright
15 * notice, this list of conditions and the following disclaimer in the
16 * documentation and/or other materials provided with the distribution.
17 * 3. All advertising materials mentioning features or use of this software
18 * must display the following acknowledgement:
19 * This product includes software developed by the University of
20 * California, Berkeley and its contributors.
21 * 4. Neither the name of the University nor the names of its contributors
22 * may be used to endorse or promote products derived from this software
23 * without specific prior written permission.
25 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
26 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
27 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
28 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
29 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
30 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
31 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
32 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
33 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
34 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
37 * @(#) Copyright (c) 1991, 1993 The Regents of the University of California. All rights reserved.
38 * @(#)quiz.c 8.3 (Berkeley) 5/4/95
39 * $FreeBSD: src/games/quiz/quiz.c,v 1.12 1999/12/12 02:29:54 billf Exp $
40 * $DragonFly: src/games/quiz/quiz.c,v 1.7 2007/09/08 10:49:00 swildner Exp $
43 #include <sys/types.h>
54 #include "pathnames.h"
57 static int catone
, cattwo
, tflag
;
60 char *appdstr (char *, char *, size_t);
61 void downcase (char *);
62 void err (const char *, ...);
63 void get_cats (char *, char *);
64 void get_file (const char *);
65 char *next_cat (char *);
67 void score (u_int
, u_int
, u_int
);
68 void show_index (void);
72 main(int argc
, char **argv
)
75 const char *indexfile
;
80 indexfile
= _PATH_QUIZIDX
;
81 while ((ch
= getopt(argc
, argv
, "i:t")) != -1)
103 get_cats(argv
[0], argv
[1]);
113 get_file(const char *file
)
120 if ((fp
= fopen(file
, "r")) == NULL
)
121 err("%s: %s", file
, strerror(errno
));
125 * Should really free up space from any earlier read list
126 * but there are no reverse pointers to do so with.
130 while ((lp
= fgetln(fp
, &len
)) != NULL
) {
131 if (qp
->q_text
&& qp
->q_text
[strlen(qp
->q_text
) - 1] == '\\')
132 qp
->q_text
= appdstr(qp
->q_text
, lp
, len
);
134 if ((qp
->q_next
= malloc(sizeof(QE
))) == NULL
)
135 err("%s", strerror(errno
));
138 if ((qp
->q_text
= strdup(lp
)) == NULL
)
139 err("%s", strerror(errno
));
140 qp
->q_asked
= qp
->q_answered
= FALSE
;
155 if ((pf
= popen(_PATH_PAGER
, "w")) == NULL
)
156 err("%s: %s", _PATH_PAGER
, strerror(errno
));
157 (void)fprintf(pf
, "Subjects:\n\n");
158 for (qp
= qlist
.q_next
; qp
; qp
= qp
->q_next
) {
159 for (s
= next_cat(qp
->q_text
); s
; s
= next_cat(s
)) {
162 if ((p
= rxp_expand()) != '\0')
163 (void)fprintf(pf
, "%s ", p
);
165 (void)fprintf(pf
, "\n");
167 (void)fprintf(pf
, "\n%s\n%s\n%s\n",
168 "For example, \"quiz victim killer\" prints a victim's name and you reply",
169 "with the killer, and \"quiz killer victim\" works the other way around.",
170 "Type an empty line to get the correct answer.");
175 get_cats(char *cat1
, char *cat2
)
183 for (qp
= qlist
.q_next
; qp
; qp
= qp
->q_next
) {
184 s
= next_cat(qp
->q_text
);
185 catone
= cattwo
= i
= 0;
196 if (catone
&& cattwo
&& catone
!= cattwo
) {
197 if (!rxp_compile(qp
->q_text
))
199 get_file(rxp_expand());
203 err("invalid categories");
212 u_int guesses
, rights
, wrongs
;
214 char *answer
, *s
, *t
, question
[LINE_SZ
];
217 guesses
= rights
= wrongs
= 0;
221 next
= random() % qsize
;
223 for (i
= 0; i
< next
; i
++)
225 while (qp
&& qp
->q_answered
)
231 if (tflag
&& random() % 100 > 20) {
232 /* repeat questions in tutorial mode */
233 while (qp
&& (!qp
->q_asked
|| qp
->q_answered
))
239 for (i
= 0; i
< catone
- 1; i
++)
244 if (!t
|| *t
== '\0') {
245 qp
->q_answered
= TRUE
;
248 (void)strcpy(question
, t
);
250 for (i
= 0; i
< cattwo
- 1; i
++)
255 if (!t
|| *t
== '\0') {
256 qp
->q_answered
= TRUE
;
260 (void)printf("%s?\n", question
);
262 if ((answer
= fgetln(stdin
, &len
)) == NULL
) {
263 score(rights
, wrongs
, guesses
);
266 answer
[len
- 1] = '\0';
268 if (rxp_match(answer
)) {
269 (void)printf("Right!\n");
271 qp
->q_answered
= TRUE
;
274 if (*answer
== '\0') {
275 (void)printf("%s\n", t
);
278 qp
->q_answered
= TRUE
;
281 (void)printf("What?\n");
284 score(rights
, wrongs
, guesses
);
304 appdstr(char *s
, char *tp
, size_t len
)
309 if ((m
= malloc(strlen(s
) + len
+ 1)) == NULL
)
310 err("%s", strerror(errno
));
313 for (; (*mp
++ = *sp
++););
316 if (*(mp
- 1) == '\\')
320 if (mp
[len
- 1] == '\n')
328 score(u_int r
, u_int w
, u_int g
)
330 (void)printf("Rights %d, wrongs %d,", r
, w
);
332 (void)printf(" extra guesses %d,", g
);
333 (void)printf(" score %d%%\n", (r
+ w
+ g
) ? r
* 100 / (r
+ w
+ g
) : 0);
341 for (; (ch
= *p
); ++p
)
342 if (isascii(ch
) && isupper(ch
))
349 (void)fprintf(stderr
, "quiz [-t] [-i file] category1 category2\n");
356 err(const char *fmt
, ...)
360 (void)fprintf(stderr
, "quiz: ");
361 (void)vfprintf(stderr
, fmt
, ap
);
363 (void)fprintf(stderr
, "\n");