2N/A/*
2N/A * CDDL HEADER START
2N/A *
2N/A * The contents of this file are subject to the terms of the
2N/A * Common Development and Distribution License (the "License").
2N/A * You may not use this file except in compliance with the License.
2N/A *
2N/A * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
2N/A * or http://www.opensolaris.org/os/licensing.
2N/A * See the License for the specific language governing permissions
2N/A * and limitations under the License.
2N/A *
2N/A * When distributing Covered Code, include this CDDL HEADER in each
2N/A * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
2N/A * If applicable, add the following below this CDDL HEADER, with the
2N/A * fields enclosed by brackets "[]" replaced with your own identifying
2N/A * information: Portions Copyright [yyyy] [name of copyright owner]
2N/A *
2N/A * CDDL HEADER END
2N/A */
2N/A
2N/A/*
2N/A * Copyright (c) 2009, 2012, Oracle and/or its affiliates. All rights reserved.
2N/A */
2N/A
2N/A#include <sys/types.h>
2N/A#include <sys/wait.h>
2N/A
2N/A#include <sys/fm/protocol.h>
2N/A#include <fm/fmd_msg.h>
2N/A
2N/A#include <unistd.h>
2N/A#include <signal.h>
2N/A#include <strings.h>
2N/A#include <stdlib.h>
2N/A#include <stdio.h>
2N/A#include <errno.h>
2N/A
2N/A#define TEST_ARR_SZ 2
2N/A
2N/Aint
2N/Amain(int argc, char *argv[])
2N/A{
2N/A fmd_msg_hdl_t *h;
2N/A pid_t pid;
2N/A int i, err = 0;
2N/A char *s;
2N/A
2N/A nvlist_t *fmri, *list, *test_arr[TEST_ARR_SZ];
2N/A nvlist_t *auth;
2N/A const char *code = "TEST-8000-08";
2N/A int64_t tod[] = { 0x9400000, 0 };
2N/A
2N/A if (argc > 1) {
2N/A (void) fprintf(stderr, "Usage: %s\n", argv[0]);
2N/A return (2);
2N/A }
2N/A
2N/A /*
2N/A * Build up a valid list.suspect event for a fictional diagnosis
2N/A * using a diagnosis code from our test dictionary so we can format
2N/A * messages.
2N/A */
2N/A if (nvlist_alloc(&auth, NV_UNIQUE_NAME, 0) != 0 ||
2N/A nvlist_alloc(&fmri, NV_UNIQUE_NAME, 0) != 0 ||
2N/A nvlist_alloc(&list, NV_UNIQUE_NAME, 0) != 0) {
2N/A (void) fprintf(stderr, "%s: nvlist_alloc failed\n", argv[0]);
2N/A return (1);
2N/A }
2N/A
2N/A /* system authority */
2N/A err |= nvlist_add_uint8(auth, FM_VERSION, FM_FMRI_AUTH_VERSION);
2N/A err |= nvlist_add_string(auth, FM_FMRI_AUTH_V1_SYSTEM_MFG,
2N/A "system_mfg");
2N/A err |= nvlist_add_string(auth, FM_FMRI_AUTH_V1_SYSTEM_NM,
2N/A "system_name");
2N/A err |= nvlist_add_string(auth, FM_FMRI_AUTH_V1_SYSTEM_PN,
2N/A "system_pn");
2N/A err |= nvlist_add_string(auth, FM_FMRI_AUTH_V1_SYSTEM_SN,
2N/A "system_sn");
2N/A
2N/A /* component system authority */
2N/A err |= nvlist_add_string(auth, FM_FMRI_AUTH_V1_SYS_COMP_MFG,
2N/A "sys_comp_mfg");
2N/A err |= nvlist_add_string(auth, FM_FMRI_AUTH_V1_SYS_COMP_NM,
2N/A "sys_comp_name");
2N/A err |= nvlist_add_string(auth, FM_FMRI_AUTH_V1_SYS_COMP_PN,
2N/A "sys_comp_pn");
2N/A err |= nvlist_add_string(auth, FM_FMRI_AUTH_V1_SYS_COMP_SN,
2N/A "sys_comp_sn");
2N/A
2N/A if (err != 0) {
2N/A (void) fprintf(stderr, "%s: failed to build auth nvlist: %s\n",
2N/A argv[0], strerror(err));
2N/A return (1);
2N/A }
2N/A
2N/A err |= nvlist_add_uint8(fmri, FM_VERSION, FM_FMD_SCHEME_VERSION);
2N/A err |= nvlist_add_string(fmri, FM_FMRI_SCHEME, FM_FMRI_SCHEME_FMD);
2N/A err |= nvlist_add_nvlist(fmri, FM_FMRI_AUTHORITY, auth);
2N/A err |= nvlist_add_string(fmri, FM_FMRI_FMD_NAME, "fmd_msg_test");
2N/A err |= nvlist_add_string(fmri, FM_FMRI_FMD_VERSION, "1.0");
2N/A
2N/A if (err != 0) {
2N/A (void) fprintf(stderr, "%s: failed to build fmri nvlist: %s\n",
2N/A argv[0], strerror(err));
2N/A return (1);
2N/A }
2N/A
2N/A err |= nvlist_add_uint8(list, FM_VERSION, FM_SUSPECT_VERSION);
2N/A err |= nvlist_add_string(list, FM_CLASS, FM_LIST_SUSPECT_CLASS);
2N/A err |= nvlist_add_string(list, FM_SUSPECT_UUID, "12345678");
2N/A err |= nvlist_add_string(list, FM_SUSPECT_DIAG_CODE, code);
2N/A err |= nvlist_add_int64_array(list, FM_SUSPECT_DIAG_TIME, tod, 2);
2N/A err |= nvlist_add_nvlist(list, FM_SUSPECT_DE, fmri);
2N/A err |= nvlist_add_uint32(list, FM_SUSPECT_FAULT_SZ, 0);
2N/A
2N/A /*
2N/A * Add a contrived nvlist array to our list.suspect so that we can
2N/A * exercise the expansion syntax for dereferencing nvlist array members
2N/A */
2N/A for (i = 0; i < TEST_ARR_SZ; i++) {
2N/A if (nvlist_alloc(&test_arr[i], NV_UNIQUE_NAME, 0) != 0) {
2N/A (void) fprintf(stderr, "%s: failed to alloc nvlist "
2N/A "array: %s\n", argv[0], strerror(err));
2N/A return (1);
2N/A }
2N/A err |= nvlist_add_uint8(test_arr[i], "index", i);
2N/A }
2N/A err |= nvlist_add_nvlist_array(list, "test_arr", test_arr, TEST_ARR_SZ);
2N/A
2N/A if (err != 0) {
2N/A (void) fprintf(stderr, "%s: failed to build list nvlist: %s\n",
2N/A argv[0], strerror(err));
2N/A return (1);
2N/A }
2N/A
2N/A /*
2N/A * Now initialize the libfmd_msg library for testing, using the message
2N/A * catalogs found in the proto area of the current workspace.
2N/A */
2N/A if ((h = fmd_msg_init(getenv("ROOT"), FMD_MSG_VERSION)) == NULL) {
2N/A (void) fprintf(stderr, "%s: fmd_msg_init failed: %s\n",
2N/A argv[0], strerror(errno));
2N/A return (1);
2N/A }
2N/A
2N/A /*
2N/A * Test 0: Verify that both fmd_msg_getitem_id and fmd_msg_gettext_id
2N/A * return NULL and EINVAL for an illegal message code, and NULL
2N/A * and ENOENT for a valid but not defined message code.
2N/A */
2N/A s = fmd_msg_getitem_id(h, NULL, "I_AM_NOT_VALID", 0);
2N/A if (s != NULL || errno != EINVAL) {
2N/A (void) fprintf(stderr, "%s: test0 FAIL: illegal code returned "
2N/A "s = %p, errno = %d\n", argv[0], (void *)s, errno);
2N/A return (1);
2N/A }
2N/A
2N/A s = fmd_msg_gettext_id(h, NULL, "I_AM_NOT_VALID");
2N/A if (s != NULL || errno != EINVAL) {
2N/A (void) fprintf(stderr, "%s: test0 FAIL: illegal code returned "
2N/A "s = %p, errno = %d\n", argv[0], (void *)s, errno);
2N/A return (1);
2N/A }
2N/A
2N/A s = fmd_msg_getitem_id(h, NULL, "I_AM_NOT_HERE-0000-0000", 0);
2N/A if (s != NULL || errno != ENOENT) {
2N/A (void) fprintf(stderr, "%s: test0 FAIL: missing code returned "
2N/A "s = %p, errno = %d\n", argv[0], (void *)s, errno);
2N/A return (1);
2N/A }
2N/A
2N/A s = fmd_msg_gettext_id(h, NULL, "I_AM_NOT_HERE-0000-0000");
2N/A if (s != NULL || errno != ENOENT) {
2N/A (void) fprintf(stderr, "%s: test0 FAIL: missing code returned "
2N/A "s = %p, errno = %d\n", argv[0], (void *)s, errno);
2N/A return (1);
2N/A }
2N/A
2N/A /*
2N/A * Test 1: Use fmd_msg_getitem_id to retrieve the item strings for
2N/A * a known message code without having any actual event handle.
2N/A */
2N/A for (i = 0; i < FMD_MSG_ITEM_MAX; i++) {
2N/A if ((s = fmd_msg_getitem_id(h, NULL, code, i)) == NULL) {
2N/A (void) fprintf(stderr, "%s: fmd_msg_getitem_id failed "
2N/A "for %s, item %d: %s\n",
2N/A argv[0], code, i, strerror(errno));
2N/A }
2N/A
2N/A (void) printf("code %s item %d = <<%s>>\n", code, i, s);
2N/A free(s);
2N/A }
2N/A
2N/A /*
2N/A * Test 2: Use fmd_msg_gettext_id to retrieve the complete message for
2N/A * a known message code without having any actual event handle.
2N/A */
2N/A if ((s = fmd_msg_gettext_id(h, NULL, code)) == NULL) {
2N/A (void) fprintf(stderr, "%s: fmd_msg_gettext_id failed for %s: "
2N/A "%s\n", argv[0], code, strerror(errno));
2N/A return (1);
2N/A }
2N/A
2N/A (void) printf("%s\n", s);
2N/A free(s);
2N/A
2N/A /*
2N/A * Test 3: Use fmd_msg_getitem_nv to retrieve the item strings for
2N/A * our list.suspect event handle.
2N/A */
2N/A for (i = 0; i < FMD_MSG_ITEM_MAX; i++) {
2N/A if ((s = fmd_msg_getitem_nv(h, NULL, list, i)) == NULL) {
2N/A (void) fprintf(stderr, "%s: fmd_msg_getitem_nv failed "
2N/A "for %s, item %d: %s\n",
2N/A argv[0], code, i, strerror(errno));
2N/A }
2N/A
2N/A (void) printf("code %s item %d = <<%s>>\n", code, i, s);
2N/A free(s);
2N/A }
2N/A
2N/A /*
2N/A * Test 4: Use fmd_msg_getitem_nv to retrieve the complete message for
2N/A * a known message code using our list.suspect event handle.
2N/A */
2N/A if ((s = fmd_msg_gettext_nv(h, NULL, list)) == NULL) {
2N/A (void) fprintf(stderr, "%s: fmd_msg_gettext_nv failed for %s: "
2N/A "%s\n", argv[0], code, strerror(errno));
2N/A return (1);
2N/A }
2N/A
2N/A (void) printf("%s\n", s);
2N/A free(s);
2N/A
2N/A /*
2N/A * Test 5: Use fmd_msg_getitem_nv to retrieve the complete message for
2N/A * a known message code using our list.suspect event handle, but this
2N/A * time set the URL to our own customized URL. Our contrived message
2N/A * has been designed to exercise the key aspects of the variable
2N/A * expansion syntax.
2N/A */
2N/A if (fmd_msg_url_set(h, "http://foo.bar.com/") != 0) {
2N/A (void) fprintf(stderr, "%s: fmd_msg_url_set failed: %s\n",
2N/A argv[0], strerror(errno));
2N/A }
2N/A
2N/A if ((s = fmd_msg_gettext_nv(h, NULL, list)) == NULL) {
2N/A (void) fprintf(stderr, "%s: fmd_msg_gettext_nv failed for %s: "
2N/A "%s\n", argv[0], code, strerror(errno));
2N/A return (1);
2N/A }
2N/A
2N/A (void) printf("%s\n", s);
2N/A free(s);
2N/A
2N/A for (i = 0; i < TEST_ARR_SZ; i++)
2N/A nvlist_free(test_arr[i]);
2N/A nvlist_free(fmri);
2N/A nvlist_free(auth);
2N/A nvlist_free(list);
2N/A
2N/A fmd_msg_fini(h); /* free library state before dumping core */
2N/A pid = fork(); /* fork into background to not bother make(1) */
2N/A
2N/A switch (pid) {
2N/A case -1:
2N/A (void) fprintf(stderr, "FAIL (failed to fork)\n");
2N/A return (1);
2N/A case 0:
2N/A abort();
2N/A return (1);
2N/A }
2N/A
2N/A if (waitpid(pid, &err, 0) == -1) {
2N/A (void) fprintf(stderr, "FAIL (failed to wait for %d: %s)\n",
2N/A (int)pid, strerror(errno));
2N/A return (1);
2N/A }
2N/A
2N/A if (WIFSIGNALED(err) == 0 || WTERMSIG(err) != SIGABRT) {
2N/A (void) fprintf(stderr, "FAIL (child did not SIGABRT)\n");
2N/A return (1);
2N/A }
2N/A
2N/A if (!WCOREDUMP(err)) {
2N/A (void) fprintf(stderr, "FAIL (no core generated)\n");
2N/A return (1);
2N/A }
2N/A
2N/A (void) fprintf(stderr, "done\n");
2N/A return (0);
2N/A}