1*8462SApril.Chin@Sun.COM#!/usr/bin/ksh93
2*8462SApril.Chin@Sun.COM
3*8462SApril.Chin@Sun.COM#
4*8462SApril.Chin@Sun.COM# CDDL HEADER START
5*8462SApril.Chin@Sun.COM#
6*8462SApril.Chin@Sun.COM# The contents of this file are subject to the terms of the
7*8462SApril.Chin@Sun.COM# Common Development and Distribution License (the "License").
8*8462SApril.Chin@Sun.COM# You may not use this file except in compliance with the License.
9*8462SApril.Chin@Sun.COM#
10*8462SApril.Chin@Sun.COM# You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
11*8462SApril.Chin@Sun.COM# or http://www.opensolaris.org/os/licensing.
12*8462SApril.Chin@Sun.COM# See the License for the specific language governing permissions
13*8462SApril.Chin@Sun.COM# and limitations under the License.
14*8462SApril.Chin@Sun.COM#
15*8462SApril.Chin@Sun.COM# When distributing Covered Code, include this CDDL HEADER in each
16*8462SApril.Chin@Sun.COM# file and include the License file at usr/src/OPENSOLARIS.LICENSE.
17*8462SApril.Chin@Sun.COM# If applicable, add the following below this CDDL HEADER, with the
18*8462SApril.Chin@Sun.COM# fields enclosed by brackets "[]" replaced with your own identifying
19*8462SApril.Chin@Sun.COM# information: Portions Copyright [yyyy] [name of copyright owner]
20*8462SApril.Chin@Sun.COM#
21*8462SApril.Chin@Sun.COM# CDDL HEADER END
22*8462SApril.Chin@Sun.COM#
23*8462SApril.Chin@Sun.COM
24*8462SApril.Chin@Sun.COM#
25*8462SApril.Chin@Sun.COM# Copyright 2008 Sun Microsystems, Inc.  All rights reserved.
26*8462SApril.Chin@Sun.COM# Use is subject to license terms.
27*8462SApril.Chin@Sun.COM#
28*8462SApril.Chin@Sun.COM
29*8462SApril.Chin@Sun.COM#
30*8462SApril.Chin@Sun.COM# shcalc - small shell-based calculator
31*8462SApril.Chin@Sun.COM#
32*8462SApril.Chin@Sun.COM
33*8462SApril.Chin@Sun.COM# Solaris needs /usr/xpg6/bin:/usr/xpg4/bin because the tools in /usr/bin are not POSIX-conformant
34*8462SApril.Chin@Sun.COMexport PATH=/usr/xpg6/bin:/usr/xpg4/bin:/bin:/usr/bin
35*8462SApril.Chin@Sun.COM
36*8462SApril.Chin@Sun.COM# Make sure all math stuff runs in the "C" locale to avoid problems
37*8462SApril.Chin@Sun.COM# with alternative # radix point representations (e.g. ',' instead of
38*8462SApril.Chin@Sun.COM# '.' in de_DE.*-locales). This needs to be set _before_ any
39*8462SApril.Chin@Sun.COM# floating-point constants are defined in this script).
40*8462SApril.Chin@Sun.COMif [[ "${LC_ALL}" != "" ]] ; then
41*8462SApril.Chin@Sun.COM    export \
42*8462SApril.Chin@Sun.COM        LC_MONETARY="${LC_ALL}" \
43*8462SApril.Chin@Sun.COM        LC_MESSAGES="${LC_ALL}" \
44*8462SApril.Chin@Sun.COM        LC_COLLATE="${LC_ALL}" \
45*8462SApril.Chin@Sun.COM        LC_CTYPE="${LC_ALL}"
46*8462SApril.Chin@Sun.COM        unset LC_ALL
47*8462SApril.Chin@Sun.COMfi
48*8462SApril.Chin@Sun.COMexport LC_NUMERIC=C
49*8462SApril.Chin@Sun.COM
50*8462SApril.Chin@Sun.COMfunction fatal_error
51*8462SApril.Chin@Sun.COM{
52*8462SApril.Chin@Sun.COM	print -u2 "${progname}: $*"
53*8462SApril.Chin@Sun.COM	exit 1
54*8462SApril.Chin@Sun.COM}
55*8462SApril.Chin@Sun.COM
56*8462SApril.Chin@Sun.COMfunction do_calculate
57*8462SApril.Chin@Sun.COM{
58*8462SApril.Chin@Sun.COM	typeset calcline="$1"
59*8462SApril.Chin@Sun.COM	float x=0.0
60*8462SApril.Chin@Sun.COM
61*8462SApril.Chin@Sun.COM	printf "(( x=( %s ) ))\n" "${calcline}" | source /dev/stdin
62*8462SApril.Chin@Sun.COM	if (( $? != 0 )) ; then
63*8462SApril.Chin@Sun.COM		print -f $"%s: Syntax error in %s\n" "${progname}" "${calcline}"
64*8462SApril.Chin@Sun.COM		return 1
65*8462SApril.Chin@Sun.COM	fi
66*8462SApril.Chin@Sun.COM
67*8462SApril.Chin@Sun.COM	printf "%s == %.40g\n" "${calcline}" x
68*8462SApril.Chin@Sun.COM
69*8462SApril.Chin@Sun.COM	return 0
70*8462SApril.Chin@Sun.COM}
71*8462SApril.Chin@Sun.COM
72*8462SApril.Chin@Sun.COMfunction usage
73*8462SApril.Chin@Sun.COM{
74*8462SApril.Chin@Sun.COM	OPTIND=0
75*8462SApril.Chin@Sun.COM	getopts -a "${progname}" "${shcalc_usage}" OPT '-?'
76*8462SApril.Chin@Sun.COM	exit 2
77*8462SApril.Chin@Sun.COM}
78*8462SApril.Chin@Sun.COM
79*8462SApril.Chin@Sun.COM# program start
80*8462SApril.Chin@Sun.COM# (be carefull with builtins here - they are unconditionally available
81*8462SApril.Chin@Sun.COM# in the shell's "restricted" mode)
82*8462SApril.Chin@Sun.COMbuiltin basename
83*8462SApril.Chin@Sun.COMbuiltin sum
84*8462SApril.Chin@Sun.COM
85*8462SApril.Chin@Sun.COMtypeset progname="${ basename "${0}" ; }"
86*8462SApril.Chin@Sun.COM
87*8462SApril.Chin@Sun.COMtypeset -r shcalc_usage=$'+
88*8462SApril.Chin@Sun.COM[-?\n@(#)\$Id: shcalc (Roland Mainz) 2008-11-03 \$\n]
89*8462SApril.Chin@Sun.COM[-author?Roland Mainz <roland.mainz@nrubsig.org>]
90*8462SApril.Chin@Sun.COM[+NAME?shcalc - simple shell calculator]
91*8462SApril.Chin@Sun.COM[+DESCRIPTION?\bsshcalc\b is a small calculator application which
92*8462SApril.Chin@Sun.COM	prints the results of ISO C99 math expressions read from either
93*8462SApril.Chin@Sun.COM	arguments or stdin if no arguments are given.]
94*8462SApril.Chin@Sun.COM[+SEE ALSO?\bksh93\b(1),\bceil\b(3M), \bcopysign\b(3M), \bcos\b(3M),
95*8462SApril.Chin@Sun.COM	\bcosh\b(3M), \berf\b(3M), \berfc\b(3M), \bexp\b(3M),
96*8462SApril.Chin@Sun.COM	\bexp2\b(3M), \bexpm1\b(3M), \bfabs abs\b(3M), \bfdim\b(3M),
97*8462SApril.Chin@Sun.COM	\bfinite\b(3M), \bfloor int\b(3M), \bfma\b(3M), \bfmax\b(3M), \bfmin\b(3M),
98*8462SApril.Chin@Sun.COM	\bfmod\b(3M), \bfpclassify\b(3M), \bhypot\b(3M), \bilogb\b(3M),
99*8462SApril.Chin@Sun.COM	\bisfinite\b(3M), \bisgreater\b(3M), \bisgreaterequal\b(3M), \bisinf\b(3M),
100*8462SApril.Chin@Sun.COM	\bisless\b(3M), \bislessequal\b(3M), \bislessgreater\b(3M), \bisnan\b(3M),
101*8462SApril.Chin@Sun.COM	\bisnormal\b(3M), \bissubnormal\b(3M), \bisunordered\b(3M), \biszero\b(3M),
102*8462SApril.Chin@Sun.COM	\blgamma\b(3M), \blog\b(3M), \blog1p\b(3M), \blog2\b(3M),
103*8462SApril.Chin@Sun.COM	\blogb\b(3M), \bnearbyint\b(3M), \bnextafter\b(3M), \bnexttoward\b(3M),
104*8462SApril.Chin@Sun.COM	\bpow\b(3M), \bremainder\b(3M), \brint\b(3M), \bround\b(3M),
105*8462SApril.Chin@Sun.COM	\bscalb\b(3M), \bscalbn\b(3M), \bsignbit\b(3M), \bsin\b(3M),
106*8462SApril.Chin@Sun.COM	\bsinh\b(3M), \bsqrt\b(3M), \btan\b(3M), \btanh\b(3M),
107*8462SApril.Chin@Sun.COM	\btgamma\b(3M), \btrunc\b(3M)]
108*8462SApril.Chin@Sun.COM'
109*8462SApril.Chin@Sun.COMwhile getopts -a "${progname}" "${shcalc_usage}" OPT ; do
110*8462SApril.Chin@Sun.COM#	printmsg "## OPT=|${OPT}|, OPTARG=|${OPTARG}|"
111*8462SApril.Chin@Sun.COM	case ${OPT} in
112*8462SApril.Chin@Sun.COM		*)	usage ;;
113*8462SApril.Chin@Sun.COM	esac
114*8462SApril.Chin@Sun.COMdone
115*8462SApril.Chin@Sun.COMshift $((OPTIND-1))
116*8462SApril.Chin@Sun.COM
117*8462SApril.Chin@Sun.COMinteger res
118*8462SApril.Chin@Sun.COM
119*8462SApril.Chin@Sun.COMif (( $# == 0 )) ; then
120*8462SApril.Chin@Sun.COM	# No arguments ? Switch to interactive mode...
121*8462SApril.Chin@Sun.COM
122*8462SApril.Chin@Sun.COM	# make sure "read" below uses "gmacs"-like editor keys and "multiline" mode
123*8462SApril.Chin@Sun.COM
124*8462SApril.Chin@Sun.COM	set -o gmacs
125*8462SApril.Chin@Sun.COM	set -o multiline
126*8462SApril.Chin@Sun.COM
127*8462SApril.Chin@Sun.COM	while read "calcline?calc> " ; do
128*8462SApril.Chin@Sun.COM		# quit ?
129*8462SApril.Chin@Sun.COM		[[ "${calcline}" == ~(Elri)(exit|quit|eof) ]] && break
130*8462SApril.Chin@Sun.COM
131*8462SApril.Chin@Sun.COM		# empty line ?
132*8462SApril.Chin@Sun.COM		[[ "${calcline}" == ~(Elri)([[:space:]]*) ]] && continue
133*8462SApril.Chin@Sun.COM
134*8462SApril.Chin@Sun.COM		do_calculate "$calcline"
135*8462SApril.Chin@Sun.COM		(( res=$? ))
136*8462SApril.Chin@Sun.COM	done
137*8462SApril.Chin@Sun.COM
138*8462SApril.Chin@Sun.COM	exit ${res}
139*8462SApril.Chin@Sun.COMelse
140*8462SApril.Chin@Sun.COM	while (( $# > 0 )) ; do
141*8462SApril.Chin@Sun.COM		do_calculate "$1"
142*8462SApril.Chin@Sun.COM		(( res=$? ))
143*8462SApril.Chin@Sun.COM		shift
144*8462SApril.Chin@Sun.COM	done
145*8462SApril.Chin@Sun.COM
146*8462SApril.Chin@Sun.COM	exit ${res}
147*8462SApril.Chin@Sun.COMfi
148*8462SApril.Chin@Sun.COM
149*8462SApril.Chin@Sun.COM# not reached
150*8462SApril.Chin@Sun.COM
151*8462SApril.Chin@Sun.COM# EOF.
152