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 (the "License").
6 * You may not use this file except in compliance with the License.
7 *
8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9 * or http://www.opensolaris.org/os/licensing.
10 * See the License for the specific language governing permissions
11 * and limitations under the License.
12 *
13 * When distributing Covered Code, include this CDDL HEADER in each
14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15 * If applicable, add the following below this CDDL HEADER, with the
16 * fields enclosed by brackets "[]" replaced with your own identifying
17 * information: Portions Copyright [yyyy] [name of copyright owner]
18 *
19 * CDDL HEADER END
20 */
21
22/*
23 * Copyright 2010 Sun Microsystems, Inc.  All rights reserved.
24 * Use is subject to license terms.
25 */
26
27/*
28 * String conversion routine for hardware capabilities types.
29 */
30#include	<strings.h>
31#include	<stdio.h>
32#include	<_machelf.h>
33#include	<elfcap.h>
34#include	"cap_msg.h"
35#include	"_conv.h"
36
37static int
38conv_cap(Xword val, char *str, size_t len, Half mach,
39    Conv_fmt_flags_t fmt_flags, elfcap_to_str_func_t *fptr)
40{
41	size_t	_len;
42	int do_bkt = (fmt_flags & CONV_FMT_NOBKT) == 0;
43
44	/*
45	 * Note that for the purposes of this routine, I consider
46	 * CONV_FMT_NOBKT to mean no brackets, or anything that
47	 * is placed outside of them. We also drop the hex version
48	 * of the flags that are put in front of the opening bracket.
49	 */
50	if (do_bkt) {
51		_len = sprintf(str, MSG_ORIG(MSG_GBL_OSQBRKT), EC_XWORD(val));
52
53		len -= _len;
54		str += _len;
55	}
56
57	if ((*fptr)(ELFCAP_STYLE_UC, val, str, len, ELFCAP_FMT_SNGSPACE,
58	    mach) != 0)
59		return (0);
60
61	if (do_bkt) {
62		_len = strlen(str);
63		if ((len - _len) >= MSG_GBL_CSQBRKT_SIZE) {
64			str += _len;
65			(void) strcpy(str, MSG_ORIG(MSG_GBL_CSQBRKT));
66		}
67	}
68	return (1);
69}
70
71const char *
72conv_cap_val_hw1(Xword val, Half mach, Conv_fmt_flags_t fmt_flags,
73    Conv_cap_val_hw1_buf_t *cap_val_hw1_buf)
74{
75	if (val == 0)
76		return (MSG_ORIG(MSG_GBL_ZERO));
77
78	if (conv_cap(val, cap_val_hw1_buf->buf, sizeof (cap_val_hw1_buf->buf),
79	    mach, fmt_flags, elfcap_hw1_to_str) == 0)
80		return (conv_invalid_val(&cap_val_hw1_buf->inv_buf, val, 0));
81	return ((const char *)cap_val_hw1_buf->buf);
82}
83
84const char *
85conv_cap_val_hw2(Xword val, Half mach, Conv_fmt_flags_t fmt_flags,
86    Conv_cap_val_hw2_buf_t *cap_val_hw2_buf)
87{
88	if (val == 0)
89		return (MSG_ORIG(MSG_GBL_ZERO));
90
91	if (conv_cap(val, cap_val_hw2_buf->buf, sizeof (cap_val_hw2_buf->buf),
92	    mach, fmt_flags, elfcap_hw2_to_str) == 0)
93		return (conv_invalid_val(&cap_val_hw2_buf->inv_buf, val, 0));
94	return ((const char *)cap_val_hw2_buf->buf);
95}
96
97const char *
98conv_cap_val_sf1(Xword val, Half mach, Conv_fmt_flags_t fmt_flags,
99    Conv_cap_val_sf1_buf_t *cap_val_sf1_buf)
100{
101	if (val == 0)
102		return (MSG_ORIG(MSG_GBL_ZERO));
103
104	if (conv_cap(val, cap_val_sf1_buf->buf, sizeof (cap_val_sf1_buf->buf),
105	    mach, fmt_flags, elfcap_sf1_to_str) == 0)
106		return (conv_invalid_val(&cap_val_sf1_buf->inv_buf, val, 0));
107	return ((const char *)cap_val_sf1_buf->buf);
108}
109
110const char *
111conv_cap_tag(Xword tag, Conv_fmt_flags_t fmt_flags, Conv_inv_buf_t *inv_buf)
112{
113#ifdef _ELF64
114	/*
115	 * Valid tags all fit in 32-bits, so a value larger than that
116	 * is garbage. conv_map_ds() sees 32-bit values, so test for garbage
117	 * here before passing it on.
118	 *
119	 * Since there are no valid tags with a value > 32-bits, there
120	 * is no reason to expend effort decoding the low order bits.
121	 */
122	if (tag & 0xffffffff00000000)
123		return (conv_invalid_val(inv_buf, tag, fmt_flags));
124#endif
125
126	return (conv_map_ds(ELFOSABI_NONE, EM_NONE, tag,
127	    conv_cap_tag_strings(fmt_flags), fmt_flags, inv_buf));
128}
129
130const char *
131conv_cap_val(Xword tag, Xword val, Half mach, Conv_fmt_flags_t fmt_flags,
132    Conv_cap_val_buf_t *cap_val_buf)
133{
134	switch (tag) {
135	case CA_SUNW_HW_1:
136		return (conv_cap_val_hw1(val, mach, fmt_flags,
137		    &cap_val_buf->cap_val_hw1_buf));
138
139	case CA_SUNW_SF_1:
140		return (conv_cap_val_sf1(val, mach, fmt_flags,
141		    &cap_val_buf->cap_val_sf1_buf));
142
143	case CA_SUNW_HW_2:
144		return (conv_cap_val_hw2(val, mach, fmt_flags,
145		    &cap_val_buf->cap_val_hw2_buf));
146
147	default:
148		return (conv_invalid_val(&cap_val_buf->inv_buf, val, 0));
149	}
150}
151