stringprep.c revision 1.2 1 1.1 elric /* $NetBSD: stringprep.c,v 1.2 2017/01/28 21:31:50 christos Exp $ */
2 1.1 elric
3 1.1 elric /*
4 1.1 elric * Copyright (c) 2004, 2006, 2008 Kungliga Tekniska Hgskolan
5 1.1 elric * (Royal Institute of Technology, Stockholm, Sweden).
6 1.1 elric * All rights reserved.
7 1.1 elric *
8 1.1 elric * Redistribution and use in source and binary forms, with or without
9 1.1 elric * modification, are permitted provided that the following conditions
10 1.1 elric * are met:
11 1.1 elric *
12 1.1 elric * 1. Redistributions of source code must retain the above copyright
13 1.1 elric * notice, this list of conditions and the following disclaimer.
14 1.1 elric *
15 1.1 elric * 2. Redistributions in binary form must reproduce the above copyright
16 1.1 elric * notice, this list of conditions and the following disclaimer in the
17 1.1 elric * documentation and/or other materials provided with the distribution.
18 1.1 elric *
19 1.1 elric * 3. Neither the name of the Institute nor the names of its contributors
20 1.1 elric * may be used to endorse or promote products derived from this software
21 1.1 elric * without specific prior written permission.
22 1.1 elric *
23 1.1 elric * THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND
24 1.1 elric * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
25 1.1 elric * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
26 1.1 elric * ARE DISCLAIMED. IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE
27 1.1 elric * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
28 1.1 elric * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
29 1.1 elric * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
30 1.1 elric * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
31 1.1 elric * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
32 1.1 elric * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
33 1.1 elric * SUCH DAMAGE.
34 1.1 elric */
35 1.1 elric
36 1.1 elric #ifdef HAVE_CONFIG_H
37 1.1 elric #include <config.h>
38 1.1 elric #endif
39 1.1 elric #include "windlocl.h"
40 1.1 elric #include <stdlib.h>
41 1.1 elric #include <string.h>
42 1.1 elric #include <errno.h>
43 1.1 elric
44 1.1 elric /**
45 1.1 elric * Process a input UCS4 string according a string-prep profile.
46 1.1 elric *
47 1.1 elric * @param in input UCS4 string to process
48 1.1 elric * @param in_len length of the input string
49 1.1 elric * @param out output UCS4 string
50 1.1 elric * @param out_len length of the output string.
51 1.1 elric * @param flags stringprep profile.
52 1.1 elric *
53 1.1 elric * @return returns 0 on success, an wind error code otherwise
54 1.1 elric * @ingroup wind
55 1.1 elric */
56 1.1 elric
57 1.1 elric int
58 1.1 elric wind_stringprep(const uint32_t *in, size_t in_len,
59 1.1 elric uint32_t *out, size_t *out_len,
60 1.1 elric wind_profile_flags flags)
61 1.1 elric {
62 1.1 elric size_t tmp_len = in_len * 3;
63 1.1 elric uint32_t *tmp;
64 1.1 elric int ret;
65 1.1 elric size_t olen;
66 1.1 elric
67 1.1 elric if (in_len == 0) {
68 1.1 elric *out_len = 0;
69 1.1 elric return 0;
70 1.1 elric }
71 1.1 elric
72 1.1 elric tmp = malloc(tmp_len * sizeof(uint32_t));
73 1.1 elric if (tmp == NULL)
74 1.1 elric return ENOMEM;
75 1.1 elric
76 1.1 elric ret = _wind_stringprep_map(in, in_len, tmp, &tmp_len, flags);
77 1.1 elric if (ret) {
78 1.1 elric free(tmp);
79 1.1 elric return ret;
80 1.1 elric }
81 1.1 elric
82 1.1 elric olen = *out_len;
83 1.1 elric ret = _wind_stringprep_normalize(tmp, tmp_len, tmp, &olen);
84 1.1 elric if (ret) {
85 1.1 elric free(tmp);
86 1.1 elric return ret;
87 1.1 elric }
88 1.1 elric ret = _wind_stringprep_prohibited(tmp, olen, flags);
89 1.1 elric if (ret) {
90 1.1 elric free(tmp);
91 1.1 elric return ret;
92 1.1 elric }
93 1.1 elric ret = _wind_stringprep_testbidi(tmp, olen, flags);
94 1.1 elric if (ret) {
95 1.1 elric free(tmp);
96 1.1 elric return ret;
97 1.1 elric }
98 1.1 elric
99 1.1 elric /* Insignificant Character Handling for ldap-prep */
100 1.1 elric if (flags & WIND_PROFILE_LDAP_CASE_EXACT_ATTRIBUTE) {
101 1.1 elric ret = _wind_ldap_case_exact_attribute(tmp, olen, out, out_len);
102 1.1 elric #if 0
103 1.1 elric } else if (flags & WIND_PROFILE_LDAP_CASE_EXACT_ASSERTION) {
104 1.1 elric } else if (flags & WIND_PROFILE_LDAP_NUMERIC) {
105 1.1 elric } else if (flags & WIND_PROFILE_LDAP_TELEPHONE) {
106 1.1 elric #endif
107 1.1 elric } else {
108 1.1 elric memcpy(out, tmp, sizeof(out[0]) * olen);
109 1.1 elric *out_len = olen;
110 1.1 elric }
111 1.1 elric free(tmp);
112 1.1 elric
113 1.1 elric return ret;
114 1.1 elric }
115 1.1 elric
116 1.2 christos static const struct {
117 1.1 elric const char *name;
118 1.1 elric wind_profile_flags flags;
119 1.1 elric } profiles[] = {
120 1.1 elric { "nameprep", WIND_PROFILE_NAME },
121 1.1 elric { "saslprep", WIND_PROFILE_SASL },
122 1.1 elric { "ldapprep", WIND_PROFILE_LDAP }
123 1.1 elric };
124 1.1 elric
125 1.1 elric /**
126 1.1 elric * Try to find the profile given a name.
127 1.1 elric *
128 1.1 elric * @param name name of the profile.
129 1.1 elric * @param flags the resulting profile.
130 1.1 elric *
131 1.1 elric * @return returns 0 on success, an wind error code otherwise
132 1.1 elric * @ingroup wind
133 1.1 elric */
134 1.1 elric
135 1.1 elric int
136 1.1 elric wind_profile(const char *name, wind_profile_flags *flags)
137 1.1 elric {
138 1.1 elric unsigned int i;
139 1.1 elric
140 1.1 elric for (i = 0; i < sizeof(profiles)/sizeof(profiles[0]); i++) {
141 1.1 elric if (strcasecmp(profiles[i].name, name) == 0) {
142 1.1 elric *flags = profiles[i].flags;
143 1.1 elric return 0;
144 1.1 elric }
145 1.1 elric }
146 1.1 elric return WIND_ERR_NO_PROFILE;
147 1.1 elric }
148