summaryrefslogblamecommitdiff
path: root/src/multibyte/mbsrtowcs.c
blob: e2b434807d930160053f72b53107e445de3f9eea (plain) (tree)
























































































































                                                                                                              
/* 
 * This code was written by Rich Felker in 2010; no copyright is claimed.
 * This code is in the public domain. Attribution is appreciated but
 * unnecessary.
 */

#include <stdlib.h>
#include <inttypes.h>
#include <wchar.h>
#include <errno.h>

#include "internal.h"

size_t mbsrtowcs(wchar_t *ws, const char **src, size_t wn, mbstate_t *st)
{
	unsigned c;
	const unsigned char *s = *src;
	const wchar_t *wsorig = ws;

	if (!st) st = (void *)&c, c = 0;
	else c = *(unsigned *)st;

	if (c) {
		*(unsigned *)st = 0;
		if (!ws) {
			wn = 0;
			goto resume0;
		}
		goto resume;
	}

	if (!ws) for (wn=0;;) {
		if ((unsigned)*s-SA >= SB-SA) {
			while (((unsigned)s&3) && (unsigned)*s-1<0x7f) s++, wn++;
			while (!(( *(uint32_t*)s | *(uint32_t*)s-0x01010101) & 0x80808080)) s+=4, wn+=4;
			while ((unsigned)*s-1<0x7f) s++, wn++;
			if (!*s) return wn;
			if ((unsigned)*s-SA >= SB-SA) goto ilseq2;
		}
		c = bittab[*s++-SA];
		do {
resume0:
			if (OOB(c,*s)) goto ilseq2; s++;
			c <<= 6; if (!(c&(1U<<31))) break;
#ifdef I_FAILED_TO_RTFM_RFC3629
			if ((unsigned)*s++-0x80 >= 0x40) goto ilseq2;
			c <<= 6; if (!(c&(1U<<31))) break;
			if ((unsigned)*s++-0x80 >= 0x40) goto ilseq2;
			c <<= 6; if (!(c&(1U<<31))) break;
#endif
			if ((unsigned)*s++-0x80 >= 0x40) goto ilseq2;
			c <<= 6; if (!(c&(1U<<31))) break;
			if ((unsigned)*s++-0x80 >= 0x40) goto ilseq2;
		} while (0);
		wn++; c = 0;
	}

	while (wn) {
		if ((unsigned)*s-SA >= SB-SA) {
			if (wn >= 7) {
				while (((unsigned)s&3) && (unsigned)*s-1<0x7f) {
					*ws++ = *s++;
					wn--;
				}
				while (wn>=4 && !(( *(uint32_t*)s | *(uint32_t*)s-0x01010101) & 0x80808080)) {
					*ws++ = *s++;
					*ws++ = *s++;
					*ws++ = *s++;
					*ws++ = *s++;
					wn -= 4;
				}
			}
			while (wn && (unsigned)*s-1<0x7f) {
				*ws++ = *s++;
				wn--;
			}
			if (!wn) break;
			if (!*s) {
				*ws = 0;
				*src = 0;
				return ws-wsorig;
			}
			if ((unsigned)*s-SA >= SB-SA) goto ilseq;
		}
		c = bittab[*s++-SA];
		do {
resume:
			if (OOB(c,*s)) goto ilseq;
			c = (c<<6) | *s++-0x80;
			if (!(c&(1U<<31))) break;

#ifdef I_FAILED_TO_RTFM_RFC3629
			if ((unsigned)*s-0x80 >= 0x40) goto ilseq;
			c = (c<<6) | *s++-0x80;
			if (!(c&(1U<<31))) break;

			if ((unsigned)*s-0x80 >= 0x40) goto ilseq;
			c = (c<<6) | *s++-0x80;
			if (!(c&(1U<<31))) break;
#endif

			if ((unsigned)*s-0x80 >= 0x40) goto ilseq;
			c = (c<<6) | *s++-0x80;
			if (!(c&(1U<<31))) break;

			if ((unsigned)*s-0x80 >= 0x40) goto ilseq;
			c = (c<<6) | *s++-0x80;
		} while (0);

		*ws++ = c; wn--; c = 0;
	}
	*src = s;
	return ws-wsorig;
ilseq:
	*src = s;
ilseq2:
	/* enter permanently failing state */
	*(unsigned *)st = FAILSTATE;
	errno = EILSEQ;
	return -1;
}