[go: up one dir, main page]

File: interesting-numbers.c

package info (click to toggle)
trinity 1.8-6
  • links: PTS, VCS
  • area: main
  • in suites: buster
  • size: 4,284 kB
  • sloc: ansic: 33,847; sh: 536; makefile: 234
file content (116 lines) | stat: -rw-r--r-- 3,023 bytes parent folder | download | duplicates (4)
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
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
#include <stdio.h>
#include <stdlib.h>
#include "arch.h"
#include "random.h"
#include "sanitise.h"

static unsigned int plus_minus_two(unsigned int num)
{
	/* Now munge it for off-by-ones. */
	switch (rnd() % 4) {
	case 0:	num -= 2;
		break;
	case 1:	num -= 1;
		break;
	case 2:	num += 1;
		break;
	case 3:	num += 2;
		break;
	}
	return num;
}

static unsigned char get_interesting_8bit_value(void)
{
	switch (rnd() % 5) {
	case 0: return 1;					// one
	case 1: return 0xff;				// max
	case 2: return 1UL << (rnd() & 7);	// 2^n (1 -> 128)
	case 3: return RAND_BYTE();			// 0 -> 0xff
	default: return 0;					// zero
	}
}

static unsigned short get_interesting_16bit_value(void)
{
	switch (rnd() % 4) {
	case 0: return 0x8000 >> (rnd() & 7);		// 2^n (0x100 -> 0x8000)
	case 1: return rnd() & 0xffff;				// 0 -> 0xffff
	case 2: return 0xff00 | RAND_BYTE();		// 0xff00 -> 0xffff
	default: return 0xffff;						// max
	}
}

static unsigned int get_interesting_32bit_value(void)
{
	switch (rnd() % 10) {
	case 0: return 0x80000000 >> (rnd() & 0x1f);	// 2^n (1 -> 0x10000)
	case 1: return rnd();							// 0 -> RAND_MAX (likely 0x7fffffff)
	case 2: return (unsigned int) 0xff << (4 * (rnd() % 7));
	case 3: return 0xffff0000;
	case 4: return 0xffffe000;
	case 5: return 0xffffff00 | RAND_BYTE();
	case 6: return 0xffffffff - page_size;
	case 7: return page_size;
	case 8: return page_size * ((rnd() % (0xffffffff/page_size)) + 1);
	default: return 0xffffffff;						// max
	}
}

#if WORD_BIT != 32
static unsigned long per_arch_interesting_addr(unsigned long low)
{
	int i = 0;

#if defined(__x86_64__)
	i = rnd() % 4;

	switch (i) {
	case 0: return 0x00007fffffffffffUL;			// x86-64 canonical addr end.
	case 1: return 0x0000800000000000UL;			// First x86-64 non-canonical addr
	case 2: return 0xffff800000000000UL | (low << 4);		// x86-64 canonical addr range 2 begin
	case 3: return VDSO_ADDR | (low & 0x0fffff);
	}
#endif

	// FIXME: Add more arch specific addresses here.

	return i | low;
}
#endif	/* WORD_BIT */

unsigned long get_interesting_value(void)
{
	unsigned long low = 0;

	switch (rnd() % 3) {
	case 0:	low = get_interesting_8bit_value();
		break;
	case 1:	low = get_interesting_16bit_value();
		break;
	case 2: low = get_interesting_32bit_value();
		break;
	}

	low = (rnd() & 0xf) ? low : plus_minus_two(low);	// 1 in 16 call plus_minus_two
#if WORD_BIT != 32

	if (RAND_BOOL()) {	// FIXME: This should likely be less aggressive than 50/50
		switch (rnd() % 11) {
		case 0: return 0x0000000100000000UL | low;
		case 1: return 0x7fffffff00000000UL | low;
		case 2: return 0x8000000000000000UL | low;
		case 3: return 0xffffffff00000000UL | low;
		case 4: return 0xffffffffffffff00UL | RAND_BYTE();
		case 5: return 0xffffffffffffffffUL - page_size;
		case 6: return PAGE_OFFSET | (low << 4);
		case 7: return KERNEL_ADDR | (low & 0xffffff);
		case 8: return MODULE_ADDR | (low & 0xffffff);
		case 9: return per_arch_interesting_addr(low);
		case 10: return (low << 32);
		}
	}

#endif	/* WORD_BIT */
	return low;
}