Document, document, document.
This commit is contained in:
parent
24575e61ec
commit
0efd68d3f0
113
src/nyancat.c
113
src/nyancat.c
@ -47,29 +47,55 @@
|
|||||||
#include <signal.h>
|
#include <signal.h>
|
||||||
#include <time.h>
|
#include <time.h>
|
||||||
#include <setjmp.h>
|
#include <setjmp.h>
|
||||||
|
|
||||||
|
/*
|
||||||
|
* telnet.h contains some #defines for the various
|
||||||
|
* commands, escape characters, and modes for telnet.
|
||||||
|
* (it surprises some people that telnet is, really,
|
||||||
|
* a protocol, and not just raw text transmission)
|
||||||
|
*/
|
||||||
#include "telnet.h"
|
#include "telnet.h"
|
||||||
|
|
||||||
/* The animation frames are stored separately. */
|
/*
|
||||||
|
* The animation frames are stored separately in
|
||||||
|
* this header so they don't clutter the core source
|
||||||
|
*/
|
||||||
#include "animation.h"
|
#include "animation.h"
|
||||||
|
|
||||||
/* The color palette is not yet set */
|
/*
|
||||||
|
* Color palette to use for final output
|
||||||
|
* Specifically, this should be either control sequences
|
||||||
|
* or raw characters (ie, for vt220 mode)
|
||||||
|
*/
|
||||||
char * colors[256] = {NULL};
|
char * colors[256] = {NULL};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* For most modes, we ouput spaces, but for some
|
* For most modes, we ouput spaces, but for some
|
||||||
* we will use block characters.
|
* we will use block characters (or even nothing)
|
||||||
*/
|
*/
|
||||||
char * output = " ";
|
char * output = " ";
|
||||||
|
|
||||||
/* Telnet mode? */
|
/*
|
||||||
|
* Are we currently in telnet mode?
|
||||||
|
*/
|
||||||
int telnet = 0;
|
int telnet = 0;
|
||||||
|
|
||||||
/* I refuse to include libm */
|
/*
|
||||||
double log10(double x) {
|
* Environment to use for setjmp/longjmp
|
||||||
int val = (int)x;
|
* when breaking out of options handler
|
||||||
|
*/
|
||||||
|
jmp_buf environment;
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* I refuse to include libm to keep this low
|
||||||
|
* on external dependencies.
|
||||||
|
* */
|
||||||
|
int digits(int val) {
|
||||||
int d = 1, c;
|
int d = 1, c;
|
||||||
if (val >= 0) for (c = 10; c <= val; c *= 10) d++;
|
if (val >= 0) for (c = 10; c <= val; c *= 10) d++;
|
||||||
else for (c = -10 ; c >= val; c *= 10) d++;
|
else for (c = -10 ; c >= val; c *= 10) d++;
|
||||||
return (c < 0) ? (double)++d : (double)d;
|
return (c < 0) ? ++d : d;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -90,8 +116,6 @@ void SIGINT_handler(int sig){
|
|||||||
exit(0);
|
exit(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
jmp_buf environment;
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Handle the alarm which breaks us off of options
|
* Handle the alarm which breaks us off of options
|
||||||
* handling if we didn't receive a terminal
|
* handling if we didn't receive a terminal
|
||||||
@ -102,16 +126,23 @@ void SIGALRM_handler(int sig) {
|
|||||||
/* Unreachable */
|
/* Unreachable */
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Telnet requires us to send a specific sequence
|
||||||
|
* for a line break (\r\000\n), so let's make it happy.
|
||||||
|
*/
|
||||||
void newline(int n) {
|
void newline(int n) {
|
||||||
int i = 0;
|
int i = 0;
|
||||||
for (i = 0; i < n; ++i) {
|
for (i = 0; i < n; ++i) {
|
||||||
if (telnet) {
|
/* We will send `n` linefeeds to the client */
|
||||||
putc('\r', stdout);
|
if (telnet) {
|
||||||
putc(0, stdout);
|
/* Send the telnet newline sequence */
|
||||||
putc('\n', stdout);
|
putc('\r', stdout);
|
||||||
} else {
|
putc(0, stdout);
|
||||||
putc('\n', stdout);
|
putc('\n', stdout);
|
||||||
}
|
} else {
|
||||||
|
/* Send a regular line feed */
|
||||||
|
putc('\n', stdout);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -155,6 +186,10 @@ void set_options() {
|
|||||||
telnet_willack[NEW_ENVIRON] = DO;
|
telnet_willack[NEW_ENVIRON] = DO;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Send a command (cmd) to the telnet client
|
||||||
|
* Also does special handling for DO/DONT/WILL/WONT
|
||||||
|
*/
|
||||||
void send_command(int cmd, int opt) {
|
void send_command(int cmd, int opt) {
|
||||||
/* Send a command to the telnet client */
|
/* Send a command to the telnet client */
|
||||||
if (cmd == DO || cmd == DONT) {
|
if (cmd == DO || cmd == DONT) {
|
||||||
@ -234,6 +269,8 @@ int main(int argc, char ** argv) {
|
|||||||
/* End of extended option mode */
|
/* End of extended option mode */
|
||||||
sb_mode = 0;
|
sb_mode = 0;
|
||||||
if (sb[0] == TTYPE) {
|
if (sb[0] == TTYPE) {
|
||||||
|
/* This was a response to the TTYPE command, meaning
|
||||||
|
* that this should be a terminal type */
|
||||||
alarm(0);
|
alarm(0);
|
||||||
strcpy(term, &sb[2]);
|
strcpy(term, &sb[2]);
|
||||||
goto ready;
|
goto ready;
|
||||||
@ -249,6 +286,7 @@ int main(int argc, char ** argv) {
|
|||||||
/* Will / Won't Negotiation */
|
/* Will / Won't Negotiation */
|
||||||
opt = getchar();
|
opt = getchar();
|
||||||
if (!telnet_willack[opt]) {
|
if (!telnet_willack[opt]) {
|
||||||
|
/* We default to WONT */
|
||||||
telnet_willack[opt] = WONT;
|
telnet_willack[opt] = WONT;
|
||||||
}
|
}
|
||||||
send_command(telnet_willack[opt], opt);
|
send_command(telnet_willack[opt], opt);
|
||||||
@ -264,6 +302,7 @@ int main(int argc, char ** argv) {
|
|||||||
/* Do / Don't Negotation */
|
/* Do / Don't Negotation */
|
||||||
opt = getchar();
|
opt = getchar();
|
||||||
if (!telnet_options[opt]) {
|
if (!telnet_options[opt]) {
|
||||||
|
/* We default to DONT */
|
||||||
telnet_options[opt] = DONT;
|
telnet_options[opt] = DONT;
|
||||||
}
|
}
|
||||||
send_command(telnet_options[opt], opt);
|
send_command(telnet_options[opt], opt);
|
||||||
@ -281,7 +320,7 @@ int main(int argc, char ** argv) {
|
|||||||
memset(sb, 0, 1024);
|
memset(sb, 0, 1024);
|
||||||
break;
|
break;
|
||||||
case IAC:
|
case IAC:
|
||||||
/* Connection Closed During Negotiation */
|
/* IAC IAC? That's probably not right. */
|
||||||
done = 1;
|
done = 1;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
@ -290,6 +329,13 @@ int main(int argc, char ** argv) {
|
|||||||
} else if (sb_mode) {
|
} else if (sb_mode) {
|
||||||
/* Extended Option Mode -> Accept character */
|
/* Extended Option Mode -> Accept character */
|
||||||
if (sb_len < 1023) {
|
if (sb_len < 1023) {
|
||||||
|
/* Append this character to the SB string,
|
||||||
|
* but only if it doesn't put us over
|
||||||
|
* our limit; honestly, we shouldn't hit
|
||||||
|
* the limit, as we're only collecting characters
|
||||||
|
* for a terminal type, but better safe than
|
||||||
|
* sorry (and vulernable).
|
||||||
|
*/
|
||||||
sb[sb_len] = i;
|
sb[sb_len] = i;
|
||||||
sb_len++;
|
sb_len++;
|
||||||
}
|
}
|
||||||
@ -298,11 +344,20 @@ int main(int argc, char ** argv) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
/* Otherwise, we were run standalone, find the terminal type from the environement */
|
/* We are running standalone, retrieve the
|
||||||
|
* terminal type from the environment. */
|
||||||
char * nterm = getenv("TERM");
|
char * nterm = getenv("TERM");
|
||||||
strcpy(term, nterm);
|
strcpy(term, nterm);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Labels. Yes, and I used a goto.
|
||||||
|
* If you're going to complain about this, you
|
||||||
|
* really need to reevaluate your approach to getting
|
||||||
|
* things done. This works, it works well, and there
|
||||||
|
* is absolutely, positively nothing wrong with using
|
||||||
|
* goto in C, so I'm going to do it.
|
||||||
|
*/
|
||||||
ready:
|
ready:
|
||||||
|
|
||||||
/* Convert the entire terminal string to lower case */
|
/* Convert the entire terminal string to lower case */
|
||||||
@ -478,10 +533,10 @@ ready:
|
|||||||
time_t start, current;
|
time_t start, current;
|
||||||
time(&start);
|
time(&start);
|
||||||
|
|
||||||
int playing = 1;
|
int playing = 1; /* Animation should continue [left here for modifications] */
|
||||||
size_t i = 0;
|
size_t i = 0; /* Current frame # */
|
||||||
char last = 0;
|
char last = 0; /* Last color index rendered */
|
||||||
size_t y, x;
|
size_t y, x; /* x/y coordinates of what we're drawing */
|
||||||
while (playing) {
|
while (playing) {
|
||||||
/* Render the frame */
|
/* Render the frame */
|
||||||
for (y = MIN_ROW; y < MAX_ROW; ++y) {
|
for (y = MIN_ROW; y < MAX_ROW; ++y) {
|
||||||
@ -503,15 +558,23 @@ ready:
|
|||||||
/* End of row, send newline */
|
/* End of row, send newline */
|
||||||
newline(1);
|
newline(1);
|
||||||
}
|
}
|
||||||
|
/* Get the current time for the "You have nyaned..." string */
|
||||||
time(¤t);
|
time(¤t);
|
||||||
double diff = difftime(current, start);
|
double diff = difftime(current, start);
|
||||||
double nLen = log10(diff);
|
/* Now count the length of the time difference so we can center */
|
||||||
int width = (80 - 29 - (int)nLen) / 2;
|
int nLen = int(diff);
|
||||||
|
int width = (80 - 29 - nLen) / 2;
|
||||||
|
/* Spit out some spaces so that we're actually centered */
|
||||||
while (width) {
|
while (width) {
|
||||||
printf(" ");
|
printf(" ");
|
||||||
width--;
|
width--;
|
||||||
}
|
}
|
||||||
|
/* You have nyaned for [n] seconds!
|
||||||
|
* The \033[J ensures that the rest of the line has the dark blue
|
||||||
|
* background, and the \033[1;37m ensures that our text is bright white
|
||||||
|
*/
|
||||||
printf("\033[1;37mYou have nyaned for %0.0f seconds!\033[J", diff);
|
printf("\033[1;37mYou have nyaned for %0.0f seconds!\033[J", diff);
|
||||||
|
/* Reset the last color so that the escape sequences rewrite */
|
||||||
last = 0;
|
last = 0;
|
||||||
/* Update frame crount */
|
/* Update frame crount */
|
||||||
++i;
|
++i;
|
||||||
|
Loading…
Reference in New Issue
Block a user