| /* |
| * strspn, strcspn |
| */ |
| |
| #include <string.h> |
| #include <stddef.h> |
| #include <inttypes.h> |
| #include <limits.h> |
| |
| #ifndef LONG_BIT |
| #define LONG_BIT (CHAR_BIT*sizeof(long)) |
| #endif |
| |
| static void set_bit(unsigned long *bitmap, unsigned int bit) |
| { |
| bitmap[bit / LONG_BIT] |= 1UL << (bit % LONG_BIT); |
| } |
| |
| static int test_bit(unsigned long *bitmap, unsigned int bit) |
| { |
| return (int)(bitmap[bit / LONG_BIT] >> (bit % LONG_BIT)) & 1; |
| } |
| |
| static size_t strxspn(const char *s, const char *map, int parity) |
| { |
| unsigned long matchmap[((1 << CHAR_BIT) + LONG_BIT - 1) / LONG_BIT]; |
| size_t n = 0; |
| |
| /* Create bitmap */ |
| memset(matchmap, 0, sizeof matchmap); |
| while (*map) |
| set_bit(matchmap, (unsigned char)*map++); |
| |
| /* Make sure the null character never matches */ |
| if (parity) |
| set_bit(matchmap, 0); |
| |
| /* Calculate span length */ |
| while (test_bit(matchmap, (unsigned char)*s++) ^ parity) |
| n++; |
| |
| return n; |
| } |
| |
| size_t strspn(const char *s, const char *accept) |
| { |
| return strxspn(s, accept, 0); |
| } |
| |
| size_t strcspn(const char *s, const char *reject) |
| { |
| return strxspn(s, reject, 1); |
| } |
| |
| char *strpbrk(const char *s, const char *accept) |
| { |
| const char *ss = s + strxspn(s, accept, 1); |
| |
| return *ss ? (char *)ss : NULL; |
| } |