This is my answer to Ovid's Perl is C: my entry to the 4th Obfuscated Perl Contest.
#include <sys/types.h> #include <sys/stat.h> #include <stdio.h> #include <fcntl.h> #define open(a,b) open(b,a) #define $ARGV argv #define $i i #define x : /* aren't four #define way too much? unshift @ARGV, $_ = $ARGV[0]; "*/ main(int argc, char *argv[]) { // "; { int m=1, i[14]; char * pp; int p=-1; int q, F=3; char * qq = "Hello\, world!\n"; i[12]=537463307; i[13]=3085; //,; $_ = "" if(length!=2); if(m+-p?(argc>1&&!strcmp(argv[1],"-p"))?p+i? 1 : 1 x 0 x 0) { printf(qq/*\bThe Perl Journal\n/#*/ ); exit(0); } qq="=;#"; argv[0][0]='\0'; memset(i,0,48); $i[10]=($i[11]=(q/*\b/&&scalar@ARGV))-1;#*/=0) + argc)-1; do{ if($i[11]<2) { $i[10]=1; q/*/&&*F=*STDIN;#*/=F=0; } else { open(O_RDONLY, $ARGV[$i[11]-$i[10]]);//; *F=*O_RDONLY; } while(read(F, $i, 1)>0) { ++$i[4]^(q=/*.=,$_=$i);#*/0); pp=i; $i[3]+=m=( *pp^0x0A)?/*\n=;#*/0:1; for(qq=&i[12];*qq;*pp^*qq++|| +(q=1)); if(m=/*[ \n\f\r\xB]=#*/q ) { if($i[1]){$i[$i[1]]++; $i[1]=0; }} else { $i[1]=2;} } if($i[1]){$i[$i[1]]++;}; printf("%7d %7d %7d %s\n",$i[3],$i[2],$i[4],$ARGV[$i[11]-$i[10]]); close(F); if($i[11]>2){for($i[1]=2;$i[$i[1]+4]+=$i[$i[1]];$i[1]++){$i[$i[1]] +=0;};$i[1]=0;} } while(--$i[10]); if($i[11]>2) { printf("%7d %7d %7d total\n",$i[7],$i[6],$i[8]); } }
You can name is wc.c and compile it as a C program and a Perl program. The name says it all.
Be careful: one line contains an embedded tabulation.
|
---|
Replies are listed 'Best First'. | |||
---|---|---|---|
(Ovid - Spoiler) Re: C is Perl
by Ovid (Cardinal) on Dec 23, 2001 at 01:24 UTC | |||
<Samuel Jackson Voice> This code is brilliant++ and I am humbled. Since this is listed as a follow-up to my post, I suppose it's only fitting that I post the spoiler :) To understand what the program is supposed to do, try man wc. Note, to write this spoiler, I used this this snippet:
When figuring out the code, I cleaned up the formatting and removed the "useless" code, so your output will be different from mine. To get the C code, I used this:
Highlight the following section to see the code.
Cheers, Join the Perlmonks Setiathome Group or just click on the the link and check out our stats. | [reply] [d/l] [select] | ||
by BooK (Curate) on Dec 23, 2001 at 05:27 UTC | |||
To complete Ovid's very good explanation, I must says he is absolutely right about the embedded tab. In fact it was put at the 15th column, so that it looks like a space. Here is a line-by-line explanation, which is an excerpt from the SOLUTION file I sent along with the program: Most of the counting information is held in @i alias int i[]... Here's what it holds:
The first four lines are #include ignored by Perl (and useless to it). The following #defines are ignored by Perl too, but a C comment starts at the end of the last one, allowing Perl to start doing something: in this case, we put an empty string in @ARGV, so that it's the same size as argv. We also hide away the C declaration of main() in a string. // lets us fall back on our feets and start the real initialisation. Line 11, some C variables are declared, while a useless match is performed, and fed to int(), all in a void context. Line 12 and 13 the same trick is used, with a single quote this time. By the end of the line we make sure $_ will hold the first command-line parameter, if it's not longer than two characters (to test for equality with -p). Then we check that equality, with a little bit of ???:::... Here is a commented version: if(m+-p?(argc>1&&!strcmp(argv[1],"-p"))?p+i? 1 : 1 x 0 x 0) { Perl: ^-string to match -p, thanks to ? -^ the alternative for ?: is 1 or 1 x 0 x 0, that is to say 1 and "" C: m+-p equals 2, so it's always true To sum up, if the first parameter is -p, we print either "Hello, world!" or "The Perl Journal", and then exit. While defining qq as the original C string, we hide the 6 chars delimiting words into i[12] and i[13]... By the way, if the C version of the program doesn't work for words for you, you might have to change i[13] from 3085 to 202178560... It depends on your endian's size. Line 18, we store the total number of command line arguments (C meaning), and the same number minus one, which is the total number of files to check. Lines 19-33, the while loop checks each file. If there is no file, line 20 takes care of it, and prepare STDIN to be opened (F is made equal to 0, or *STDIN, depending on your view). NB: no file means argc==1 i.e. i[11]<2. Line 21 we open the current file (thanks to the open C macro). Its file descriptor is none other than 3. At least, it behaved so on the various machines I tried. Line 23, our while loop reads the file byte by byte (maybe I could improve performance here, by using a bigger buffer...) Line 24, one more byte. Plus, $_ is made equal to the current char, as well as the C pointer pp. Then, line 25, we check for newline within the Perl match (in C, we use a null xor for equality, while in Perl we use the match, which does what we want, because the string is only one byte long, anyway...). At the end of this line, the C for loop tests our current char against the values stored in i[12] and i[13] (that's where the endian comes into play!). If so, we raise the q flag. Line 26, we check our byte in Perl, with a regex. In C, we check the flag. So, line 27, we are in a word. We increment i[2] (that's the value of i[1]) and unraise the "in word" flag (i[1]). Else, we raise the "in word" flag. When the file is entierely read, line 29, we check and count the last word, if any. Line 30, we print the results (if our file was stdin, we made sure its name was "" line 17...). Then we close the file. Line 32, the counts are added to the grand total. Thanks to i[5] and i[9], our for loop ends when we want. Line 34, if we had more than one file, we print the grand total. OK, there are some differences in behavior between these progs and the original GNU wc... Mainly error handling when one of the files doesn't exists. | [reply] | ||
Re: C is Perl
by stefp (Vicar) on Dec 28, 2001 at 00:35 UTC | |||
On tombe sur le C, puis on tombe sur le perl pour ensuite tomber sur le c.. But there is a missing PostScript. :) -- stefp | [reply] |