File: wcstombs.c

package info (click to toggle)
picolibc 1.8.11-2
  • links: PTS, VCS
  • area: main
  • in suites: sid
  • size: 50,064 kB
  • sloc: ansic: 404,031; asm: 24,984; sh: 2,585; python: 2,289; perl: 680; pascal: 329; exp: 287; makefile: 222; cpp: 71; xml: 40
file content (101 lines) | stat: -rw-r--r-- 2,753 bytes parent folder | download
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
/*
Copyright (c) 1990 Regents of the University of California.
All rights reserved.
 */
/*
FUNCTION
<<wcstombs>>---minimal wide char string to multibyte string converter

INDEX
        wcstombs

SYNOPSIS
        #include <stdlib.h>
        size_t wcstombs(char *restrict <[s]>, const wchar_t *restrict <[pwc]>, size_t <[n]>);

DESCRIPTION
When __MB_CAPABLE is not defined, this is a minimal ANSI-conforming
implementation of <<wcstombs>>.  In this case,
all wide-characters are expected to represent single bytes and so
are converted simply by casting to char.

When __MB_CAPABLE is defined, this routine uses a state variable to
allow state dependent decoding.  The result is based on the locale
setting which may be restricted to a defined set of locales.

RETURNS
This implementation of <<wcstombs>> returns <<0>> if
<[s]> is <<NULL>> or is the empty string;
it returns <<-1>> if __MB_CAPABLE and one of the
wide-char characters does not represent a valid multi-byte character;
otherwise it returns the minimum of: <<n>> or the
number of bytes that are transferred to <<s>>, not including the
nul terminator.

If the return value is -1, the state of the <<pwc>> string is
indeterminate.  If the input has a length of 0, the output
string will be modified to contain a wchar_t nul terminator if
<<n>> > 0.

PORTABILITY
<<wcstombs>> is required in the ANSI C standard.  However, the precise
effects vary with the locale.

<<wcstombs>> requires no supporting OS subroutines.
*/

#include <stdlib.h>
#include <wchar.h>
#include "local.h"

size_t
wcstombs(char * __restrict s, const wchar_t * __restrict pwcs, size_t n)
{
#ifdef __MB_CAPABLE
    mbstate_t state;
    state.__count = 0;

    char  *ptr = s;
    size_t max = n;
    char   buff[8];
    int    i, bytes, num_to_copy;

    if (s == NULL) {
        size_t num_bytes = 0;
        while (*pwcs != 0) {
            bytes = __WCTOMB(buff, *pwcs++, &state);
            if (bytes == -1)
                return -1;
            num_bytes += bytes;
        }
        return num_bytes;
    } else {
        while (n > 0) {
            bytes = __WCTOMB(buff, *pwcs, &state);
            if (bytes == -1)
                return -1;
            num_to_copy = (n > (size_t)bytes ? bytes : (int)n);
            for (i = 0; i < num_to_copy; ++i)
                *ptr++ = buff[i];

            if (*pwcs == 0x00)
                return ptr - s - (n >= (size_t)bytes);
            ++pwcs;
            n -= num_to_copy;
        }
        return max;
    }
#else  /* not __MB_CAPABLE */
    int count = 0;

    if (n != 0) {
        do {
            if ((*s++ = (char)*pwcs++) == 0)
                break;
            count++;
        } while (--n != 0);
    }

    return count;
#endif /* not __MB_CAPABLE */
}