| 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 |  * Copyright 2007 Sun Microsystems, Inc.  All rights reserved. | 
| 23 |  * Use is subject to license terms. | 
| 24 |  */ | 
| 25 | /* | 
| 26 |  * Copyright (c) 2013, 2016 by Delphix. All rights reserved. | 
| 27 |  * Copyright (c) 2013 Joyent, Inc. All rights reserved. | 
| 28 |  */ | 
| 29 |  | 
| 30 | #ifndef	_DT_PARSER_H | 
| 31 | #define	_DT_PARSER_H | 
| 32 |  | 
| 33 | #include <sys/types.h> | 
| 34 | #include <sys/dtrace.h> | 
| 35 |  | 
| 36 | #include <libctf.h> | 
| 37 | #include <stdarg.h> | 
| 38 | #include <stdio.h> | 
| 39 |  | 
| 40 | #ifdef	__cplusplus | 
| 41 | extern "C"  { | 
| 42 | #endif | 
| 43 |  | 
| 44 | #include <dt_errtags.h> | 
| 45 | #include <dt_ident.h> | 
| 46 | #include <dt_decl.h> | 
| 47 | #include <dt_xlator.h> | 
| 48 | #include <dt_list.h> | 
| 49 |  | 
| 50 | typedef struct dt_node { | 
| 51 | 	ctf_file_t *dn_ctfp;	/* CTF type container for node's type */ | 
| 52 | 	ctf_id_t dn_type;	/* CTF type reference for node's type */ | 
| 53 | 	uchar_t dn_kind;	/* node kind (DT_NODE_*, defined below) */ | 
| 54 | 	uchar_t dn_flags;	/* node flags (DT_NF_*, defined below) */ | 
| 55 | 	ushort_t dn_op;		/* operator (DT_TOK_*, defined by lex) */ | 
| 56 | 	int dn_line;		/* line number for error messages */ | 
| 57 | 	int dn_reg;		/* register allocated by cg */ | 
| 58 | 	dtrace_attribute_t dn_attr; /* node stability attributes */ | 
| 59 |  | 
| 60 | 	/* | 
| 61 | 	 * D compiler nodes, as is the usual style, contain a union of the | 
| 62 | 	 * different sub-elements required by the various kinds of nodes. | 
| 63 | 	 * These sub-elements are accessed using the macros defined below. | 
| 64 | 	 */ | 
| 65 | 	union { | 
| 66 | 		struct { | 
| 67 | 			uintmax_t _value;	/* integer value */ | 
| 68 | 			char *_string;		/* string value */ | 
| 69 | 		} _const; | 
| 70 |  | 
| 71 | 		struct { | 
| 72 | 			dt_ident_t *_ident;	/* identifier reference */ | 
| 73 | 			struct dt_node *_links[3]; /* child node pointers */ | 
| 74 | 		} _nodes; | 
| 75 |  | 
| 76 | 		struct { | 
| 77 | 			struct dt_node *_descs;	/* list of descriptions */ | 
| 78 | 			struct dt_node *_pred;	/* predicate expression */ | 
| 79 | 			struct dt_node *_acts;	/* action statement list */ | 
| 80 | 			dt_idhash_t *_locals;	/* local variable hash */ | 
| 81 | 			dtrace_attribute_t _attr; /* context attributes */ | 
| 82 | 		} _clause; | 
| 83 |  | 
| 84 | 		struct { | 
| 85 | 			char *_spec;		/* specifier string (if any) */ | 
| 86 | 			dtrace_probedesc_t *_desc; /* final probe description */ | 
| 87 | 		} _pdesc; | 
| 88 |  | 
| 89 | 		struct { | 
| 90 | 			char *_name;		/* string name of member */ | 
| 91 | 			struct dt_node *_expr;	/* expression node pointer */ | 
| 92 | 			dt_xlator_t *_xlator;	/* translator reference */ | 
| 93 | 			uint_t _id;		/* member identifier */ | 
| 94 | 		} _member; | 
| 95 |  | 
| 96 | 		struct { | 
| 97 | 			dt_xlator_t *_xlator;	/* translator reference */ | 
| 98 | 			struct dt_node *_xmemb;	/* individual xlator member */ | 
| 99 | 			struct dt_node *_membs;	/* list of member nodes */ | 
| 100 | 		} _xlator; | 
| 101 |  | 
| 102 | 		struct { | 
| 103 | 			char *_name;		/* string name of provider */ | 
| 104 | 			struct dt_provider *_pvp; /* provider references */ | 
| 105 | 			struct dt_node *_probes;  /* list of probe nodes */ | 
| 106 | 			int _redecl;		/* provider redeclared */ | 
| 107 | 		} _provider; | 
| 108 |  | 
| 109 | 		struct { | 
| 110 | 			struct dt_node *_conditional; | 
| 111 | 			struct dt_node *_body; | 
| 112 | 			struct dt_node *_alternate_body; | 
| 113 | 		} _conditional; | 
| 114 | 	} dn_u; | 
| 115 |  | 
| 116 | 	struct dt_node *dn_list; /* parse tree list link */ | 
| 117 | 	struct dt_node *dn_link; /* allocation list link */ | 
| 118 | } dt_node_t; | 
| 119 |  | 
| 120 | #define	dn_value	dn_u._const._value	/* DT_NODE_INT */ | 
| 121 | #define	dn_string	dn_u._const._string	/* STRING, IDENT, TYPE */ | 
| 122 | #define	dn_ident	dn_u._nodes._ident	/* VAR,SYM,FUN,AGG,INL,PROBE */ | 
| 123 | #define	dn_args		dn_u._nodes._links[0]	/* DT_NODE_VAR, FUNC */ | 
| 124 | #define	dn_child	dn_u._nodes._links[0]	/* DT_NODE_OP1 */ | 
| 125 | #define	dn_left		dn_u._nodes._links[0]	/* DT_NODE_OP2, OP3 */ | 
| 126 | #define	dn_right	dn_u._nodes._links[1]	/* DT_NODE_OP2, OP3 */ | 
| 127 | #define	dn_expr		dn_u._nodes._links[2]	/* DT_NODE_OP3, DEXPR */ | 
| 128 | #define	dn_aggfun	dn_u._nodes._links[0]	/* DT_NODE_AGG */ | 
| 129 | #define	dn_aggtup	dn_u._nodes._links[1]	/* DT_NODE_AGG */ | 
| 130 | #define	dn_pdescs	dn_u._clause._descs	/* DT_NODE_CLAUSE */ | 
| 131 | #define	dn_pred		dn_u._clause._pred	/* DT_NODE_CLAUSE */ | 
| 132 | #define	dn_acts		dn_u._clause._acts	/* DT_NODE_CLAUSE */ | 
| 133 | #define	dn_locals	dn_u._clause._locals	/* DT_NODE_CLAUSE */ | 
| 134 | #define	dn_ctxattr	dn_u._clause._attr	/* DT_NODE_CLAUSE */ | 
| 135 | #define	dn_spec		dn_u._pdesc._spec	/* DT_NODE_PDESC */ | 
| 136 | #define	dn_desc		dn_u._pdesc._desc	/* DT_NODE_PDESC */ | 
| 137 | #define	dn_membname	dn_u._member._name	/* DT_NODE_MEMBER */ | 
| 138 | #define	dn_membexpr	dn_u._member._expr	/* DT_NODE_MEMBER */ | 
| 139 | #define	dn_membxlator	dn_u._member._xlator	/* DT_NODE_MEMBER */ | 
| 140 | #define	dn_membid	dn_u._member._id	/* DT_NODE_MEMBER */ | 
| 141 | #define	dn_xlator	dn_u._xlator._xlator	/* DT_NODE_XLATOR */ | 
| 142 | #define	dn_xmember	dn_u._xlator._xmemb	/* DT_NODE_XLATOR */ | 
| 143 | #define	dn_members	dn_u._xlator._membs	/* DT_NODE_XLATOR */ | 
| 144 | #define	dn_provname	dn_u._provider._name	/* DT_NODE_PROVIDER */ | 
| 145 | #define	dn_provider	dn_u._provider._pvp	/* DT_NODE_PROVIDER */ | 
| 146 | #define	dn_provred	dn_u._provider._redecl	/* DT_NODE_PROVIDER */ | 
| 147 | #define	dn_probes	dn_u._provider._probes	/* DT_NODE_PROVIDER */ | 
| 148 |  | 
| 149 | /* DT_NODE_IF: */ | 
| 150 | #define	dn_conditional		dn_u._conditional._conditional | 
| 151 | #define	dn_body			dn_u._conditional._body | 
| 152 | #define	dn_alternate_body	dn_u._conditional._alternate_body | 
| 153 |  | 
| 154 | #define	DT_NODE_FREE	0	/* unused node (waiting to be freed) */ | 
| 155 | #define	DT_NODE_INT	1	/* integer value */ | 
| 156 | #define	DT_NODE_STRING	2	/* string value */ | 
| 157 | #define	DT_NODE_IDENT	3	/* identifier */ | 
| 158 | #define	DT_NODE_VAR	4	/* variable reference */ | 
| 159 | #define	DT_NODE_SYM	5	/* symbol reference */ | 
| 160 | #define	DT_NODE_TYPE	6	/* type reference or formal parameter */ | 
| 161 | #define	DT_NODE_FUNC	7	/* function call */ | 
| 162 | #define	DT_NODE_OP1	8	/* unary operator */ | 
| 163 | #define	DT_NODE_OP2	9	/* binary operator */ | 
| 164 | #define	DT_NODE_OP3	10	/* ternary operator */ | 
| 165 | #define	DT_NODE_DEXPR	11	/* D expression action */ | 
| 166 | #define	DT_NODE_DFUNC	12	/* D function action */ | 
| 167 | #define	DT_NODE_AGG	13	/* aggregation */ | 
| 168 | #define	DT_NODE_PDESC	14	/* probe description */ | 
| 169 | #define	DT_NODE_CLAUSE	15	/* clause definition */ | 
| 170 | #define	DT_NODE_INLINE	16	/* inline definition */ | 
| 171 | #define	DT_NODE_MEMBER	17	/* member definition */ | 
| 172 | #define	DT_NODE_XLATOR	18	/* translator definition */ | 
| 173 | #define	DT_NODE_PROBE	19	/* probe definition */ | 
| 174 | #define	DT_NODE_PROVIDER 20	/* provider definition */ | 
| 175 | #define	DT_NODE_PROG	21	/* program translation unit */ | 
| 176 | #define	DT_NODE_IF	22	/* if statement */ | 
| 177 |  | 
| 178 | #define	DT_NF_SIGNED	0x01	/* data is a signed quantity (else unsigned) */ | 
| 179 | #define	DT_NF_COOKED	0x02	/* data is a known type (else still cooking) */ | 
| 180 | #define	DT_NF_REF	0x04	/* pass by reference (array, struct, union) */ | 
| 181 | #define	DT_NF_LVALUE	0x08	/* node is an l-value according to ANSI-C */ | 
| 182 | #define	DT_NF_WRITABLE	0x10	/* node is writable (can be modified) */ | 
| 183 | #define	DT_NF_BITFIELD	0x20	/* node is an integer bitfield */ | 
| 184 | #define	DT_NF_USERLAND	0x40	/* data is a userland address */ | 
| 185 |  | 
| 186 | #define	DT_TYPE_NAMELEN	128	/* reasonable size for ctf_type_name() */ | 
| 187 |  | 
| 188 | extern int dt_node_is_integer(const dt_node_t *); | 
| 189 | extern int dt_node_is_float(const dt_node_t *); | 
| 190 | extern int dt_node_is_scalar(const dt_node_t *); | 
| 191 | extern int dt_node_is_arith(const dt_node_t *); | 
| 192 | extern int dt_node_is_vfptr(const dt_node_t *); | 
| 193 | extern int dt_node_is_dynamic(const dt_node_t *); | 
| 194 | extern int dt_node_is_stack(const dt_node_t *); | 
| 195 | extern int dt_node_is_symaddr(const dt_node_t *); | 
| 196 | extern int dt_node_is_usymaddr(const dt_node_t *); | 
| 197 | extern int dt_node_is_string(const dt_node_t *); | 
| 198 | extern int dt_node_is_strcompat(const dt_node_t *); | 
| 199 | extern int dt_node_is_pointer(const dt_node_t *); | 
| 200 | extern int dt_node_is_void(const dt_node_t *); | 
| 201 | extern int dt_node_is_ptrcompat(const dt_node_t *, const dt_node_t *, | 
| 202 | 	ctf_file_t **, ctf_id_t *); | 
| 203 | extern int dt_node_is_argcompat(const dt_node_t *, const dt_node_t *); | 
| 204 | extern int dt_node_is_posconst(const dt_node_t *); | 
| 205 | extern int dt_node_is_actfunc(const dt_node_t *); | 
| 206 |  | 
| 207 | extern dt_node_t *dt_node_int(uintmax_t); | 
| 208 | extern dt_node_t *dt_node_string(char *); | 
| 209 | extern dt_node_t *dt_node_ident(char *); | 
| 210 | extern dt_node_t *dt_node_type(dt_decl_t *); | 
| 211 | extern dt_node_t *dt_node_vatype(void); | 
| 212 | extern dt_node_t *dt_node_decl(void); | 
| 213 | extern dt_node_t *dt_node_func(dt_node_t *, dt_node_t *); | 
| 214 | extern dt_node_t *dt_node_offsetof(dt_decl_t *, char *); | 
| 215 | extern dt_node_t *dt_node_op1(int, dt_node_t *); | 
| 216 | extern dt_node_t *dt_node_op2(int, dt_node_t *, dt_node_t *); | 
| 217 | extern dt_node_t *dt_node_op3(dt_node_t *, dt_node_t *, dt_node_t *); | 
| 218 | extern dt_node_t *dt_node_statement(dt_node_t *); | 
| 219 | extern dt_node_t *dt_node_pdesc_by_name(char *); | 
| 220 | extern dt_node_t *dt_node_pdesc_by_id(uintmax_t); | 
| 221 | extern dt_node_t *dt_node_clause(dt_node_t *, dt_node_t *, dt_node_t *); | 
| 222 | extern dt_node_t *dt_node_inline(dt_node_t *); | 
| 223 | extern dt_node_t *dt_node_member(dt_decl_t *, char *, dt_node_t *); | 
| 224 | extern dt_node_t *dt_node_xlator(dt_decl_t *, dt_decl_t *, char *, dt_node_t *); | 
| 225 | extern dt_node_t *dt_node_probe(char *, int, dt_node_t *, dt_node_t *); | 
| 226 | extern dt_node_t *dt_node_provider(char *, dt_node_t *); | 
| 227 | extern dt_node_t *dt_node_program(dt_node_t *); | 
| 228 | extern dt_node_t *dt_node_if(dt_node_t *, dt_node_t *, dt_node_t *); | 
| 229 |  | 
| 230 | extern dt_node_t *dt_node_link(dt_node_t *, dt_node_t *); | 
| 231 | extern dt_node_t *dt_node_cook(dt_node_t *, uint_t); | 
| 232 |  | 
| 233 | extern dt_node_t *dt_node_xalloc(dtrace_hdl_t *, int); | 
| 234 | extern void dt_node_free(dt_node_t *); | 
| 235 |  | 
| 236 | extern dtrace_attribute_t dt_node_list_cook(dt_node_t **, uint_t); | 
| 237 | extern void dt_node_list_free(dt_node_t **); | 
| 238 | extern void dt_node_link_free(dt_node_t **); | 
| 239 |  | 
| 240 | extern void dt_node_attr_assign(dt_node_t *, dtrace_attribute_t); | 
| 241 | extern void dt_node_type_assign(dt_node_t *, ctf_file_t *, ctf_id_t, boolean_t); | 
| 242 | extern void dt_node_type_propagate(const dt_node_t *, dt_node_t *); | 
| 243 | extern const char *dt_node_type_name(const dt_node_t *, char *, size_t); | 
| 244 | extern size_t dt_node_type_size(const dt_node_t *); | 
| 245 |  | 
| 246 | extern dt_ident_t *dt_node_resolve(const dt_node_t *, uint_t); | 
| 247 | extern size_t dt_node_sizeof(const dt_node_t *); | 
| 248 | extern void dt_node_promote(dt_node_t *, dt_node_t *, dt_node_t *); | 
| 249 |  | 
| 250 | extern void dt_node_diftype(dtrace_hdl_t *, | 
| 251 |     const dt_node_t *, dtrace_diftype_t *); | 
| 252 | extern void dt_node_printr(dt_node_t *, FILE *, int); | 
| 253 | extern void dt_printd(dt_node_t *, FILE *, int); | 
| 254 | extern const char *dt_node_name(const dt_node_t *, char *, size_t); | 
| 255 | extern int dt_node_root(dt_node_t *); | 
| 256 |  | 
| 257 | struct dtrace_typeinfo;	/* see <dtrace.h> */ | 
| 258 | struct dt_pcb;		/* see <dt_impl.h> */ | 
| 259 |  | 
| 260 | #define	IS_CHAR(e) \ | 
| 261 | 	(((e).cte_format & (CTF_INT_CHAR | CTF_INT_SIGNED)) == \ | 
| 262 | 	(CTF_INT_CHAR | CTF_INT_SIGNED) && (e).cte_bits == NBBY) | 
| 263 |  | 
| 264 | #define	IS_VOID(e) \ | 
| 265 | 	((e).cte_offset == 0 && (e).cte_bits == 0) | 
| 266 |  | 
| 267 | extern int dt_type_lookup(const char *, struct dtrace_typeinfo *); | 
| 268 | extern int dt_type_pointer(struct dtrace_typeinfo *); | 
| 269 | extern const char *dt_type_name(ctf_file_t *, ctf_id_t, char *, size_t); | 
| 270 |  | 
| 271 | typedef enum { | 
| 272 | 	YYS_CLAUSE,	/* lex/yacc state for finding program clauses */ | 
| 273 | 	YYS_DEFINE,	/* lex/yacc state for parsing persistent definitions */ | 
| 274 | 	YYS_EXPR,	/* lex/yacc state for parsing D expressions */ | 
| 275 | 	YYS_DONE,	/* lex/yacc state for indicating parse tree is done */ | 
| 276 | 	YYS_CONTROL	/* lex/yacc state for parsing control lines */ | 
| 277 | } yystate_t; | 
| 278 |  | 
| 279 | extern void dnerror(const dt_node_t *, dt_errtag_t, const char *, ...) | 
| 280 |     __printflike(3, 4) __dead; | 
| 281 | extern void dnwarn(const dt_node_t *, dt_errtag_t, const char *, ...) | 
| 282 |     __printflike(3, 4); | 
| 283 |  | 
| 284 | extern void xyerror(dt_errtag_t, const char *, ...) __printflike(2, 3) | 
| 285 |     __dead; | 
| 286 | extern void xywarn(dt_errtag_t, const char *, ...) __printflike(2, 3); | 
| 287 | extern void xyvwarn(dt_errtag_t, const char *, va_list) __printflike(2, 0); | 
| 288 |  | 
| 289 | extern void yyerror(const char *, ...) __printflike(1, 2) __dead; | 
| 290 | extern void yywarn(const char *, ...) __printflike(1, 2); | 
| 291 | extern void yyvwarn(const char *, va_list) __printflike(1, 0); | 
| 292 |  | 
| 293 | extern void yylabel(const char *); | 
| 294 | extern void yybegin(yystate_t); | 
| 295 | extern void yyinit(struct dt_pcb *); | 
| 296 |  | 
| 297 | extern int yyparse(void); | 
| 298 | extern int yyinput(void); | 
| 299 |  | 
| 300 | #ifdef	__cplusplus | 
| 301 | } | 
| 302 | #endif | 
| 303 |  | 
| 304 | #endif	/* _DT_PARSER_H */ | 
| 305 |  |