]> CyberLeo.Net >> Repos - FreeBSD/releng/10.0.git/blob - tools/regression/kgssapi/gsstest.c
- Copy stable/10 (r259064) to releng/10.0 as part of the
[FreeBSD/releng/10.0.git] / tools / regression / kgssapi / gsstest.c
1 /*-
2  * Copyright (c) 2008 Isilon Inc http://www.isilon.com/
3  * Authors: Doug Rabson <dfr@rabson.org>
4  * Developed with Red Inc: Alfred Perlstein <alfred@freebsd.org>
5  *
6  * Redistribution and use in source and binary forms, with or without
7  * modification, are permitted provided that the following conditions
8  * are met:
9  * 1. Redistributions of source code must retain the above copyright
10  *    notice, this list of conditions and the following disclaimer.
11  * 2. Redistributions in binary form must reproduce the above copyright
12  *    notice, this list of conditions and the following disclaimer in the
13  *    documentation and/or other materials provided with the distribution.
14  *
15  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
16  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
17  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
18  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
19  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
20  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
21  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
22  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
23  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
24  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
25  * SUCH DAMAGE.
26  *
27  * $FreeBSD$
28  */
29
30 #include <sys/types.h>
31 #include <sys/syscall.h>
32 #include <sys/module.h>
33
34 #include <stdio.h>
35 #include <string.h>
36 #include <err.h>
37 #include <unistd.h>
38 #include <stdlib.h>
39
40 #include <krb5.h>
41 #include <gssapi/gssapi.h>
42 #include <gssapi/gssapi_krb5.h>
43
44 struct gsstest_2_args {
45         int step;               /* test step number */
46         gss_buffer_desc input_token; /* token from userland */
47         gss_buffer_desc output_token; /* buffer to receive reply token */
48 };
49 struct gsstest_2_res {
50         OM_uint32 maj_stat;     /* maj_stat from kernel */
51         OM_uint32 min_stat;     /* min_stat from kernel */
52         gss_buffer_desc output_token; /* reply token (using space from gsstest_2_args.output) */
53 };
54
55 static void
56 report_error(gss_OID mech, OM_uint32 maj, OM_uint32 min)
57 {
58         OM_uint32 maj_stat, min_stat;
59         OM_uint32 message_context;
60         gss_buffer_desc buf;
61
62         printf("major_stat=%d, minor_stat=%d\n", maj, min);
63         message_context = 0;
64         do {
65                 maj_stat = gss_display_status(&min_stat, maj,
66                     GSS_C_GSS_CODE, GSS_C_NO_OID, &message_context, &buf);
67                 printf("%.*s\n", (int)buf.length, (char *) buf.value);
68                 gss_release_buffer(&min_stat, &buf);
69         } while (message_context);
70         if (mech) {
71                 message_context = 0;
72                 do {
73                         maj_stat = gss_display_status(&min_stat, min,
74                             GSS_C_MECH_CODE, mech, &message_context, &buf);
75                         printf("%.*s\n", (int)buf.length, (char *) buf.value);
76                         gss_release_buffer(&min_stat, &buf);
77                 } while (message_context);
78         }
79 }
80
81 int
82 main(int argc, char **argv)
83 {
84         struct module_stat stat;
85         int mod;
86         int syscall_num;
87
88         stat.version = sizeof(stat);
89         mod = modfind("gsstest_syscall");
90         if (mod < 0) {
91                 fprintf(stderr, "%s: kernel support not present\n", argv[0]);
92                 exit(1);
93         }
94         modstat(mod, &stat);
95         syscall_num = stat.data.intval;
96
97         switch (atoi(argv[1])) {
98         case 1:
99                 syscall(syscall_num, 1, NULL, NULL);
100                 break;
101
102         case 2: {
103                 struct gsstest_2_args args;
104                 struct gsstest_2_res res;
105                 char hostname[512];
106                 char token_buffer[8192];
107                 OM_uint32 maj_stat, min_stat;
108                 gss_ctx_id_t client_context = GSS_C_NO_CONTEXT;
109                 gss_cred_id_t client_cred;
110                 gss_OID mech_type = GSS_C_NO_OID;
111                 gss_buffer_desc name_buf, message_buf;
112                 gss_name_t name;
113                 int32_t enctypes[] = {
114                         ETYPE_DES_CBC_CRC,
115                         ETYPE_ARCFOUR_HMAC_MD5,
116                         ETYPE_ARCFOUR_HMAC_MD5_56,
117                         ETYPE_AES256_CTS_HMAC_SHA1_96,
118                         ETYPE_AES128_CTS_HMAC_SHA1_96,
119                         ETYPE_DES3_CBC_SHA1,
120                 };
121                 int num_enctypes = sizeof(enctypes) / sizeof(enctypes[0]);
122                 int established;
123                 int i;
124
125                 for (i = 0; i < num_enctypes; i++) {
126                         printf("testing etype %d\n", enctypes[i]);
127                         args.output_token.length = sizeof(token_buffer);
128                         args.output_token.value = token_buffer;
129
130                         gethostname(hostname, sizeof(hostname));
131                         snprintf(token_buffer, sizeof(token_buffer),
132                             "nfs@%s", hostname);
133                         name_buf.length = strlen(token_buffer);
134                         name_buf.value = token_buffer;
135                         maj_stat = gss_import_name(&min_stat, &name_buf,
136                             GSS_C_NT_HOSTBASED_SERVICE, &name);
137                         if (GSS_ERROR(maj_stat)) {
138                                 printf("gss_import_name failed\n");
139                                 report_error(mech_type, maj_stat, min_stat);
140                                 goto out;
141                         }
142
143                         maj_stat = gss_acquire_cred(&min_stat, GSS_C_NO_NAME,
144                             0, GSS_C_NO_OID_SET, GSS_C_INITIATE, &client_cred,
145                             NULL, NULL);
146                         if (GSS_ERROR(maj_stat)) {
147                                 printf("gss_acquire_cred (client) failed\n");
148                                 report_error(mech_type, maj_stat, min_stat);
149                                 goto out;
150                         }
151
152                         maj_stat = gss_krb5_set_allowable_enctypes(&min_stat,
153                             client_cred, 1, &enctypes[i]);
154                         if (GSS_ERROR(maj_stat)) {
155                                 printf("gss_krb5_set_allowable_enctypes failed\n");
156                                 report_error(mech_type, maj_stat, min_stat);
157                                 goto out;
158                         }
159
160                         res.output_token.length = 0;
161                         res.output_token.value = 0;
162                         established = 0;
163                         while (!established) {
164                                 maj_stat = gss_init_sec_context(&min_stat,
165                                     client_cred,
166                                     &client_context,
167                                     name,
168                                     GSS_C_NO_OID,
169                                     (GSS_C_MUTUAL_FLAG
170                                         |GSS_C_CONF_FLAG
171                                         |GSS_C_INTEG_FLAG
172                                         |GSS_C_SEQUENCE_FLAG
173                                         |GSS_C_REPLAY_FLAG),
174                                     0,
175                                     GSS_C_NO_CHANNEL_BINDINGS,
176                                     &res.output_token,
177                                     &mech_type,
178                                     &args.input_token,
179                                     NULL,
180                                     NULL);
181                                 if (GSS_ERROR(maj_stat)) {
182                                         printf("gss_init_sec_context failed\n");
183                                         report_error(mech_type, maj_stat, min_stat);
184                                         goto out;
185                                 }
186                                 if (args.input_token.length) {
187                                         args.step = 1;
188                                         syscall(syscall_num, 2, &args, &res);
189                                         gss_release_buffer(&min_stat,
190                                             &args.input_token);
191                                         if (res.maj_stat != GSS_S_COMPLETE
192                                             && res.maj_stat != GSS_S_CONTINUE_NEEDED) {
193                                                 printf("gss_accept_sec_context (kernel) failed\n");
194                                                 report_error(mech_type, res.maj_stat,
195                                                     res.min_stat);
196                                                 goto out;
197                                         }
198                                 }
199                                 if (maj_stat == GSS_S_COMPLETE)
200                                         established = 1;
201                         }
202
203                         message_buf.value = "Hello world";
204                         message_buf.length = strlen((char *) message_buf.value);
205
206                         maj_stat = gss_get_mic(&min_stat, client_context,
207                             GSS_C_QOP_DEFAULT, &message_buf, &args.input_token);
208                         if (GSS_ERROR(maj_stat)) {
209                                 printf("gss_get_mic failed\n");
210                                 report_error(mech_type, maj_stat, min_stat);
211                                 goto out;
212                         }
213                 
214                         args.step = 2;
215                         syscall(syscall_num, 2, &args, &res);
216                         gss_release_buffer(&min_stat, &args.input_token);
217                         if (GSS_ERROR(res.maj_stat)) {
218                                 printf("kernel gss_verify_mic failed\n");
219                                 report_error(mech_type, res.maj_stat, res.min_stat);
220                                 goto out;
221                         }
222
223                         maj_stat = gss_verify_mic(&min_stat, client_context,
224                             &message_buf, &res.output_token, NULL);
225                         if (GSS_ERROR(maj_stat)) {
226                                 printf("gss_verify_mic failed\n");
227                                 report_error(mech_type, maj_stat, min_stat);
228                                 goto out;
229                         }
230
231                         maj_stat = gss_wrap(&min_stat, client_context,
232                             TRUE, GSS_C_QOP_DEFAULT, &message_buf, NULL,
233                             &args.input_token);
234                         if (GSS_ERROR(maj_stat)) {
235                                 printf("gss_wrap failed\n");
236                                 report_error(mech_type, maj_stat, min_stat);
237                                 goto out;
238                         }
239
240                         args.step = 3;
241                         syscall(syscall_num, 2, &args, &res);
242                         gss_release_buffer(&min_stat, &args.input_token);
243                         if (GSS_ERROR(res.maj_stat)) {
244                                 printf("kernel gss_unwrap failed\n");
245                                 report_error(mech_type, res.maj_stat, res.min_stat);
246                                 goto out;
247                         }
248
249                         maj_stat = gss_unwrap(&min_stat, client_context,
250                             &res.output_token, &message_buf, NULL, NULL);
251                         if (GSS_ERROR(maj_stat)) {
252                                 printf("gss_unwrap failed\n");
253                                 report_error(mech_type, maj_stat, min_stat);
254                                 goto out;
255                         }
256                         gss_release_buffer(&min_stat, &message_buf);
257
258                         maj_stat = gss_wrap(&min_stat, client_context,
259                             FALSE, GSS_C_QOP_DEFAULT, &message_buf, NULL,
260                             &args.input_token);
261                         if (GSS_ERROR(maj_stat)) {
262                                 printf("gss_wrap failed\n");
263                                 report_error(mech_type, maj_stat, min_stat);
264                                 goto out;
265                         }
266
267                         args.step = 4;
268                         syscall(syscall_num, 2, &args, &res);
269                         gss_release_buffer(&min_stat, &args.input_token);
270                         if (GSS_ERROR(res.maj_stat)) {
271                                 printf("kernel gss_unwrap failed\n");
272                                 report_error(mech_type, res.maj_stat, res.min_stat);
273                                 goto out;
274                         }
275
276                         maj_stat = gss_unwrap(&min_stat, client_context,
277                             &res.output_token, &message_buf, NULL, NULL);
278                         if (GSS_ERROR(maj_stat)) {
279                                 printf("gss_unwrap failed\n");
280                                 report_error(mech_type, maj_stat, min_stat);
281                                 goto out;
282                         }
283                         gss_release_buffer(&min_stat, &message_buf);
284
285                         args.step = 5;
286                         syscall(syscall_num, 2, &args, &res);
287
288                         gss_release_name(&min_stat, &name);
289                         gss_release_cred(&min_stat, &client_cred);
290                         gss_delete_sec_context(&min_stat, &client_context,
291                             GSS_C_NO_BUFFER);
292                 }
293
294                 break;
295         }
296         case 3:
297                 syscall(syscall_num, 3, NULL, NULL);
298                 break;
299         case 4:
300                 syscall(syscall_num, 4, NULL, NULL);
301                 break;
302         }
303         return (0);
304
305 out:
306         return (1);
307 }