1 /* 2 * CDDL HEADER START 3 * 4 * The contents of this file are subject to the terms of the 5 * Common Development and Distribution License, Version 1.0 only 6 * (the "License"). You may not use this file except in compliance 7 * with the License. 8 * 9 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 10 * or http://www.opensolaris.org/os/licensing. 11 * See the License for the specific language governing permissions 12 * and limitations under the License. 13 * 14 * When distributing Covered Code, include this CDDL HEADER in each 15 * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 16 * If applicable, add the following below this CDDL HEADER, with the 17 * fields enclosed by brackets "[]" replaced with your own identifying 18 * information: Portions Copyright [yyyy] [name of copyright owner] 19 * 20 * CDDL HEADER END 21 */ 22 /* 23 * Copyright 2017 Gary Mills 24 * Copyright 2003 Sun Microsystems, Inc. All rights reserved. 25 * Use is subject to license terms. 26 */ 27 28 /* 29 * Test client for gssd. This program is not shipped on the binary 30 * release. 31 */ 32 33 #include <stdio.h> 34 #include <strings.h> 35 #include <ctype.h> 36 #include <stdlib.h> 37 #include <gssapi/gssapi.h> 38 #include <gssapi/gssapi_ext.h> 39 #include "gssd.h" 40 #include <rpc/rpc.h> 41 42 #define _KERNEL 43 #include <gssapi/gssapi.h> 44 #undef _KERNEL 45 46 int gss_major_code; 47 int gss_minor_code; 48 49 int init_sec_context_phase = 0; 50 int accept_sec_context_phase = 0; 51 52 gss_ctx_id_t initiator_context_handle; 53 gss_ctx_id_t acceptor_context_handle; 54 gss_cred_id_t acceptor_credentials; 55 gss_buffer_desc init_token_buffer; 56 gss_buffer_desc accept_token_buffer; 57 gss_buffer_desc delete_token_buffer; 58 gss_buffer_desc message_buffer; 59 gss_buffer_desc msg_token; 60 61 #define LOOP_COUNTER 100 62 #define GSS_KRB5_MECH_OID "1.2.840.113554.1.2.2" 63 #define GSS_DUMMY_MECH_OID "1.3.6.1.4.1.42.2.26.1.2" 64 #ifdef _KERNEL 65 #define OCTAL_MACRO "%03o." 66 #define MALLOC(n) kmem_alloc((n), KM_SLEEP) 67 #define CALLOC(n, s) kmem_zalloc((n)*(s), KM_SLEEP) 68 #define FREE(x, n) kmem_free((x), (n)) 69 #define memcpy(dst, src, n) bcopy((src), (dst), (n)) 70 #define fprintf(s, m) printf(m) 71 #define isspace(s) ((s) == ' ' || (s) == '\t' || (s) == '\n' || \ 72 (s) == '\r' || (s) == '\v' || (s) == '\f') 73 74 static char *strdup(const char *s) 75 { 76 int len = strlen(s); 77 char *new = MALLOC(len+1); 78 strcpy(new, s); 79 return (new); 80 } 81 82 #else /* !_KERNEL */ 83 #define OCTAL_MACRO "%03.3o." 84 #define MALLOC(n) malloc(n) 85 #define CALLOC(n, s) calloc((n), (s)) 86 #define FREE(x, n) free(x) 87 #endif /* _KERNEL */ 88 89 static gss_OID gss_str2oid(char *); 90 static char * gss_oid2str(gss_OID); 91 static void instructs(); 92 static void usage(); 93 static int parse_input_line(char *, int *, char ***); 94 extern uid_t getuid(); 95 96 static void _gss_init_sec_context(int, char **); 97 static void _gss_acquire_cred(int, char **); 98 static void _gss_add_cred(int, char **); 99 static void _gss_sign(int, char **); 100 static void _gss_release_cred(int, char **); 101 static void _gss_accept_sec_context(int, char **); 102 static void _gss_process_context_token(int, char **); 103 static void _gss_delete_sec_context(int, char **); 104 static void _gss_context_time(int, char **); 105 static void _gss_verify(int, char **); 106 static void _gss_seal(int, char **); 107 static void _gss_unseal(int, char **); 108 static void _gss_display_status(int, char **); 109 static void _gss_indicate_mechs(int, char **); 110 static void _gss_inquire_cred(int, char **); 111 static void _gssd_expname_to_unix_cred(int, char **); 112 static void _gssd_name_to_unix_cred(int, char **); 113 static void _gssd_get_group_info(int, char **); 114 115 static int do_gssdtest(char *buf); 116 117 118 #ifndef _KERNEL 119 static int read_line(char *buf, int size) 120 { 121 int len; 122 123 /* read the next line. If cntl-d, return with zero char count */ 124 printf(gettext("\n> ")); 125 126 if (fgets(buf, size, stdin) == NULL) 127 return (0); 128 129 len = strlen(buf); 130 buf[--len] = '\0'; 131 return (len); 132 } 133 134 int 135 main() 136 { 137 char buf[512]; 138 int len, ret; 139 140 /* Print out usage and instructions to start off the session */ 141 142 instructs(); 143 usage(); 144 145 /* 146 * Loop, repeatedly calling parse_input_line() to get the 147 * next line and parse it into argc and argv. Act on the 148 * arguements found on the line. 149 */ 150 151 do { 152 len = read_line(buf, 512); 153 if (len) 154 ret = do_gssdtest(buf); 155 } while (len && !ret); 156 157 return (0); 158 } 159 #endif /* !_KERNEL */ 160 161 static int 162 do_gssdtest(char *buf) 163 { 164 int argc, seal_argc; 165 int i; 166 char **argv, **argv_array; 167 168 char *cmd; 169 char *seal_ini_array [] = { "initiator", " Hello"}; 170 char *seal_acc_array [] = { "acceptor", " Hello"}; 171 char *unseal_acc_array [] = {"acceptor"}; 172 char *unseal_ini_array [] = {"initiator"}; 173 char *delet_acc_array [] = {"acceptor"}; 174 char *delet_ini_array [] = {"initiator"}; 175 176 argv = 0; 177 178 if (parse_input_line(buf, &argc, &argv) == 0) { 179 printf(gettext("\n")); 180 return (1); 181 } 182 183 if (argc == 0) { 184 usage(); 185 /*LINTED*/ 186 FREE(argv_array, (argc+1)*sizeof (char *)); 187 return (0); 188 } 189 190 /* 191 * remember argv_array address, which is memory calloc'd by 192 * parse_input_line, so it can be free'd at the end of the loop. 193 */ 194 195 argv_array = argv; 196 197 cmd = argv[0]; 198 199 argc--; 200 argv++; 201 202 if (strcmp(cmd, "gss_loop") == 0 || 203 strcmp(cmd, "loop") == 0) { 204 205 if (argc < 1) { 206 usage(); 207 FREE(argv_array, (argc+2) * sizeof (char *)); 208 return (0); 209 } 210 for (i = 0; i < LOOP_COUNTER; i++) { 211 printf(gettext("Loop Count is %d \n"), i); 212 /* 213 * if (i > 53) 214 * printf ("Loop counter is greater than 55\n"); 215 */ 216 _gss_acquire_cred(argc, argv); 217 _gss_init_sec_context(argc, argv); 218 _gss_accept_sec_context(0, argv); 219 _gss_init_sec_context(argc, argv); 220 221 seal_argc = 2; 222 _gss_seal(seal_argc, seal_ini_array); 223 seal_argc = 1; 224 _gss_unseal(seal_argc, unseal_acc_array); 225 seal_argc = 2; 226 _gss_seal(seal_argc, seal_acc_array); 227 seal_argc = 1; 228 _gss_unseal(seal_argc, unseal_ini_array); 229 seal_argc = 2; 230 _gss_sign(seal_argc, seal_ini_array); 231 seal_argc = 1; 232 _gss_verify(seal_argc, unseal_acc_array); 233 seal_argc = 2; 234 _gss_sign(seal_argc, seal_acc_array); 235 seal_argc = 1; 236 _gss_verify(seal_argc, unseal_ini_array); 237 _gss_delete_sec_context(argc, delet_acc_array); 238 _gss_delete_sec_context(argc, delet_ini_array); 239 } 240 } 241 if (strcmp(cmd, "gss_all") == 0 || 242 strcmp(cmd, "all") == 0) { 243 _gss_acquire_cred(argc, argv); 244 _gss_init_sec_context(argc, argv); 245 _gss_accept_sec_context(0, argv); 246 _gss_init_sec_context(argc, argv); 247 248 seal_argc = 2; 249 _gss_seal(seal_argc, seal_acc_array); 250 seal_argc = 1; 251 _gss_unseal(seal_argc, unseal_ini_array); 252 seal_argc = 2; 253 _gss_seal(seal_argc, seal_ini_array); 254 seal_argc = 1; 255 _gss_unseal(seal_argc, unseal_acc_array); 256 seal_argc = 2; 257 _gss_sign(seal_argc, seal_ini_array); 258 seal_argc = 1; 259 _gss_verify(seal_argc, unseal_acc_array); 260 seal_argc = 2; 261 _gss_sign(seal_argc, seal_acc_array); 262 seal_argc = 1; 263 _gss_verify(seal_argc, unseal_ini_array); 264 265 } 266 if (strcmp(cmd, "gss_acquire_cred") == 0 || 267 strcmp(cmd, "acquire") == 0) { 268 _gss_acquire_cred(argc, argv); 269 if (argc == 1) 270 _gss_add_cred(argc, argv); 271 } 272 273 else if (strcmp(cmd, "gss_release_cred") == 0 || 274 strcmp(cmd, "release") == 0) 275 _gss_release_cred(argc, argv); 276 else if (strcmp(cmd, "gss_init_sec_context") == 0 || 277 strcmp(cmd, "init") == 0) 278 _gss_init_sec_context(argc, argv); 279 else if (strcmp(cmd, "gss_accept_sec_context") == 0 || 280 strcmp(cmd, "accept") == 0) 281 _gss_accept_sec_context(argc, argv); 282 else if (strcmp(cmd, "gss_process_context_token") == 0 || 283 strcmp(cmd, "process") == 0) 284 _gss_process_context_token(argc, argv); 285 else if (strcmp(cmd, "gss_delete_sec_context") == 0 || 286 strcmp(cmd, "delete") == 0) 287 _gss_delete_sec_context(argc, argv); 288 else if (strcmp(cmd, "gss_context_time") == 0 || 289 strcmp(cmd, "time") == 0) 290 _gss_context_time(argc, argv); 291 else if (strcmp(cmd, "gss_sign") == 0 || 292 strcmp(cmd, "sign") == 0) 293 _gss_sign(argc, argv); 294 else if (strcmp(cmd, "gss_verify") == 0 || 295 strcmp(cmd, "verify") == 0) 296 _gss_verify(argc, argv); 297 else if (strcmp(cmd, "gss_seal") == 0 || 298 strcmp(cmd, "seal") == 0) 299 _gss_seal(argc, argv); 300 else if (strcmp(cmd, "gss_unseal") == 0 || 301 strcmp(cmd, "unseal") == 0) 302 _gss_unseal(argc, argv); 303 else if (strcmp(cmd, "gss_display_status") == 0|| 304 strcmp(cmd, "status") == 0) 305 _gss_display_status(argc, argv); 306 else if (strcmp(cmd, "gss_indicate_mechs") == 0 || 307 strcmp(cmd, "indicate") == 0) 308 _gss_indicate_mechs(argc, argv); 309 else if (strcmp(cmd, "gss_inquire_cred") == 0 || 310 strcmp(cmd, "inquire") == 0) 311 _gss_inquire_cred(argc, argv); 312 else if (strcmp(cmd, "expname2unixcred") == 0 || 313 strcmp(cmd, "gsscred_expname_to_unix_cred") == 0) 314 _gssd_expname_to_unix_cred(argc, argv); 315 else if (strcmp(cmd, "name2unixcred") == 0 || 316 strcmp(cmd, "gsscred_name_to_unix_cred") == 0) 317 _gssd_name_to_unix_cred(argc, argv); 318 else if (strcmp(cmd, "grpinfo") == 0 || 319 strcmp(cmd, "gss_get_group_info") == 0) 320 _gssd_get_group_info(argc, argv); 321 else if (strcmp(cmd, "exit") == 0) { 322 printf(gettext("\n")); 323 FREE(argv_array, (argc+2) * sizeof (char *)); 324 return (1); 325 } else 326 usage(); 327 328 /* free argv array */ 329 330 FREE(argv_array, (argc+2) * sizeof (char *)); 331 return (0); 332 } 333 334 static void 335 _gss_acquire_cred(argc, argv) 336 int argc; 337 char **argv; 338 { 339 340 OM_UINT32 status, minor_status; 341 gss_buffer_desc name; 342 gss_name_t desired_name = (gss_name_t) 0; 343 OM_uint32 time_req; 344 gss_OID_set_desc desired_mechs_desc; 345 gss_OID_set desired_mechs = &desired_mechs_desc; 346 int cred_usage; 347 gss_OID_set actual_mechs = GSS_C_NULL_OID_SET; 348 gss_OID_set inquire_mechs = GSS_C_NULL_OID_SET; 349 OM_UINT32 time_rec; 350 char * string; 351 char * inq_string; 352 uid_t uid; 353 gss_OID mech_type; 354 355 /* 356 * First set up the command line independent input arguments. 357 */ 358 359 time_req = (OM_uint32) 0; 360 cred_usage = GSS_C_ACCEPT; 361 uid = getuid(); 362 363 /* Parse the command line for the variable input arguments */ 364 365 if (argc == 0) { 366 usage(); 367 return; 368 } 369 370 /* 371 * Get the name of the principal. 372 */ 373 374 name.length = strlen(argv[0])+1; 375 name.value = argv[0]; 376 377 /* 378 * Now convert the string given by the first argument into internal 379 * form suitable for input to gss_acquire_cred() 380 */ 381 382 if ((status = gss_import_name(&minor_status, &name, 383 (gss_OID)GSS_C_NT_HOSTBASED_SERVICE, &desired_name)) 384 != GSS_S_COMPLETE) { 385 printf(gettext( 386 "could not parse desired name: err (octal) %o (%s)\n"), 387 status, gettext("gss_acquire_cred error")); 388 return; 389 } 390 391 argc--; 392 argv++; 393 394 /* 395 * The next argument is an OID in dotted decimal form. 396 */ 397 398 if (argc == 0) { 399 printf(gettext("Assuming Kerberos V5 as the mechanism\n")); 400 printf(gettext( 401 "The mech OID 1.2.840.113554.1.2.2 will be used\n")); 402 mech_type = gss_str2oid((char *)GSS_KRB5_MECH_OID); 403 } else 404 mech_type = gss_str2oid(argv[0]); 405 406 if (mech_type == 0 || mech_type->length == 0) { 407 printf(gettext("improperly formated mechanism OID\n")); 408 return; 409 } 410 411 /* 412 * set up desired_mechs so it points to mech_type. 413 */ 414 415 desired_mechs = (gss_OID_set) MALLOC(sizeof (gss_OID_desc)); 416 417 desired_mechs->count = 1; 418 desired_mechs->elements = mech_type; 419 420 status = kgss_acquire_cred( 421 &minor_status, 422 desired_name, 423 time_req, 424 desired_mechs, 425 cred_usage, 426 &acceptor_credentials, 427 &actual_mechs, 428 &time_rec, 429 uid); 430 431 /* store major and minor status for gss_display_status() call */ 432 433 gss_major_code = status; 434 gss_minor_code = minor_status; 435 436 if (status == GSS_S_COMPLETE) { 437 /* process returned values */ 438 439 printf(gettext("\nacquire succeeded\n\n")); 440 441 /* 442 * print out the actual mechs returned NB: Since only one 443 * mechanism is specified in desired_mechs, only one 444 * can be returned in actual_mechs. Consequently, 445 * actual_mechs->elements points to an array of only one 446 * element. 447 */ 448 449 if ((string = gss_oid2str(actual_mechs->elements)) == 0) { 450 printf(gettext("actual mechs == NULL\n\n")); 451 } else { 452 printf(gettext("actual mechs = %s\n\n"), string); 453 FREE(string, (actual_mechs->elements->length+1)*4+1); 454 } 455 456 if (cred_usage == GSS_C_BOTH) 457 printf(gettext("GSS_C_BOTH\n\n")); 458 459 if (cred_usage == GSS_C_INITIATE) 460 printf(gettext("GSS_C_INITIATE\n\n")); 461 462 if (cred_usage == GSS_C_ACCEPT) 463 printf(gettext("GSS_C_ACCEPT\n\n")); 464 status = kgss_inquire_cred( 465 &minor_status, 466 acceptor_credentials, 467 NULL, 468 &time_req, 469 &cred_usage, 470 &inquire_mechs, 471 uid); 472 473 if (status != GSS_S_COMPLETE) 474 printf(gettext("server ret err (octal) %o (%s)\n"), 475 status, gettext("gss_inquire_cred error")); 476 else { 477 if ((inq_string = 478 gss_oid2str(inquire_mechs->elements)) == 0) { 479 printf(gettext 480 ("mechs from inquire == NULL\n\n")); 481 } else { 482 printf(gettext 483 ("mechs from inquiry = %s\n\n"), 484 inq_string); 485 FREE(inq_string, 486 (inquire_mechs->elements->length+1)*4+1); 487 } 488 printf(gettext("inquire_cred successful \n\n")); 489 } 490 491 } else { 492 printf(gettext("server ret err (octal) %o (%s)\n"), 493 status, gettext("gss_acquire_cred error")); 494 } 495 496 /* free allocated memory */ 497 498 /* actual mechs is allocated by clnt_stubs. Release it here */ 499 if (actual_mechs != GSS_C_NULL_OID_SET) 500 gss_release_oid_set_and_oids(&minor_status, &actual_mechs); 501 if (inquire_mechs != GSS_C_NULL_OID_SET) 502 gss_release_oid_set_and_oids(&minor_status, &inquire_mechs); 503 504 gss_release_name(&minor_status, &desired_name); 505 506 /* mech_type and desired_mechs are allocated above. Release it here */ 507 508 FREE(mech_type->elements, mech_type->length); 509 FREE(mech_type, sizeof (gss_OID_desc)); 510 FREE(desired_mechs, sizeof (gss_OID_desc)); 511 } 512 513 static void 514 _gss_add_cred(argc, argv) 515 int argc; 516 char **argv; 517 { 518 519 OM_UINT32 status, minor_status; 520 gss_buffer_desc name; 521 gss_name_t desired_name = (gss_name_t) 0; 522 OM_uint32 time_req; 523 OM_uint32 initiator_time_req; 524 OM_uint32 acceptor_time_req; 525 int cred_usage; 526 gss_OID_set actual_mechs = GSS_C_NULL_OID_SET; 527 gss_OID_set inquire_mechs = GSS_C_NULL_OID_SET; 528 char * string; 529 uid_t uid; 530 gss_OID mech_type; 531 int i; 532 533 /* 534 * First set up the command line independent input arguments. 535 */ 536 537 initiator_time_req = (OM_uint32) 0; 538 acceptor_time_req = (OM_uint32) 0; 539 cred_usage = GSS_C_ACCEPT; 540 uid = getuid(); 541 542 /* Parse the command line for the variable input arguments */ 543 544 if (argc == 0) { 545 usage(); 546 return; 547 } 548 549 /* 550 * Get the name of the principal. 551 */ 552 553 name.length = strlen(argv[0])+1; 554 name.value = argv[0]; 555 556 /* 557 * Now convert the string given by the first argument into internal 558 * form suitable for input to gss_acquire_cred() 559 */ 560 561 if ((status = gss_import_name(&minor_status, &name, 562 (gss_OID)GSS_C_NT_HOSTBASED_SERVICE, &desired_name)) 563 != GSS_S_COMPLETE) { 564 printf(gettext( 565 "could not parse desired name: err (octal) %o (%s)\n"), 566 status, gettext("gss_acquire_cred error")); 567 return; 568 } 569 570 argc--; 571 argv++; 572 573 /* 574 * The next argument is an OID in dotted decimal form. 575 */ 576 577 if (argc == 0) { 578 printf(gettext("Assuming dummy as the mechanism\n")); 579 printf(gettext( 580 "The mech OID 1.3.6.1.4.1.42.2.26.1.2 will be used\n")); 581 mech_type = gss_str2oid((char *)GSS_DUMMY_MECH_OID); 582 } else 583 mech_type = gss_str2oid(argv[0]); 584 585 if (mech_type == 0 || mech_type->length == 0) { 586 printf(gettext("improperly formated mechanism OID\n")); 587 return; 588 } 589 590 /* 591 * set up desired_mechs so it points to mech_type. 592 */ 593 594 status = kgss_add_cred( 595 &minor_status, 596 acceptor_credentials, 597 desired_name, 598 mech_type, 599 cred_usage, 600 initiator_time_req, 601 acceptor_time_req, 602 &actual_mechs, 603 NULL, 604 NULL, 605 uid); 606 607 /* store major and minor status for gss_display_status() call */ 608 609 gss_major_code = status; 610 gss_minor_code = minor_status; 611 if (status == GSS_S_COMPLETE) { 612 /* process returned values */ 613 614 printf(gettext("\nadd succeeded\n\n")); 615 if (actual_mechs) { 616 for (i = 0; i < actual_mechs->count; i++) { 617 if ((string = 618 gss_oid2str 619 (&actual_mechs->elements[i])) == 0) { 620 printf(gettext 621 ("actual mechs == NULL\n\n")); 622 } else { 623 printf(gettext 624 ("actual mechs = %s\n\n"), string); 625 FREE(string, 626 (actual_mechs->elements->length+1)*4+1); 627 } 628 } 629 } 630 /* 631 * Try adding the cred again for the same mech 632 * We should get GSS_S_DUPLICATE_ELEMENT 633 * if not return an error 634 */ 635 status = kgss_add_cred( 636 &minor_status, 637 acceptor_credentials, 638 desired_name, 639 mech_type, 640 cred_usage, 641 initiator_time_req, 642 acceptor_time_req, 643 NULL, /* &actual_mechs, */ 644 NULL, 645 NULL, 646 uid); 647 if (status != GSS_S_DUPLICATE_ELEMENT) { 648 printf(gettext("Expected duplicate element, Got " 649 " (octal) %o (%s)\n"), 650 status, gettext("gss_add_cred error")); 651 } 652 status = kgss_inquire_cred( 653 &minor_status, 654 acceptor_credentials, 655 NULL, 656 &time_req, 657 &cred_usage, 658 &inquire_mechs, 659 uid); 660 661 if (status != GSS_S_COMPLETE) 662 printf(gettext("server ret err (octal) %o (%s)\n"), 663 status, gettext("gss_inquire_cred error")); 664 else { 665 for (i = 0; i < inquire_mechs->count; i++) { 666 if ((string = 667 gss_oid2str 668 (&inquire_mechs->elements[i])) == 0) { 669 printf(gettext 670 ("inquire_mechs mechs == NULL\n\n")); 671 } else { 672 printf(gettext 673 ("inquire_cred mechs = %s\n\n"), 674 string); 675 FREE(string, 676 (inquire_mechs->elements->length+1)*4 677 +1); 678 } 679 } 680 printf(gettext("inquire_cred successful \n\n")); 681 } 682 683 } else { 684 printf(gettext("server ret err (octal) %o (%s)\n"), 685 status, gettext("gss_acquire_cred error")); 686 } 687 688 /* Let us do inquire_cred_by_mech for both mechanisms */ 689 status = kgss_inquire_cred_by_mech( 690 &minor_status, 691 acceptor_credentials, 692 mech_type, 693 uid); 694 if (status != GSS_S_COMPLETE) 695 printf(gettext("server ret err (octal) %o (%s)\n"), 696 status, gettext("gss_inquire_cred_by_mech")); 697 else 698 printf(gettext("gss_inquire_cred_by_mech successful")); 699 700 701 FREE(mech_type->elements, mech_type->length); 702 FREE(mech_type, sizeof (gss_OID_desc)); 703 mech_type = gss_str2oid((char *)GSS_KRB5_MECH_OID); 704 status = kgss_inquire_cred_by_mech( 705 &minor_status, 706 acceptor_credentials, 707 mech_type, 708 uid); 709 if (status != GSS_S_COMPLETE) 710 printf(gettext("server ret err (octal) %o (%s)\n"), 711 status, gettext 712 ("gss_inquire_cred_by_mech for dummy mech error")); 713 714 /* free allocated memory */ 715 716 /* actual mechs is allocated by clnt_stubs. Release it here */ 717 if (actual_mechs != GSS_C_NULL_OID_SET) 718 gss_release_oid_set_and_oids(&minor_status, &actual_mechs); 719 if (inquire_mechs != GSS_C_NULL_OID_SET) 720 gss_release_oid_set_and_oids(&minor_status, &inquire_mechs); 721 722 gss_release_name(&minor_status, &desired_name); 723 724 /* mech_type and desired_mechs are allocated above. Release it here */ 725 726 FREE(mech_type->elements, mech_type->length); 727 FREE(mech_type, sizeof (gss_OID_desc)); 728 } 729 730 /*ARGSUSED*/ 731 static void 732 _gss_release_cred(argc, argv) 733 int argc; 734 char **argv; 735 { 736 OM_UINT32 status; 737 OM_UINT32 minor_status; 738 uid_t uid; 739 740 /* set up input arguments here */ 741 742 if (argc != 0) { 743 usage(); 744 return; 745 } 746 747 uid = getuid(); 748 749 status = kgss_release_cred( 750 &minor_status, 751 &acceptor_credentials, 752 uid); 753 754 /* store major and minor status for gss_display_status() call */ 755 756 gss_major_code = status; 757 gss_minor_code = minor_status; 758 759 if (status == GSS_S_COMPLETE) { 760 printf(gettext("\nrelease succeeded\n\n")); 761 } else { 762 printf(gettext("server ret err (octal) %o (%s)\n"), 763 status, gettext("gss_release_cred error")); 764 } 765 } 766 767 static void 768 _gss_init_sec_context(argc, argv) 769 int argc; 770 char **argv; 771 { 772 773 OM_uint32 status; 774 775 OM_uint32 minor_status; 776 gss_cred_id_t claimant_cred_handle; 777 gss_name_t target_name = (gss_name_t) 0; 778 gss_OID mech_type = (gss_OID) 0; 779 int req_flags; 780 OM_uint32 time_req; 781 gss_channel_bindings_t input_chan_bindings; 782 gss_buffer_t input_token; 783 gss_buffer_desc context_token; 784 gss_OID actual_mech_type; 785 int ret_flags; 786 OM_uint32 time_rec; 787 uid_t uid; 788 char * string; 789 gss_buffer_desc name; 790 791 /* 792 * If this is the first phase of the context establishment, 793 * clear initiator_context_handle and indicate next phase. 794 */ 795 796 if (init_sec_context_phase == 0) { 797 initiator_context_handle = GSS_C_NO_CONTEXT; 798 input_token = GSS_C_NO_BUFFER; 799 init_sec_context_phase = 1; 800 } else 801 input_token = &init_token_buffer; 802 803 /* 804 * First set up the non-variable command line independent input 805 * arguments 806 */ 807 808 claimant_cred_handle = GSS_C_NO_CREDENTIAL; 809 810 req_flags = GSS_C_MUTUAL_FLAG; 811 time_req = (OM_uint32) 0; 812 input_chan_bindings = GSS_C_NO_CHANNEL_BINDINGS; 813 uid = getuid(); 814 815 /* Now parse the command line for the remaining input arguments */ 816 817 if (argc == 0) { 818 usage(); 819 return; 820 } 821 822 /* 823 * Get the name of the target. 824 */ 825 826 name.length = strlen(argv[0])+1; 827 name.value = argv[0]; 828 829 /* 830 * Now convert the string given by the first argument into a target 831 * name suitable for input to gss_init_sec_context() 832 */ 833 834 if ((status = gss_import_name(&minor_status, &name, 835 /* GSS_C_NULL_OID, &target_name)) */ 836 (gss_OID)GSS_C_NT_HOSTBASED_SERVICE, &target_name)) 837 != GSS_S_COMPLETE) { 838 printf(gettext( 839 "could not parse target name: err (octal) %o (%s)\n"), 840 status, 841 gettext("gss_init_sec_context error")); 842 if (input_token != GSS_C_NO_BUFFER) 843 gss_release_buffer(&minor_status, &init_token_buffer); 844 init_sec_context_phase = 0; 845 return; 846 } 847 848 argc--; 849 argv++; 850 851 if (argc == 0) { 852 printf(gettext("Assuming Kerberos V5 as the mechanism\n")); 853 printf(gettext( 854 "The mech OID 1.2.840.113554.1.2.2 will be used\n")); 855 mech_type = gss_str2oid((char *)GSS_KRB5_MECH_OID); 856 } else { 857 mech_type = gss_str2oid(argv[0]); 858 } 859 860 if (mech_type == 0 || mech_type->length == 0) { 861 printf(gettext("improperly formated mechanism OID\n")); 862 if (input_token != GSS_C_NO_BUFFER) 863 gss_release_buffer(&minor_status, &init_token_buffer); 864 init_sec_context_phase = 0; 865 return; 866 } 867 868 /* call kgss_init_sec_context */ 869 870 status = kgss_init_sec_context(&minor_status, 871 claimant_cred_handle, 872 &initiator_context_handle, 873 target_name, 874 mech_type, 875 req_flags, 876 time_req, 877 input_chan_bindings, 878 input_token, 879 &actual_mech_type, 880 &accept_token_buffer, 881 &ret_flags, 882 &time_rec, 883 uid); 884 885 /* store major and minor status for gss_display_status() call */ 886 gss_major_code = status; 887 gss_minor_code = minor_status; 888 889 if (status != GSS_S_COMPLETE && 890 status != GSS_S_CONTINUE_NEEDED) { 891 892 printf(gettext("server ret err (octal) %o (%s)\n"), 893 status, "gss_init_sec_context error"); 894 init_sec_context_phase = 0; 895 if (status == GSS_S_NO_CRED) 896 printf(gettext(" : no credentials")); 897 if (input_token != GSS_C_NO_BUFFER) 898 gss_release_buffer(&minor_status, &init_token_buffer); 899 if (status != GSS_S_FAILURE && minor_status != 0xffffffff) 900 status = kgss_delete_sec_context(&minor_status, 901 &initiator_context_handle, 902 &msg_token); 903 return; 904 905 } else if (status == GSS_S_COMPLETE) { 906 907 /* process returned values */ 908 909 printf(gettext("\ninit succeeded\n\n")); 910 911 /* print out the actual mechanism type */ 912 913 if ((string = gss_oid2str(actual_mech_type)) == 0) { 914 915 printf(gettext( 916 "gssapi internal err : actual " 917 "mech type null\n")); 918 init_sec_context_phase = 0; 919 if (input_token != GSS_C_NO_BUFFER) 920 gss_release_buffer(&minor_status, 921 &init_token_buffer); 922 gss_release_buffer(&minor_status, &accept_token_buffer); 923 status = kgss_delete_sec_context(&minor_status, 924 &initiator_context_handle, 925 &msg_token); 926 return; 927 } else { 928 printf(gettext("actual mech type = %s\n\n"), string); 929 FREE(string, (actual_mech_type->length+1)*4+1); 930 } 931 932 /* print out value of ret_flags and time_req */ 933 934 if (ret_flags & GSS_C_DELEG_FLAG) 935 printf(gettext("GSS_C_DELEG_FLAG = True\n")); 936 else 937 printf(gettext("GSS_C_DELEG_FLAG = False\n")); 938 939 if (ret_flags & GSS_C_MUTUAL_FLAG) 940 printf(gettext("GSS_C_MUTUAL_FLAG = True\n")); 941 else 942 printf(gettext("GSS_C_MUTUAL_FLAG = False\n")); 943 944 if (ret_flags & GSS_C_REPLAY_FLAG) 945 printf(gettext("GSS_C_REPLAY_FLAG = True\n")); 946 else 947 printf(gettext("GSS_C_REPLAY_FLAG = False\n")); 948 949 if (ret_flags & GSS_C_SEQUENCE_FLAG) 950 printf(gettext("GSS_C_SEQUENCE_FLAG = True\n")); 951 else 952 printf(gettext("GSS_C_SEQUENCE_FLAG = False\n")); 953 954 if (ret_flags & GSS_C_CONF_FLAG) 955 printf(gettext("GSS_C_CONF_FLAG = True\n")); 956 else 957 printf(gettext("GSS_C_CONF_FLAG = False\n")); 958 959 if (ret_flags & GSS_C_INTEG_FLAG) 960 printf(gettext("GSS_C_INTEG_FLAG = True\n\n")); 961 else 962 printf(gettext("GSS_C_INTEG_FLAG = False\n\n")); 963 964 printf(gettext("time_req = %u seconds\n\n"), time_rec); 965 966 /* free allocated memory */ 967 968 FREE(mech_type->elements, mech_type->length); 969 FREE(mech_type, sizeof (gss_OID_desc)); 970 971 /* these two were malloc'd by kgss_init_sec_context() */ 972 973 FREE(actual_mech_type->elements, actual_mech_type->length); 974 FREE(actual_mech_type, sizeof (gss_OID_desc)); 975 976 gss_release_name(&minor_status, &target_name); 977 978 if (input_token != GSS_C_NO_BUFFER) 979 gss_release_buffer(&minor_status, &init_token_buffer); 980 981 /* 982 * if status == GSS_S_COMPLETE, reset the phase to 0 and 983 * release token in accept_token_buffer 984 */ 985 986 init_sec_context_phase = 0; 987 /* Save and restore the context */ 988 status = kgss_export_sec_context(&minor_status, 989 &initiator_context_handle, 990 &context_token); 991 if (status != GSS_S_COMPLETE) { 992 printf(gettext("server ret err (octal) %o (%s)\n"), 993 status, gettext("gss_export_sec_context_error")); 994 return; 995 } 996 status = kgss_import_sec_context(&minor_status, 997 &context_token, 998 &initiator_context_handle); 999 if (status != GSS_S_COMPLETE) { 1000 printf(gettext("server ret err (octal) %o (%s)\n"), 1001 status, gettext("gss_import_sec_context_error")); 1002 return; 1003 } 1004 (void) gss_release_buffer(&minor_status, &context_token); 1005 1006 /* gss_export & gss_import secxc_context worked, return */ 1007 printf(gettext("\nexport and import of contexts succeeded\n")); 1008 printf(gettext("\ninit completed")); 1009 1010 } else { 1011 printf(gettext("\nfirst phase of init succeeded")); 1012 printf(gettext("\ninit must be called again\n\n")); 1013 } 1014 1015 } 1016 1017 /*ARGSUSED*/ 1018 static void 1019 _gss_accept_sec_context(argc, argv) 1020 int argc; 1021 char **argv; 1022 { 1023 OM_UINT32 status; 1024 1025 OM_uint32 minor_status; 1026 gss_channel_bindings_t input_chan_bindings; 1027 gss_OID mech_type; 1028 int ret_flags; 1029 OM_uint32 time_rec; 1030 gss_cred_id_t delegated_cred_handle; 1031 uid_t uid; 1032 char *string; 1033 gss_buffer_desc src_name, src_name_string; 1034 gss_buffer_desc output_token; 1035 gss_name_t gss_name; 1036 gss_buffer_desc context_token; 1037 1038 /* 1039 * If this is the first phase of the context establishment, 1040 * clear acceptor_context_handle and indicate next phase. 1041 */ 1042 1043 if (accept_sec_context_phase == 0) { 1044 acceptor_context_handle = GSS_C_NO_CONTEXT; 1045 accept_sec_context_phase = 1; 1046 } 1047 1048 /* Now set up the other command line independent input arguments */ 1049 1050 input_chan_bindings = GSS_C_NO_CHANNEL_BINDINGS; 1051 1052 uid = (uid_t) getuid(); 1053 1054 if (argc != 0) { 1055 usage(); 1056 return; 1057 } 1058 1059 status = kgss_accept_sec_context(&minor_status, 1060 &acceptor_context_handle, 1061 acceptor_credentials, 1062 &accept_token_buffer, 1063 input_chan_bindings, 1064 &src_name, 1065 &mech_type, 1066 &init_token_buffer, 1067 &ret_flags, 1068 &time_rec, 1069 &delegated_cred_handle, 1070 uid); 1071 1072 /* store major and minor status for gss_display_status() call */ 1073 1074 gss_major_code = status; 1075 gss_minor_code = minor_status; 1076 1077 if (status != GSS_S_COMPLETE && status != GSS_S_CONTINUE_NEEDED) { 1078 printf(gettext("server ret err (octal) %o (%s)\n"), 1079 status, gettext("gss_accept_sec_context error")); 1080 gss_release_buffer(&minor_status, &accept_token_buffer); 1081 return; 1082 } else if (status == GSS_S_COMPLETE) { 1083 1084 /* process returned values */ 1085 1086 printf(gettext("\naccept succeeded\n\n")); 1087 1088 /* 1089 * convert the exported name returned in src_name into 1090 * a string and print it. 1091 */ 1092 if ((status = gss_import_name(&minor_status, &src_name, 1093 (gss_OID) GSS_C_NT_EXPORT_NAME, &gss_name)) 1094 != GSS_S_COMPLETE) { 1095 printf(gettext( 1096 "could not import src name 0x%x\n"), status); 1097 accept_sec_context_phase = 0; 1098 status = kgss_delete_sec_context(&minor_status, 1099 &acceptor_context_handle, 1100 &output_token); 1101 gss_release_buffer(&minor_status, &accept_token_buffer); 1102 if (status == GSS_S_CONTINUE_NEEDED) 1103 gss_release_buffer(&minor_status, 1104 &init_token_buffer); 1105 gss_release_buffer(&minor_status, &src_name); 1106 return; 1107 } 1108 1109 memset(&src_name_string, 0, sizeof (src_name_string)); 1110 if ((status = gss_display_name(&minor_status, gss_name, 1111 &src_name_string, NULL)) != GSS_S_COMPLETE) { 1112 printf(gettext("could not display src name: " 1113 "err (octal) %o (%s)\n"), status, 1114 "gss_init_sec_context error"); 1115 accept_sec_context_phase = 0; 1116 status = kgss_delete_sec_context(&minor_status, 1117 &acceptor_context_handle, 1118 &output_token); 1119 gss_release_buffer(&minor_status, &accept_token_buffer); 1120 if (status == GSS_S_CONTINUE_NEEDED) 1121 gss_release_buffer(&minor_status, 1122 &init_token_buffer); 1123 gss_release_buffer(&minor_status, &src_name); 1124 return; 1125 } 1126 printf(gettext("src name = %s\n"), src_name_string.value); 1127 gss_release_name(&minor_status, &gss_name); 1128 gss_release_buffer(&minor_status, &src_name_string); 1129 gss_release_buffer(&minor_status, &src_name); 1130 1131 /* print out the mechanism type */ 1132 1133 if ((string = gss_oid2str(mech_type)) == 0) { 1134 1135 printf(gettext( 1136 "gssapi internal err :" 1137 " actual mech type null\n")); 1138 accept_sec_context_phase = 0; 1139 status = kgss_delete_sec_context(&minor_status, 1140 &acceptor_context_handle, 1141 &output_token); 1142 gss_release_buffer(&minor_status, &accept_token_buffer); 1143 if (status == GSS_S_CONTINUE_NEEDED) 1144 gss_release_buffer(&minor_status, 1145 &init_token_buffer); 1146 return; 1147 } else { 1148 1149 printf(gettext("actual mech type = %s\n\n"), string); 1150 FREE(string, (mech_type->length+1)*4+1); 1151 } 1152 1153 /* Save and restore the context */ 1154 status = kgss_export_sec_context(&minor_status, 1155 &initiator_context_handle, 1156 &context_token); 1157 if (status != GSS_S_COMPLETE) { 1158 printf(gettext("server ret err (octal) %o (%s)\n"), 1159 status, gettext("gss_export_sec_context_error")); 1160 return; 1161 } 1162 status = kgss_import_sec_context(&minor_status, 1163 &context_token, 1164 &initiator_context_handle); 1165 if (status != GSS_S_COMPLETE) { 1166 printf(gettext("server ret err (octal) %o (%s)\n"), 1167 status, gettext("gss_import_sec_context_error")); 1168 return; 1169 } 1170 (void) gss_release_buffer(&minor_status, &context_token); 1171 1172 /* gss_export & gss_import secxc_context worked, return */ 1173 1174 /* print out value of ret_flags and time_req */ 1175 1176 if (ret_flags & GSS_C_DELEG_FLAG) 1177 printf(gettext("GSS_C_DELEG_FLAG = True\n")); 1178 else 1179 printf(gettext("GSS_C_DELEG_FLAG = False\n")); 1180 1181 if (ret_flags & GSS_C_MUTUAL_FLAG) 1182 printf(gettext("GSS_C_MUTUAL_FLAG = True\n")); 1183 else 1184 printf(gettext("GSS_C_MUTUAL_FLAG = False\n")); 1185 1186 if (ret_flags & GSS_C_REPLAY_FLAG) 1187 printf(gettext("GSS_C_REPLAY_FLAG = True\n")); 1188 else 1189 printf(gettext("GSS_C_REPLAY_FLAG = False\n")); 1190 1191 if (ret_flags & GSS_C_SEQUENCE_FLAG) 1192 printf(gettext("GSS_C_SEQUENCE_FLAG = True\n")); 1193 else 1194 printf(gettext("GSS_C_SEQUENCE_FLAG = False\n")); 1195 1196 if (ret_flags & GSS_C_CONF_FLAG) 1197 printf(gettext("GSS_C_CONF_FLAG = True\n")); 1198 else 1199 printf(gettext("GSS_C_CONF_FLAG = False\n")); 1200 1201 if (ret_flags & GSS_C_INTEG_FLAG) 1202 printf(gettext("GSS_C_INTEG_FLAG = True\n\n")); 1203 else 1204 printf(gettext("GSS_C_INTEG_FLAG = False\n\n")); 1205 1206 printf(gettext("time_rec = %d seconds\n\n"), time_rec); 1207 1208 /* free allocated memory */ 1209 1210 printf(gettext("\nexport and import of contexts succeeded\n")); 1211 1212 FREE(mech_type->elements, mech_type->length); 1213 FREE(mech_type, sizeof (gss_OID_desc)); 1214 } else { 1215 printf(gettext("\nfirst phase of accept succeeded")); 1216 printf(gettext("\naccept must be called again\n\n")); 1217 } 1218 1219 1220 /* free the input token in accept_token_buffer */ 1221 gss_release_buffer(&minor_status, &accept_token_buffer); 1222 1223 /* if status == GSS_S_COMPLETE, reset the phase to 0 */ 1224 1225 if (status == GSS_S_COMPLETE) 1226 accept_sec_context_phase = 0; 1227 1228 /* gss_accept_sec_context worked, return */ 1229 } 1230 1231 void 1232 _gss_process_context_token(argc, argv) 1233 int argc; 1234 char **argv; 1235 { 1236 OM_UINT32 status; 1237 1238 gss_ctx_id_t context_handle; 1239 OM_uint32 minor_status; 1240 uid_t uid; 1241 1242 uid = (uid_t) getuid(); 1243 1244 /* parse the command line to determine the variable input argument */ 1245 1246 if (argc == 0) { 1247 usage(); 1248 return; 1249 } 1250 1251 if (strcmp(argv[0], "initiator") == 0) 1252 context_handle = initiator_context_handle; 1253 else if (strcmp(argv[0], "acceptor") == 0) 1254 context_handle = acceptor_context_handle; 1255 else { 1256 printf(gettext( 1257 "must specify either \"initiator\" or \"acceptor\"\n")); 1258 return; 1259 } 1260 1261 argc--; 1262 argv++; 1263 1264 if (argc != 0) { 1265 usage(); 1266 return; 1267 } 1268 1269 status = kgss_process_context_token(&minor_status, 1270 context_handle, 1271 delete_token_buffer, 1272 uid); 1273 1274 /* store major and minor status for gss_display_status() call */ 1275 1276 gss_major_code = status; 1277 gss_minor_code = minor_status; 1278 1279 if (status != GSS_S_COMPLETE) { 1280 printf(gettext("server ret err (octal) %o (%s)\n"), 1281 status, gettext("gss_process_context_token error")); 1282 return; 1283 1284 } else { 1285 printf(gettext("\nprocess succeeded\n\n")); 1286 return; 1287 } 1288 } 1289 1290 static void 1291 _gss_delete_sec_context(argc, argv) 1292 int argc; 1293 char **argv; 1294 { 1295 OM_UINT32 status; 1296 gss_ctx_id_t *context_handle; 1297 OM_uint32 minor_status; 1298 1299 1300 /* parse the command line to determine the variable input argument */ 1301 1302 if (argc == 0) { 1303 usage(); 1304 return; 1305 } 1306 1307 if (strcmp(argv[0], "initiator") == 0) { 1308 context_handle = &initiator_context_handle; 1309 } else if (strcmp(argv[0], "acceptor") == 0) { 1310 context_handle = &acceptor_context_handle; 1311 } else { 1312 printf(gettext( 1313 "must specify either \"initiator\" or \"acceptor\"\n")); 1314 return; 1315 } 1316 1317 argc--; 1318 argv++; 1319 1320 if (argc != 0) { 1321 usage(); 1322 return; 1323 } 1324 1325 1326 status = kgss_delete_sec_context(&minor_status, 1327 context_handle, 1328 &delete_token_buffer); 1329 1330 1331 /* store major and minor status for gss_display_status() call */ 1332 1333 gss_major_code = status; 1334 gss_minor_code = minor_status; 1335 1336 if (status != GSS_S_COMPLETE) { 1337 1338 printf(gettext("server ret err (octal) %o (%s)\n"), 1339 status, gettext("gss_delete_sec_context error")); 1340 return; 1341 1342 } else { 1343 printf(gettext("\ndelete succeeded\n\n")); 1344 return; 1345 } 1346 } 1347 1348 /*ARGSUSED*/ 1349 static void 1350 _gss_context_time(argc, argv) 1351 int argc; 1352 char **argv; 1353 { 1354 /* 1355 * set up input arguments here 1356 * this function is unimplemented. Call usage() and return 1357 */ 1358 1359 printf(gettext("\nunimplemented function")); 1360 } 1361 1362 static void 1363 _gss_sign(argc, argv) 1364 int argc; 1365 char **argv; 1366 { 1367 OM_UINT32 status; 1368 OM_uint32 minor_status; 1369 gss_ctx_id_t context_handle; 1370 int qop_req; 1371 uid_t uid; 1372 1373 uid = (uid_t) getuid(); 1374 1375 /* specify the default quality of protection */ 1376 1377 qop_req = GSS_C_QOP_DEFAULT; 1378 1379 /* set up the arguments specified in the input parameters */ 1380 1381 if (argc == 0) { 1382 usage(); 1383 return; 1384 } 1385 1386 1387 if (strcmp(argv[0], "initiator") == 0) 1388 context_handle = initiator_context_handle; 1389 else if (strcmp(argv[0], "acceptor") == 0) 1390 context_handle = acceptor_context_handle; 1391 else { 1392 printf(gettext( 1393 "must specify either \"initiator\" or \"acceptor\"\n")); 1394 return; 1395 } 1396 1397 argc--; 1398 argv++; 1399 1400 if (argc == 0) { 1401 usage(); 1402 return; 1403 } 1404 1405 message_buffer.length = strlen(argv[0])+1; 1406 message_buffer.value = (void *) MALLOC(message_buffer.length); 1407 strcpy(message_buffer.value, argv[0]); 1408 1409 argc--; 1410 argv++; 1411 1412 if (argc != 0) { 1413 usage(); 1414 return; 1415 } 1416 1417 status = kgss_sign(&minor_status, 1418 context_handle, 1419 qop_req, 1420 &message_buffer, 1421 &msg_token, 1422 uid); 1423 1424 /* store major and minor status for gss_display_status() call */ 1425 1426 gss_major_code = status; 1427 gss_minor_code = minor_status; 1428 1429 if (status != GSS_S_COMPLETE) { 1430 printf(gettext("server ret err (octal) %o (%s)\n"), 1431 status, gettext("gss_sign error")); 1432 return; 1433 1434 } else { 1435 printf(gettext("\nsign succeeded\n\n")); 1436 return; 1437 } 1438 } 1439 1440 static void 1441 _gss_verify(argc, argv) 1442 int argc; 1443 char **argv; 1444 { 1445 OM_UINT32 status, minor_status; 1446 gss_ctx_id_t context_handle; 1447 int qop_state; 1448 uid_t uid; 1449 1450 uid = (uid_t) getuid(); 1451 1452 /* set up the arguments specified in the input parameters */ 1453 1454 if (argc == 0) { 1455 usage(); 1456 return; 1457 } 1458 1459 1460 if (strcmp(argv[0], "initiator") == 0) 1461 context_handle = initiator_context_handle; 1462 else if (strcmp(argv[0], "acceptor") == 0) 1463 context_handle = acceptor_context_handle; 1464 else { 1465 printf(gettext( 1466 "must specify either \"initiator\" or \"acceptor\"\n")); 1467 return; 1468 } 1469 1470 argc--; 1471 argv++; 1472 1473 if (argc != 0) { 1474 usage(); 1475 return; 1476 } 1477 1478 status = kgss_verify(&minor_status, 1479 context_handle, 1480 &message_buffer, 1481 &msg_token, 1482 &qop_state, 1483 uid); 1484 1485 /* store major and minor status for gss_display_status() call */ 1486 1487 gss_major_code = status; 1488 gss_minor_code = minor_status; 1489 1490 if (status != GSS_S_COMPLETE) { 1491 printf(gettext("server ret err (octal) %o (%s)\n"), 1492 status, gettext("gss_verify error")); 1493 return; 1494 } else { 1495 1496 /* print out the verified message */ 1497 1498 printf(gettext( 1499 "verified message = \"%s\"\n\n"), message_buffer.value); 1500 1501 /* print out the quality of protection returned */ 1502 1503 printf(gettext("quality of protection = %d \n\n"), qop_state); 1504 1505 /* free the message buffer and message token and return */ 1506 1507 gss_release_buffer(&minor_status, &message_buffer); 1508 gss_release_buffer(&minor_status, &msg_token); 1509 1510 return; 1511 } 1512 } 1513 1514 static void 1515 _gss_seal(argc, argv) 1516 int argc; 1517 char **argv; 1518 { 1519 OM_UINT32 status; 1520 1521 OM_uint32 minor_status; 1522 gss_ctx_id_t context_handle; 1523 int conf_req_flag; 1524 int qop_req; 1525 gss_buffer_desc input_message_buffer; 1526 int conf_state; 1527 uid_t uid; 1528 1529 uid = (uid_t) getuid(); 1530 1531 /* 1532 * specify the default confidentiality requested (both integrity 1533 * and confidentiality) and quality of protection 1534 */ 1535 1536 conf_req_flag = 1; 1537 qop_req = GSS_C_QOP_DEFAULT; 1538 1539 /* set up the arguments specified in the input parameters */ 1540 1541 if (argc == 0) { 1542 usage(); 1543 return; 1544 } 1545 1546 1547 if (strcmp(argv[0], "initiator") == 0) 1548 context_handle = initiator_context_handle; 1549 else if (strcmp(argv[0], "acceptor") == 0) 1550 context_handle = acceptor_context_handle; 1551 else { 1552 printf(gettext( 1553 "must specify either \"initiator\" or \"acceptor\"\n")); 1554 return; 1555 } 1556 1557 argc--; 1558 argv++; 1559 1560 if (argc == 0) { 1561 usage(); 1562 return; 1563 } 1564 1565 1566 input_message_buffer.length = strlen(argv[0])+1; 1567 input_message_buffer.value = 1568 (void *) MALLOC(input_message_buffer.length); 1569 strcpy(input_message_buffer.value, argv[0]); 1570 1571 argc--; 1572 argv++; 1573 1574 if (argc != 0) { 1575 usage(); 1576 return; 1577 } 1578 1579 status = kgss_seal(&minor_status, 1580 context_handle, 1581 conf_req_flag, 1582 qop_req, 1583 &input_message_buffer, 1584 &conf_state, 1585 &message_buffer, 1586 uid); 1587 1588 /* store major and minor status for gss_display_status() call */ 1589 1590 gss_major_code = status; 1591 gss_minor_code = minor_status; 1592 1593 /* free the inputmessage buffer */ 1594 1595 gss_release_buffer(&minor_status, &input_message_buffer); 1596 1597 if (status != GSS_S_COMPLETE) { 1598 printf(gettext("server ret err (octal) %o (%s)\n"), 1599 status, gettext("gss_seal error")); 1600 return; 1601 } else { 1602 printf(gettext("\nseal succeeded\n\n")); 1603 return; 1604 } 1605 } 1606 1607 static void 1608 _gss_unseal(argc, argv) 1609 int argc; 1610 char **argv; 1611 { 1612 OM_UINT32 status; 1613 1614 OM_uint32 minor_status; 1615 gss_ctx_id_t context_handle; 1616 gss_buffer_desc output_message_buffer; 1617 int conf_state; 1618 int qop_state; 1619 uid_t uid; 1620 1621 uid = (uid_t) getuid(); 1622 1623 /* set up the arguments specified in the input parameters */ 1624 1625 if (argc == 0) { 1626 usage(); 1627 return; 1628 } 1629 1630 1631 if (strcmp(argv[0], "initiator") == 0) 1632 context_handle = initiator_context_handle; 1633 else if (strcmp(argv[0], "acceptor") == 0) 1634 context_handle = acceptor_context_handle; 1635 else { 1636 printf(gettext( 1637 "must specify either \"initiator\" or \"acceptor\"\n")); 1638 return; 1639 } 1640 1641 argc--; 1642 argv++; 1643 1644 if (argc != 0) { 1645 usage(); 1646 return; 1647 } 1648 1649 status = kgss_unseal(&minor_status, 1650 context_handle, 1651 &message_buffer, 1652 &output_message_buffer, 1653 &conf_state, 1654 &qop_state, 1655 uid); 1656 1657 /* store major and minor status for gss_display_status() call */ 1658 1659 gss_major_code = status; 1660 gss_minor_code = minor_status; 1661 1662 if (status == GSS_S_COMPLETE) { 1663 printf(gettext("\nunseal succeeded\n\n")); 1664 printf(gettext("unsealed message = \"%s\"\n\n"), 1665 output_message_buffer.value); 1666 if (conf_state) 1667 printf(gettext("confidentiality and integrity used\n")); 1668 else 1669 printf(gettext("only integrity used\n")); 1670 printf(gettext("quality of protection = %d\n\n"), qop_state); 1671 gss_release_buffer(&minor_status, &output_message_buffer); 1672 } else { 1673 printf(gettext("server ret err (octal) %o (%s)\n"), 1674 status, gettext("gss_unseal error")); 1675 } 1676 1677 /* free the message buffer and return */ 1678 1679 gss_release_buffer(&minor_status, &message_buffer); 1680 } 1681 1682 static void 1683 _gss_display_status(argc, argv) 1684 int argc; 1685 char **argv; 1686 { 1687 OM_UINT32 status; 1688 OM_uint32 minor_status; 1689 int status_type; 1690 int status_value; 1691 gss_OID mech_type = (gss_OID) 0; 1692 int message_context; 1693 gss_buffer_desc status_string; 1694 uid_t uid; 1695 1696 uid = (uid_t) getuid(); 1697 1698 /* initialize message context to zero */ 1699 1700 message_context = 0; 1701 1702 if (argc == 0) { 1703 printf(gettext("Assuming Kerberos V5 as the mechanism\n")); 1704 printf(gettext( 1705 "The mech OID 1.2.840.113554.1.2.2 will be used\n")); 1706 mech_type = gss_str2oid((char *)GSS_KRB5_MECH_OID); 1707 } else 1708 mech_type = gss_str2oid(argv[0]); 1709 1710 if (mech_type == 0 || mech_type->length == 0) { 1711 printf(gettext("improperly formated mechanism OID\n")); 1712 return; 1713 } 1714 1715 /* Is this call for the major or minor status? */ 1716 1717 if (strcmp(argv[0], "major") == 0) { 1718 status_type = GSS_C_GSS_CODE; 1719 status_value = gss_major_code; 1720 } else if (strcmp(argv[0], "minor") == 0) { 1721 status_type = GSS_C_MECH_CODE; 1722 status_value = gss_minor_code; 1723 } else { 1724 printf(gettext("must specify either \"major\" or \"minor\"\n")); 1725 return; 1726 } 1727 1728 argc--; 1729 argv++; 1730 1731 if (argc != 0) { 1732 usage(); 1733 return; 1734 } 1735 1736 status = kgss_display_status(&minor_status, 1737 status_value, 1738 status_type, 1739 mech_type, 1740 &message_context, 1741 &status_string, 1742 uid); 1743 1744 if (status == GSS_S_COMPLETE) { 1745 printf(gettext("status =\n %s\n\n"), status_string.value); 1746 } else if (status == GSS_S_BAD_MECH) { 1747 printf(gettext("invalide mechanism OID\n\n")); 1748 } else { 1749 printf(gettext("server ret err (octal) %o (%s)\n"), 1750 status, gettext("gss_display_status error")); 1751 } 1752 } 1753 1754 /*ARGSUSED*/ 1755 static void 1756 _gss_indicate_mechs(argc, argv) 1757 int argc; 1758 char **argv; 1759 { 1760 OM_UINT32 status; 1761 OM_UINT32 minor_status; 1762 gss_OID_set oid_set = GSS_C_NULL_OID_SET; 1763 uid_t uid; 1764 1765 uid = (uid_t) getuid(); 1766 1767 /* set up input arguments here */ 1768 1769 if (argc != 0) { 1770 usage(); 1771 return; 1772 } 1773 1774 status = kgss_indicate_mechs(&minor_status, &oid_set, uid); 1775 1776 if (status == GSS_S_COMPLETE) { 1777 int i; 1778 char *string; 1779 1780 printf(gettext("%d supported mechanism%s%s\n"), oid_set->count, 1781 (oid_set->count == 1) ? "" : "s", 1782 (oid_set->count > 0) ? ":" : ""); 1783 1784 for (i = 0; i < oid_set->count; i++) { 1785 string = gss_oid2str(&oid_set->elements[i]); 1786 printf(gettext("\t%s\n"), string); 1787 FREE(string, ((oid_set->elements[i].length+1)*4)+1); 1788 } 1789 printf("\n"); 1790 1791 } else { 1792 printf(gettext("server ret err (octal) %o (%s)\n"), 1793 status, gettext("gss_indicate_mechs error")); 1794 } 1795 1796 if (oid_set) 1797 gss_release_oid_set_and_oids(&minor_status, &oid_set); 1798 } 1799 1800 /*ARGSUSED*/ 1801 static void 1802 _gss_inquire_cred(argc, argv) 1803 int argc; 1804 char **argv; 1805 { 1806 /* set up input arguments here */ 1807 1808 if (argc != 0) { 1809 usage(); 1810 return; 1811 } 1812 1813 1814 /* this function is unimplemented. Call usage() and return */ 1815 1816 printf(gettext("\nUnsupported function")); 1817 } 1818 1819 static char hexChars[] = "0123456789ABCDEF"; 1820 1821 static void 1822 _gssd_expname_to_unix_cred(argc, argv) 1823 int argc; 1824 char **argv; 1825 { 1826 OM_uint32 major; 1827 gss_buffer_desc expName; 1828 char krb5_root_name[] = "040100092A864886F712010202000000" 1829 "25000A2A864886F71201020101726F6F744053554E534F46" 1830 "542E454E472E53554E2E434F4D00"; 1831 unsigned char *byteStr, *hexStr; 1832 uid_t uidOut, uidIn; 1833 gid_t *gids, gidOut; 1834 int gidsLen, i, newLen; 1835 1836 /* set up the arguments */ 1837 uidIn = (uid_t) getuid(); 1838 1839 if (argc < 1) { 1840 printf(gettext( 1841 "Using principal name of root for krberos_v5\n")); 1842 expName.value = (void*)krb5_root_name; 1843 expName.length = strlen(krb5_root_name); 1844 } else { 1845 expName.value = (void*)argv[0]; 1846 expName.length = strlen(argv[0]); 1847 } 1848 1849 /* convert the name from hex to byte... */ 1850 hexStr = (unsigned char *)expName.value; 1851 newLen = expName.length/2; 1852 byteStr = (unsigned char *)MALLOC(newLen+1); 1853 expName.value = (char *)byteStr; 1854 for (i = 0; i < expName.length; i += 2) { 1855 *byteStr = (strchr(hexChars, *hexStr++) - hexChars) << 4; 1856 *byteStr += (strchr(hexChars, *hexStr++) - hexChars); 1857 byteStr++; 1858 } 1859 expName.length = newLen; 1860 1861 major = kgsscred_expname_to_unix_cred(&expName, &uidOut, &gidOut, 1862 &gids, &gidsLen, uidIn); 1863 1864 FREE(expName.value, newLen); 1865 1866 if (major == GSS_S_COMPLETE) { 1867 printf(gettext("uid = <%d>\tgid = <%d>\t"), uidOut, gidOut); 1868 if (gidsLen > 0) 1869 printf(gettext(" %d gids <"), gidsLen); 1870 else 1871 printf(gettext( 1872 " no supplementary group information\n")); 1873 for (i = 0; i < gidsLen; i++) 1874 printf(" %d ", gids[i]); 1875 if (gidsLen > 0) { 1876 printf(">\n"); 1877 FREE(gids, gidsLen * sizeof (gid_t)); 1878 } 1879 } else { 1880 printf(gettext("server ret err (octal) %o (%s)\n"), 1881 major, gettext("gsscred_expname_to_unix_cred")); 1882 } 1883 } 1884 1885 static void 1886 _gssd_name_to_unix_cred(argc, argv) 1887 int argc; 1888 char **argv; 1889 { 1890 OM_uint32 major, minor; 1891 gss_name_t gssName; 1892 gss_buffer_desc gssBuf = GSS_C_EMPTY_BUFFER; 1893 int gidsLen, i; 1894 gid_t *gids, gidOut; 1895 uid_t uidOut, uid; 1896 char defaultPrincipal[] = "root"; 1897 gss_OID mechType, nameType; 1898 1899 uid = getuid(); 1900 1901 /* optional argument 1 - contains principal name */ 1902 if (argc > 0) { 1903 gssBuf.value = (void *)argv[0]; 1904 gssBuf.length = strlen((char *)argv[0]); 1905 } else { 1906 gssBuf.value = (void *)defaultPrincipal; 1907 gssBuf.length = strlen(defaultPrincipal); 1908 } 1909 printf(gettext( 1910 "Using <%s> as the principal name.\n"), (char *)gssBuf.value); 1911 1912 1913 /* optional argument 2 - contains name oid */ 1914 if (argc > 1) 1915 nameType = gss_str2oid((char *) argv[1]); 1916 else 1917 nameType = (gss_OID)GSS_C_NT_USER_NAME; 1918 1919 if (nameType == NULL || nameType->length == 0) { 1920 printf(gettext("improperly formated name OID\n")); 1921 return; 1922 } 1923 printf(gettext("Principal name of type: <%s>.\n"), 1924 (argc > 1) ? argv[1] : "GSS_C_NT_USER_NAME"); 1925 1926 1927 /* optional argument 3 - contains mech oid */ 1928 if (argc > 2) 1929 mechType = gss_str2oid(argv[2]); 1930 else 1931 mechType = gss_str2oid((char *)GSS_KRB5_MECH_OID); 1932 1933 if (mechType == NULL || mechType->length == 0) { 1934 FREE(nameType->elements, nameType->length); 1935 FREE(nameType, sizeof (gss_OID_desc)); 1936 printf(gettext("improperly formated mech OID\n")); 1937 return; 1938 } 1939 printf(gettext("Mechanism oid: <%s>.\n"), 1940 (argc > 2) ? argv[2] : 1941 (char *)GSS_KRB5_MECH_OID "(Kerberos v5)"); 1942 1943 1944 /* convert the name to internal format */ 1945 if ((major = gss_import_name(&minor, &gssBuf, 1946 nameType, &gssName)) != GSS_S_COMPLETE) { 1947 printf(gettext("could not parse name: err (octal) %o (%s)\n"), 1948 major, "gss_import_name"); 1949 1950 FREE(nameType->elements, nameType->length); 1951 FREE(nameType, sizeof (gss_OID_desc)); 1952 return; 1953 } 1954 1955 major = kgsscred_name_to_unix_cred(gssName, mechType, &uidOut, 1956 &gidOut, &gids, &gidsLen, uid); 1957 1958 gss_release_name(&minor, &gssName); 1959 FREE(mechType->elements, mechType->length); 1960 FREE(mechType, sizeof (gss_OID_desc)); 1961 if (argc > 1) { 1962 FREE(nameType->elements, nameType->length); 1963 FREE(nameType, sizeof (gss_OID_desc)); 1964 } 1965 1966 if (major == GSS_S_COMPLETE) { 1967 printf("uid = <%d>\tgid = <%d>\t", uidOut, gidOut); 1968 if (gidsLen > 0) 1969 printf(gettext(" %d gids <"), gidsLen); 1970 else 1971 printf(gettext( 1972 " no supplementary group information\n")); 1973 for (i = 0; i < gidsLen; i++) 1974 printf(" %d ", gids[i]); 1975 if (gidsLen > 0) { 1976 printf(">\n"); 1977 FREE(gids, gidsLen * sizeof (gid_t)); 1978 } 1979 } else { 1980 printf(gettext("server ret err (octal) %o (%s)\n"), 1981 major, gettext("gsscred_name_to_unix_cred")); 1982 } 1983 } 1984 1985 static void 1986 _gssd_get_group_info(argc, argv) 1987 int argc; 1988 char **argv; 1989 { 1990 OM_uint32 major; 1991 uid_t puid, uidIn; 1992 gid_t *gids, gidOut; 1993 int gidsLen, i; 1994 1995 /* set up the arguments */ 1996 uidIn = (uid_t) getuid(); 1997 1998 if (argc < 1) 1999 puid = 0; 2000 else 2001 puid = atol(argv[0]); 2002 2003 printf(gettext("Retrieving group info for uid of <%d>\n"), puid); 2004 2005 major = kgss_get_group_info(puid, &gidOut, &gids, &gidsLen, uidIn); 2006 2007 if (major == GSS_S_COMPLETE) { 2008 printf(gettext("group id = <%d>\t"), gidOut); 2009 if (gidsLen > 0) 2010 printf(gettext(" %d gids <"), gidsLen); 2011 else 2012 printf(gettext( 2013 " no supplementary group information\n")); 2014 for (i = 0; i < gidsLen; i++) 2015 printf(" %d ", gids[i]); 2016 if (gidsLen > 0) { 2017 printf(">\n"); 2018 FREE(gids, gidsLen * sizeof (gid_t)); 2019 } 2020 } else { 2021 printf(gettext("server ret err (octal) %o (%s)\n"), 2022 major, "gss_get_group_info"); 2023 } 2024 } 2025 2026 static gss_OID 2027 gss_str2oid(string) 2028 char * string; 2029 { 2030 /* 2031 * a convenient wrapper routine for gss_str_to_oid 2032 * this can handle all valid oid strings. 2033 */ 2034 OM_uint32 minor; 2035 gss_buffer_desc abuf; 2036 gss_OID oidOut; 2037 2038 abuf.value = (void*)string; 2039 abuf.length = strlen(string); 2040 2041 if (gss_str_to_oid(&minor, &abuf, &oidOut) != GSS_S_COMPLETE) 2042 return (NULL); 2043 2044 return (oidOut); 2045 } 2046 2047 static char * 2048 gss_oid2str(oid) 2049 gss_OID oid; 2050 { 2051 /* 2052 * a convenient wrapper for gss_oid_to_str 2053 * this calls the GSS-API routine which should 2054 * be able to handle all types of oids. 2055 */ 2056 OM_uint32 minor; 2057 gss_buffer_desc oidStr; 2058 2059 if (gss_oid_to_str(&minor, oid, &oidStr) != GSS_S_COMPLETE) 2060 return (NULL); 2061 2062 return ((char *)oidStr.value); 2063 } /* gss_oid2str */ 2064 2065 static void 2066 instructs() 2067 { 2068 fprintf(stderr, 2069 gettext( 2070 "\nThis program must be run as root. Root must be installed on the KDC\n" 2071 "and exist in srvtab as root/<hostname>, where <hostname> is the machine on\n" 2072 "which the test runs. Before running gssdtest for Kerberos mechanism, the\n" 2073 "operator running as root must kinit as some other principal, e.g., test.\n" 2074 "There are two mechanisms avaialble: dummy and Kerberos(default).\n" 2075 "The OID for dummy mechanism is 1.3.6.1.4.1.42.2.26.1.2.\n" 2076 "The OID for Kerberos mechanism is 1.2.840.113554.1.2.2.\n" 2077 "The order of context establishment calls is important. First, acquire must" 2078 "\nbe called. This obtains the credentials used by accept. Acquire need\n" 2079 "only be called once, since the credentials it returns are used each time\n" 2080 "accept is called. Then init is called, followed by accept. Calling init\n" 2081 "twice without calling accept or calling these in a different order gives\n" 2082 "erroneous results and will cause memory leaks in the gssapi daemon. \n" 2083 "Finally, after calling init and accept, init must be called again to\n" 2084 "finish context establishment. So an example sequence (with data valid for\n" 2085 "the Kerberos mechanism and running on the machine \"elrond\" in the realm\n" 2086 "FOO.BAR.SUN.COM is :\n")); 2087 fprintf(stderr, 2088 gettext("\nacquire service@host 1.2.840.113554.1.2.2\n" 2089 "init service@host 1.2.840.113554.1.2.2\n" 2090 "accept\ninit service@host 1.2.840.113554.1.2.2\n" 2091 "\nAfter a context is established, sign, seal,\n" 2092 "verify and unseal may be called. Here are some examples\n" 2093 "for these routines : \n\n" 2094 "sign initiator ThisTestMessageIsForSigning\n" 2095 "verify acceptor\nseal initiator ThisTestMessageIsForSealing\n" 2096 "unseal acceptor\n\nEach input line is terminated by <cr>.\n" 2097 "The program is terminated by cntl-d\nor the command \"exit\"" 2098 "\nfrom the prompt\n\n")); 2099 } 2100 2101 static void 2102 usage() 2103 { 2104 fprintf(stderr, 2105 gettext( 2106 "\nusage:\t[acquire | gss_acquire_cred]" 2107 "desired_name mech_type\n" 2108 "\t[release | gss_release_cred]\n" 2109 "\t[init | gss_init_sec_context] target_name mech_type\n" 2110 "\t[accept | gss_accept_sec_context]\n" 2111 "\t[process | gss_process_context_token] initiator | acceptor\n" 2112 "\t[delete | gss_delete_sec_context] initiator | acceptor\n" 2113 "\t[time | gss_context_time] {not yet implemented}\n" 2114 "\t[sign | gss_sign] initiator | acceptor message-to-sign\n" 2115 "\t[verify | gss_verify] initiator | acceptor\n" 2116 "\t[seal | gss_seal] initiator | acceptor message-to-seal\n" 2117 "\t[unseal | gss_unseal] initiator | acceptor\n" 2118 "\t[status | gss_display_status] mech_type [major | minor] \n" 2119 "\t[indicate | gss_indicate_mechs]\n" 2120 "\t[inquire | gss_inquire_cred] {not yet implemented}\n" 2121 "\t[expname2unixcred | gsscred_expname_to_unix_cred]" 2122 " export-name\n" 2123 "\t[name2unixcred | gsscred_name_to_unix_cred] " 2124 "pname [name_type mech_type]\n" 2125 "\t[grpinfo | gss_get_group_info] uid\n" 2126 "\t[gss_all | all] desired_name\n" 2127 "\t[gss_loop | loop] desired_name\n" 2128 "\texit\n\n")); 2129 } 2130 2131 /* Copied from parse_argv(), then modified */ 2132 2133 static int 2134 parse_input_line(input_line, argc, argv) 2135 char *input_line; 2136 int * argc; 2137 char ***argv; 2138 { 2139 const char nil = '\0'; 2140 char * chptr; 2141 int chr_cnt; 2142 int arg_cnt = 0; 2143 int ch_was_space = 1; 2144 int ch_is_space; 2145 2146 chr_cnt = strlen(input_line); 2147 2148 /* Count the arguments in the input_line string */ 2149 2150 *argc = 1; 2151 2152 for (chptr = &input_line[0]; *chptr != nil; chptr++) { 2153 ch_is_space = isspace(*chptr); 2154 if (ch_is_space && !ch_was_space) { 2155 (*argc)++; 2156 } 2157 ch_was_space = ch_is_space; 2158 } 2159 2160 if (ch_was_space) { 2161 (*argc)--; 2162 } /* minus trailing spaces */ 2163 2164 /* Now that we know how many args calloc the argv array */ 2165 2166 *argv = (char **) CALLOC((*argc)+1, sizeof (char *)); 2167 chptr = (char *) (&input_line[0]); 2168 2169 for (ch_was_space = 1; *chptr != nil; chptr++) { 2170 ch_is_space = isspace(*chptr); 2171 if (ch_is_space) { 2172 *chptr = nil; /* replace each space with nil */ 2173 } else if (ch_was_space) { /* begining of word? */ 2174 (*argv)[arg_cnt++] = chptr; /* new argument ? */ 2175 } 2176 2177 ch_was_space = ch_is_space; 2178 } 2179 2180 return (chr_cnt); 2181 } 2182