blob: a219ae9fc077b2168aefad4eae2fcfd4fd4104ce (
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
|
#define _RUNE_NO_MACRO_WRAPPER 1
#include "rune.h"
#include "utf8.h"
#include "internal/common.h"
int
u8prev(rune *ch, const char8_t **p, const char8_t *start)
{
int off;
bool match = true;
const char8_t *s = *p;
ptrdiff_t d = s - start;
if (d <= 0) {
return 0;
} else if (U1(s[-1])) {
*ch = s[-1];
off = 1;
} else if (d > 1 && UC(s[-1]) && U2(s[-2])) {
*ch = ((s[-2] & 0x1F) << 6) | (s[-1] & 0x3F);
off = 2;
} else if (d > 2 && UC(s[-1]) && UC(s[-2]) && U3(s[-3])) {
*ch = ((s[-3] & 0x0F) << 12) | ((s[-2] & 0x3F) << 6) | (s[-1] & 0x3F);
off = 3;
} else if (d > 3 && UC(s[-1]) && UC(s[-2]) && UC(s[-3]) && U4(s[-4])) {
*ch = ((s[-4] & 0x07) << 18) | ((s[-3] & 0x3F) << 12)
| ((s[-2] & 0x3F) << 6) | (s[-1] & 0x3F);
off = 4;
} else
match = false;
if (!(match && u8chkr(*ch))) {
*ch = RUNE_ERROR;
off = 1;
}
*p -= off;
return off;
}
|