summaryrefslogtreecommitdiff
path: root/src/main.c
blob: 479c48bd85809332e5fb2d7023b60cc71e15fde9 (plain)
    1 #include <stdio.h>
    2 #include <stdlib.h>
    3 #include <time.h>
    4 
    5 //
    6 // Compare the whole thing by passing 0 as end
    7 //
    8 // @param str String to check for palindrome
    9 // @param end For checking a substring shorter than the end of str
   10 //
   11 // @return int Is a palindrome (1) or not (0)
   12 //
   13 char is_palindrome(char* str, long len) {
   14   //long len;
   15   long i = 0;
   16 
   17   // This handles comparisons of odd and even length strings
   18   while(len - i > 0) {
   19     if(str[i] != str[len])
   20       return 0;
   21 
   22     ++i;
   23     --len;
   24   }
   25   return 1;
   26 }
   27 
   28 
   29 //
   30 // Finds the largest palindrome in a data set
   31 //
   32 // @param data Data chunk to search for the largest palindrome
   33 //
   34 void find_largest_palindromes(char* data) {
   35   long cursor = 0;
   36   int start = 0;
   37   int end = 0;
   38 
   39   int largest = 0;
   40   int largest_start = 0;
   41   int largest_end = 0;
   42 
   43   // Writing this here for sanity purposes
   44   // 0 1 2 3 4 5 6
   45   // r a c e c a r
   46 
   47   // 0 1 2 3 4 5
   48   // a b c c b a
   49 
   50   while(data[cursor] != '\0') {
   51     // Need to align with the center
   52     // Advance end one to see if this is an even numbered palindrome
   53     if(data[end + 1] != '\0') {
   54       ++end;
   55     }
   56     // If it's not an even numbered, let's see if it's odd numbered
   57     if(start > 0 && data[start] != data[end]) {
   58       --start;
   59     }
   60     // Looks like it's not either, reset and continue
   61     if(data[start] != data[end]) {
   62       ++cursor;
   63       start = cursor;
   64       end = cursor;
   65       continue;
   66     }
   67 
   68     // It seems we have found a palindrome, let's see how big...
   69     while(data[start] == data[end]) {
   70       if(data[end + 1] != '\0') {
   71         ++end;
   72       }
   73       if(start > 0) {
   74         --start;
   75       }
   76 
   77     }
   78     if(end - start - 1 > largest) {
   79       largest = end - start - 1;
   80       largest_start = start + 1;
   81       largest_end = end - 1;
   82     }
   83 
   84     ++cursor;
   85     start = cursor;
   86     end = cursor;
   87   }
   88   // Move start one forwards since moving it backwards resulted in a non-
   89   if( largest > 1) {
   90     printf("The largest palindrome is %d long\n", largest);
   91     printf("It starts at %d, and ends at %d\n", largest_start, largest_end);
   92     printf("%.*s\n", largest, &data[largest_start]);
   93   }
   94 }
   95 
   96 
   97 //
   98 // Function for testing one-off strings
   99 // Useful for debugging
  100 //
  101 // @param str String to test for palindromeism
  102 //
  103 void palstr(char* str) {
  104    if(is_palindrome(str, 0) == 1) {
  105      printf("%s \e[1mis\e[0m a palindrome!\n", str);
  106    } else {
  107      printf("%s is \e[1mnot\e[0m a palindrome.\n", str);
  108    }
  109 }
  110 
  111 
  112 //
  113 // Ye olde' main function
  114 //
  115 int main(int argc, char *argv[]) {
  116   if(argc < 2) {
  117     printf("Please specify a string to check for palindrome-ism.\n");
  118     return 1;
  119   }
  120 
  121   struct timespec start, stop;
  122   FILE *f = fopen(argv[1], "r");
  123 
  124   // If the file can't be opened, try using the passed value as a palindrome
  125   // test
  126   if(!f) {
  127     palstr(argv[1]);
  128     return 0;
  129   }
  130   
  131   printf("Scanning file %s\n", argv[1]);
  132 
  133   // Get file length
  134   unsigned int flen = 0;
  135   while(fgetc(f) != EOF) {
  136     flen++;
  137   }
  138 
  139   rewind(f);
  140 
  141   printf("Allocating memory\n");
  142   // Allocate array for file contents
  143   char* data = malloc(flen + 1);
  144   unsigned int i = 0;
  145   char c;
  146   printf("Reading data to memory\n");
  147   while((c = fgetc(f)) != EOF) {
  148     data[i] = c;
  149     i++;
  150   }
  151   
  152   fclose(f);
  153 
  154   printf("Searching for palindromes\n");
  155 
  156   
  157   clock_gettime(CLOCK_MONOTONIC_RAW, &start);
  158   find_largest_palindromes(data);
  159   clock_gettime(CLOCK_MONOTONIC_RAW, &stop);
  160 
  161   free(data);
  162 
  163   // Calculate runtime
  164   long int secs = stop.tv_sec - start.tv_sec;
  165   long int nano = stop.tv_nsec - start.tv_nsec;
  166   // Print runtime message
  167   printf("Palindrome search took %ld.%ld seconds\n", secs, nano / 10000);
  168 
  169   return 0;
  170 }

Generated by cgit