aboutsummaryrefslogtreecommitdiffstats
path: root/src/tests/arc4.c
blob: b033cc641133f917ff0903ad05389af6d39f87f9 (plain) (blame)
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
#include <stdint.h>
#include "tests/arc4.h"

/* Swap bytes. */
static inline void
swap_byte (uint8_t *a, uint8_t *b)
{
  uint8_t t = *a;
  *a = *b;
  *b = t;
}

void
arc4_init (struct arc4 *arc4, const void *key_, size_t size)
{
  const uint8_t *key = key_;
  size_t key_idx;
  uint8_t *s;
  int i, j;

  s = arc4->s;
  arc4->i = arc4->j = 0;
  for (i = 0; i < 256; i++)
    s[i] = i;
  for (key_idx = 0, i = j = 0; i < 256; i++)
    {
      j = (j + s[i] + key[key_idx]) & 255;
      swap_byte (s + i, s + j);
      if (++key_idx >= size)
        key_idx = 0;
    }
}

void
arc4_crypt (struct arc4 *arc4, void *buf_, size_t size)
{
  uint8_t *buf = buf_;
  uint8_t *s;
  uint8_t i, j;

  s = arc4->s;
  i = arc4->i;
  j = arc4->j;
  while (size-- > 0)
    {
      i += 1;
      j += s[i];
      swap_byte (s + i, s + j);
      *buf++ ^= s[(s[i] + s[j]) & 255];
    }
  arc4->i = i;
  arc4->j = j;
}