/*
 * CDDL HEADER START
 *
 * The contents of this file are subject to the terms of the
 * Common Development and Distribution License (the "License").
 * You may not use this file except in compliance with the License.
 *
 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
 * or http://www.opensolaris.org/os/licensing.
 * See the License for the specific language governing permissions
 * and limitations under the License.
 *
 * When distributing Covered Code, include this CDDL HEADER in each
 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
 * If applicable, add the following below this CDDL HEADER, with the
 * fields enclosed by brackets "[]" replaced with your own identifying
 * information: Portions Copyright [yyyy] [name of copyright owner]
 *
 * CDDL HEADER END
 */

/*
 * Copyright 2007 Sun Microsystems, Inc.  All rights reserved.
 * Use is subject to license terms.
 */
#pragma ident	"%Z%%M%	%I%	%E% SMI"

#include	<strings.h>
#include	<dwarf.h>
#include	"_conv.h"
#include	<dwarf_ehe_msg.h>

#define	FLAGSZ	MSG_GBL_OSQBRKT_SIZE + \
		MSG_DWEHE_SLEB128_SIZE + \
		MSG_DWEHE_INDIRECT_SIZE + \
		CONV_INV_BUFSIZE + MSG_GBL_CSQBRKT_SIZE

/*
 * Ensure that Conv_dwarf_ehe_buf_t is large enough:
 *
 * FLAGSZ is the real minimum size of the buffer required by conv_dwarf_ehe().
 * However, Conv_dwarf_ehe_buf_t uses CONV_EHDR_FLAG_BUFSIZE to set the
 * buffer size. We do things this way because the definition of FLAGSZ uses
 * information that is not available in the environment of other programs
 * that include the conv.h header file.
 */
#if (CONV_DWARF_EHE_BUFSIZE != FLAGSZ) && !defined(__lint)
#define	REPORT_BUFSIZE FLAGSZ
#include "report_bufsize.h"
#error "CONV_DWARF_EHE_BUFSIZE does not match FLAGSZ"
#endif

const char *
conv_dwarf_ehe(uint_t flags, Conv_dwarf_ehe_buf_t *dwarf_ehe_buf)
{
	char	*buf = dwarf_ehe_buf->buf;
	size_t	ret = 0;

	(void) strncpy(buf, MSG_ORIG(MSG_GBL_OSQBRKT), FLAGSZ);

	if (flags == DW_EH_PE_omit)
		ret = strlcat(buf, MSG_ORIG(MSG_DWEHE_OMIT), FLAGSZ);
	else if (flags == DW_EH_PE_absptr)
		ret = strlcat(buf, MSG_ORIG(MSG_DWEHE_ABSPTR), FLAGSZ);

	if (ret >= FLAGSZ)
		return (conv_invalid_val(&dwarf_ehe_buf->inv_buf, flags, 0));

	if ((flags == DW_EH_PE_omit) || (flags == DW_EH_PE_absptr)) {
		(void) strlcat(buf, MSG_ORIG(MSG_GBL_CSQBRKT), FLAGSZ);
		return (buf);
	}

	switch (flags & 0x0f) {
	case DW_EH_PE_uleb128:
		ret = strlcat(buf, MSG_ORIG(MSG_DWEHE_ULEB128), FLAGSZ);
		break;
	case DW_EH_PE_udata2:
		ret = strlcat(buf, MSG_ORIG(MSG_DWEHE_UDATA2), FLAGSZ);
		break;
	case DW_EH_PE_udata4:
		ret = strlcat(buf, MSG_ORIG(MSG_DWEHE_UDATA4), FLAGSZ);
		break;
	case DW_EH_PE_udata8:
		ret = strlcat(buf, MSG_ORIG(MSG_DWEHE_UDATA8), FLAGSZ);
		break;
	case DW_EH_PE_sleb128:
		ret = strlcat(buf, MSG_ORIG(MSG_DWEHE_SLEB128), FLAGSZ);
		break;
	case DW_EH_PE_sdata2:
		ret = strlcat(buf, MSG_ORIG(MSG_DWEHE_SDATA2), FLAGSZ);
		break;
	case DW_EH_PE_sdata4:
		ret = strlcat(buf, MSG_ORIG(MSG_DWEHE_SDATA4), FLAGSZ);
		break;
	case DW_EH_PE_sdata8:
		ret = strlcat(buf, MSG_ORIG(MSG_DWEHE_SDATA8), FLAGSZ);
		break;
	}
	if (ret >= FLAGSZ)
		return (conv_invalid_val(&dwarf_ehe_buf->inv_buf, flags, 0));

	switch (flags & 0xf0) {
	case DW_EH_PE_pcrel:
		ret = strlcat(buf, MSG_ORIG(MSG_DWEHE_PCREL), FLAGSZ);
		break;
	case DW_EH_PE_textrel:
		ret = strlcat(buf, MSG_ORIG(MSG_DWEHE_TEXTREL), FLAGSZ);
		break;
	case DW_EH_PE_datarel:
		ret = strlcat(buf, MSG_ORIG(MSG_DWEHE_DATAREL), FLAGSZ);
		break;
	case DW_EH_PE_funcrel:
		ret = strlcat(buf, MSG_ORIG(MSG_DWEHE_FUNCREL), FLAGSZ);
		break;
	case DW_EH_PE_aligned:
		ret = strlcat(buf, MSG_ORIG(MSG_DWEHE_ALIGNED), FLAGSZ);
		break;
	case DW_EH_PE_indirect:
		ret = strlcat(buf, MSG_ORIG(MSG_DWEHE_INDIRECT), FLAGSZ);
		break;
	}
	if (ret >= FLAGSZ)
		return (conv_invalid_val(&dwarf_ehe_buf->inv_buf, flags, 0));

	(void) strlcat(buf, MSG_ORIG(MSG_GBL_CSQBRKT), FLAGSZ);
	return (buf);
}