summaryrefslogtreecommitdiff
path: root/binsearch.c
blob: a4195ba09c5da02033e277f946fda58cc0803ff9 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <stdbool.h>
#include <err.h>

static void
usage(const char *argv0)
{
   fprintf(stderr"usage: %s needle [window-size] < haystack\n", argv0);
   exit(EXIT_FAILURE);
}

static const char*
search(const char *haystack, const char *needle, const size_t window_size)
{
   for (const char *s = haystack; s < haystack + window_size; ++s) {
      if (memcmp(s, needle, window_size))
         continue;
      return s;
   }
   return NULL;
}

static void
search_and_exit_if_match(const char *haystack, const char *needle, const size_t window_size, const size_t offset)
{
   const char *match;
   if ((match = search(haystack, needle, window_size))) {
      printf("%zu\n", offset + match - haystack);
      free((void*)needle);
      free((void*)haystack);
      exit(EXIT_SUCCESS);
   }
}

int
main(int argc, const char *argv[])
{
   // default incase failure, or cant get size of file
   size_t window_size = 4096 * 1024;
   bool has_window_size = false;

   if (argc < 2)
      usage(argv[0]);
   else if (argc > 2) {
      window_size = strtoull(argv[2], NULL10);
      has_window_size = true;
   }

   FILE *f;
   if (!(f = fopen(argv[1], "rb")))
      err(EXIT_FAILURE"fopen(%s)", argv[1]);

   if (!has_window_size) {
      fseek(f, 0SEEK_END);
      const long tell = ftell(f);
      window_size = (tell > 0 ? (size_t)tell : window_size);
      fseek(f, 0SEEK_SET);
   }

   char *needle;
   if (!(needle = malloc(window_size)))
      err(EXIT_FAILURE"malloc");

   window_size = fread(needle, 1, window_size, f);
   fclose(f);

   char *haystack;
   if (!(haystack = calloc(2, window_size)))
         err(EXIT_FAILURE"calloc");

   size_t rd = 0, offset = 0;
   while ((rd = fread(haystack + window_size * !!offset, 1 + !offset, window_size, stdin))) {
      search_and_exit_if_match(haystack, needle, (rd >= window_size ? window_size : 0), offset);
      offset += window_size;
      memmove(haystack, haystack + window_size, window_size);
   }

   search_and_exit_if_match(haystack, needle, (rd >= window_size ? window_size : 0), offset);
   free(needle);
   free(haystack);
   return EXIT_FAILURE;
}