aicasm_gram.y
上传用户:jlfgdled
上传日期:2013-04-10
资源大小:33168k
文件大小:38k
源码类别:

Linux/Unix编程

开发平台:

Unix_Linux

  1. %{
  2. /*
  3.  * Parser for the Aic7xxx SCSI Host adapter sequencer assembler.
  4.  *
  5.  * Copyright (c) 1997, 1998, 2000 Justin T. Gibbs.
  6.  * Copyright (c) 2001 Adaptec Inc.
  7.  * All rights reserved.
  8.  *
  9.  * Redistribution and use in source and binary forms, with or without
  10.  * modification, are permitted provided that the following conditions
  11.  * are met:
  12.  * 1. Redistributions of source code must retain the above copyright
  13.  *    notice, this list of conditions, and the following disclaimer,
  14.  *    without modification.
  15.  * 2. Redistributions in binary form must reproduce at minimum a disclaimer
  16.  *    substantially similar to the "NO WARRANTY" disclaimer below
  17.  *    ("Disclaimer") and any redistribution must be conditioned upon
  18.  *    including a substantially similar Disclaimer requirement for further
  19.  *    binary redistribution.
  20.  * 3. Neither the names of the above-listed copyright holders nor the names
  21.  *    of any contributors may be used to endorse or promote products derived
  22.  *    from this software without specific prior written permission.
  23.  *
  24.  * Alternatively, this software may be distributed under the terms of the
  25.  * GNU General Public License ("GPL") version 2 as published by the Free
  26.  * Software Foundation.
  27.  *
  28.  * NO WARRANTY
  29.  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
  30.  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
  31.  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR
  32.  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
  33.  * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
  34.  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
  35.  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
  36.  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
  37.  * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
  38.  * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
  39.  * POSSIBILITY OF SUCH DAMAGES.
  40.  *
  41.  * $Id: //depot/aic7xxx/aic7xxx/aicasm/aicasm_gram.y#14 $
  42.  *
  43.  * $FreeBSD: src/sys/dev/aic7xxx/aicasm/aicasm_gram.y,v 1.12 2000/10/31 18:44:32 gibbs Exp $
  44.  */
  45. #include <sys/types.h>
  46. #include <inttypes.h>
  47. #include <regex.h>
  48. #include <stdio.h>
  49. #include <stdlib.h>
  50. #include <string.h>
  51. #include <sysexits.h>
  52. #ifdef __linux__
  53. #include "../queue.h"
  54. #else
  55. #include <sys/queue.h>
  56. #endif
  57. #include "aicasm.h"
  58. #include "aicasm_symbol.h"
  59. #include "aicasm_insformat.h"
  60. int yylineno;
  61. char *yyfilename;
  62. char *patch_arg_list;
  63. char *versions;
  64. static char errbuf[255];
  65. static char regex_pattern[255];
  66. static symbol_t *cur_symbol;
  67. static symbol_t *scb_or_sram_symbol;
  68. static symtype cur_symtype;
  69. static symbol_ref_t accumulator;
  70. static symbol_ref_t mode_ptr;
  71. static symbol_ref_t allones;
  72. static symbol_ref_t allzeros;
  73. static symbol_ref_t none;
  74. static symbol_ref_t sindex;
  75. static int instruction_ptr;
  76. static int num_srams;
  77. static int sram_or_scb_offset;
  78. static int download_constant_count;
  79. static int in_critical_section;
  80. static void process_bitmask(int mask_type, symbol_t *sym, int mask);
  81. static void initialize_symbol(symbol_t *symbol);
  82. static void add_macro_arg(const char *argtext, int position);
  83. static void add_macro_body(const char *bodytext);
  84. static void process_register(symbol_t **p_symbol);
  85. static void format_1_instr(int opcode, symbol_ref_t *dest,
  86.    expression_t *immed, symbol_ref_t *src, int ret);
  87. static void format_2_instr(int opcode, symbol_ref_t *dest,
  88.    expression_t *places, symbol_ref_t *src, int ret);
  89. static void format_3_instr(int opcode, symbol_ref_t *src,
  90.    expression_t *immed, symbol_ref_t *address);
  91. static void test_readable_symbol(symbol_t *symbol);
  92. static void test_writable_symbol(symbol_t *symbol);
  93. static void type_check(symbol_t *symbol, expression_t *expression, int and_op);
  94. static void make_expression(expression_t *immed, int value);
  95. static void add_conditional(symbol_t *symbol);
  96. static void add_version(const char *verstring);
  97. static int  is_download_const(expression_t *immed);
  98. #define SRAM_SYMNAME "SRAM_BASE"
  99. #define SCB_SYMNAME "SCB_BASE"
  100. %}
  101. %union {
  102. u_int value;
  103. char *str;
  104. symbol_t *sym;
  105. symbol_ref_t sym_ref;
  106. expression_t expression;
  107. }
  108. %token T_REGISTER
  109. %token <value> T_CONST
  110. %token T_EXPORT
  111. %token T_DOWNLOAD
  112. %token T_SCB
  113. %token T_SRAM
  114. %token T_ALIAS
  115. %token T_SIZE
  116. %token T_EXPR_LSHIFT
  117. %token T_EXPR_RSHIFT
  118. %token <value> T_ADDRESS
  119. %token T_ACCESS_MODE
  120. %token T_MODES
  121. %token T_DEFINE
  122. %token T_SET_SRC_MODE
  123. %token T_SET_DST_MODE
  124. %token <value> T_MODE
  125. %token T_BEGIN_CS
  126. %token T_END_CS
  127. %token T_BIT
  128. %token T_MASK
  129. %token <value> T_NUMBER
  130. %token <str> T_PATH T_STRING T_ARG T_MACROBODY
  131. %token <sym> T_CEXPR
  132. %token T_EOF T_INCLUDE T_VERSION T_PATCH_ARG_LIST
  133. %token <value> T_SHR T_SHL T_ROR T_ROL
  134. %token <value> T_MVI T_MOV T_CLR T_BMOV
  135. %token <value> T_JMP T_JC T_JNC T_JE T_JNE T_JNZ T_JZ T_CALL
  136. %token <value> T_ADD T_ADC
  137. %token <value> T_INC T_DEC
  138. %token <value> T_STC T_CLC
  139. %token <value> T_CMP T_NOT T_XOR
  140. %token <value> T_TEST T_AND
  141. %token <value> T_OR
  142. %token T_RET
  143. %token T_NOP
  144. %token T_ACCUM T_ALLONES T_ALLZEROS T_NONE T_SINDEX T_MODE_PTR
  145. %token T_A
  146. %token <sym> T_SYMBOL
  147. %token T_NL
  148. %token T_IF T_ELSE T_ELSE_IF T_ENDIF
  149. %type <sym_ref> reg_symbol address destination source opt_source
  150. %type <expression> expression immediate immediate_or_a
  151. %type <value> export ret f1_opcode f2_opcode jmp_jc_jnc_call jz_jnz je_jne
  152. %type <value> numerical_value mode_value mode_list macro_arglist
  153. %left '|'
  154. %left '&'
  155. %left T_EXPR_LSHIFT T_EXPR_RSHIFT
  156. %left '+' '-'
  157. %left '*' '/'
  158. %right '~'
  159. %nonassoc UMINUS
  160. %%
  161. program:
  162. include
  163. | program include
  164. | patch_arg_list
  165. | program patch_arg_list
  166. | version
  167. | program version
  168. | register
  169. | program register
  170. | constant
  171. | program constant
  172. | macrodefn
  173. | program macrodefn
  174. | scratch_ram
  175. | program scratch_ram
  176. | scb
  177. | program scb
  178. | label
  179. | program label
  180. | set_src_mode
  181. | program set_src_mode
  182. | set_dst_mode
  183. | program set_dst_mode
  184. | critical_section_start
  185. | program critical_section_start
  186. | critical_section_end
  187. | program critical_section_end
  188. | conditional
  189. | program conditional
  190. | code
  191. | program code
  192. ;
  193. include:
  194. T_INCLUDE '<' T_PATH '>'
  195. {
  196. include_file($3, BRACKETED_INCLUDE);
  197. }
  198. | T_INCLUDE '"' T_PATH '"'
  199. {
  200. include_file($3, QUOTED_INCLUDE);
  201. }
  202. ;
  203. patch_arg_list:
  204. T_PATCH_ARG_LIST '=' T_STRING
  205. {
  206. if (patch_arg_list != NULL)
  207. stop("Patch argument list multiply defined",
  208.      EX_DATAERR);
  209. patch_arg_list = strdup($3);
  210. if (patch_arg_list == NULL)
  211. stop("Unable to record patch arg list", EX_SOFTWARE);
  212. }
  213. ;
  214. version:
  215. T_VERSION '=' T_STRING
  216. { add_version($3); }
  217. ;
  218. register:
  219. T_REGISTER { cur_symtype = REGISTER; } reg_definition
  220. ;
  221. reg_definition:
  222. T_SYMBOL '{'
  223. {
  224. if ($1->type != UNINITIALIZED) {
  225. stop("Register multiply defined", EX_DATAERR);
  226. /* NOTREACHED */
  227. }
  228. cur_symbol = $1; 
  229. cur_symbol->type = cur_symtype;
  230. initialize_symbol(cur_symbol);
  231. }
  232. reg_attribute_list
  233. '}'
  234. {                    
  235. /*
  236.  * Default to allowing everything in for registers
  237.  * with no bit or mask definitions.
  238.  */
  239. if (cur_symbol->info.rinfo->valid_bitmask == 0)
  240. cur_symbol->info.rinfo->valid_bitmask = 0xFF;
  241. if (cur_symbol->info.rinfo->size == 0)
  242. cur_symbol->info.rinfo->size = 1;
  243. /*
  244.  * This might be useful for registers too.
  245.  */
  246. if (cur_symbol->type != REGISTER) {
  247. if (cur_symbol->info.rinfo->address == 0)
  248. cur_symbol->info.rinfo->address =
  249.     sram_or_scb_offset;
  250. sram_or_scb_offset +=
  251.     cur_symbol->info.rinfo->size;
  252. }
  253. cur_symbol = NULL;
  254. }
  255. ;
  256. reg_attribute_list:
  257. reg_attribute
  258. | reg_attribute_list reg_attribute
  259. ;
  260. reg_attribute:
  261. reg_address
  262. | size
  263. | access_mode
  264. | modes
  265. | bit_defn
  266. | mask_defn
  267. | alias
  268. | accumulator
  269. | mode_pointer
  270. | allones
  271. | allzeros
  272. | none
  273. | sindex
  274. ;
  275. reg_address:
  276. T_ADDRESS T_NUMBER
  277. {
  278. cur_symbol->info.rinfo->address = $2;
  279. }
  280. ;
  281. size:
  282. T_SIZE T_NUMBER
  283. {
  284. cur_symbol->info.rinfo->size = $2;
  285. if (scb_or_sram_symbol != NULL) {
  286. u_int max_addr;
  287. u_int sym_max_addr;
  288. max_addr = scb_or_sram_symbol->info.rinfo->address
  289.  + scb_or_sram_symbol->info.rinfo->size;
  290. sym_max_addr = cur_symbol->info.rinfo->address
  291.      + cur_symbol->info.rinfo->size;
  292. if (sym_max_addr > max_addr)
  293. stop("SCB or SRAM space exhausted", EX_DATAERR);
  294. }
  295. }
  296. ;
  297. access_mode:
  298. T_ACCESS_MODE T_MODE
  299. {
  300. cur_symbol->info.rinfo->mode = $2;
  301. }
  302. ;
  303. modes:
  304. T_MODES mode_list
  305. {
  306. cur_symbol->info.rinfo->modes = $2;
  307. }
  308. ;
  309. mode_list:
  310. mode_value
  311. {
  312. $$ = $1;
  313. }
  314. | mode_list ',' mode_value
  315. {
  316. $$ = $1 | $3;
  317. }
  318. ;
  319. mode_value:
  320. T_NUMBER
  321. {
  322. if ($1 > 4) {
  323. stop("Valid register modes range between 0 and 4.",
  324.      EX_DATAERR);
  325. /* NOTREACHED */
  326. }
  327. $$ = (0x1 << $1);
  328. }
  329. | T_SYMBOL
  330. {
  331. symbol_t *symbol;
  332. symbol = $1;
  333. if (symbol->type != CONST) {
  334. stop("Only "const" symbols allowed in "
  335.      "mode definitions.", EX_DATAERR);
  336. /* NOTREACHED */
  337. }
  338. if (symbol->info.cinfo->value > 4) {
  339. stop("Valid register modes range between 0 and 4.",
  340.      EX_DATAERR);
  341. /* NOTREACHED */
  342. }
  343. $$ = (0x1 << symbol->info.cinfo->value);
  344. }
  345. ;
  346. bit_defn:
  347. T_BIT T_SYMBOL T_NUMBER
  348. {
  349. process_bitmask(BIT, $2, $3);
  350. }
  351. ;
  352. mask_defn:
  353. T_MASK T_SYMBOL expression
  354. {
  355. process_bitmask(MASK, $2, $3.value);
  356. }
  357. ;
  358. alias:
  359. T_ALIAS T_SYMBOL
  360. {
  361. if ($2->type != UNINITIALIZED) {
  362. stop("Re-definition of register alias",
  363.      EX_DATAERR);
  364. /* NOTREACHED */
  365. }
  366. $2->type = ALIAS;
  367. initialize_symbol($2);
  368. $2->info.ainfo->parent = cur_symbol;
  369. }
  370. ;
  371. accumulator:
  372. T_ACCUM
  373. {
  374. if (accumulator.symbol != NULL) {
  375. stop("Only one accumulator definition allowed",
  376.      EX_DATAERR);
  377. /* NOTREACHED */
  378. }
  379. accumulator.symbol = cur_symbol;
  380. }
  381. ;
  382. mode_pointer:
  383. T_MODE_PTR
  384. {
  385. if (mode_ptr.symbol != NULL) {
  386. stop("Only one mode pointer definition allowed",
  387.      EX_DATAERR);
  388. /* NOTREACHED */
  389. }
  390. mode_ptr.symbol = cur_symbol;
  391. }
  392. ;
  393. allones:
  394. T_ALLONES
  395. {
  396. if (allones.symbol != NULL) {
  397. stop("Only one definition of allones allowed",
  398.      EX_DATAERR);
  399. /* NOTREACHED */
  400. }
  401. allones.symbol = cur_symbol;
  402. }
  403. ;
  404. allzeros:
  405. T_ALLZEROS
  406. {
  407. if (allzeros.symbol != NULL) {
  408. stop("Only one definition of allzeros allowed",
  409.      EX_DATAERR);
  410. /* NOTREACHED */
  411. }
  412. allzeros.symbol = cur_symbol;
  413. }
  414. ;
  415. none:
  416. T_NONE
  417. {
  418. if (none.symbol != NULL) {
  419. stop("Only one definition of none allowed",
  420.      EX_DATAERR);
  421. /* NOTREACHED */
  422. }
  423. none.symbol = cur_symbol;
  424. }
  425. ;
  426. sindex:
  427. T_SINDEX
  428. {
  429. if (sindex.symbol != NULL) {
  430. stop("Only one definition of sindex allowed",
  431.      EX_DATAERR);
  432. /* NOTREACHED */
  433. }
  434. sindex.symbol = cur_symbol;
  435. }
  436. ;
  437. expression:
  438. expression '|' expression
  439. {
  440.  $$.value = $1.value | $3.value;
  441.  symlist_merge(&$$.referenced_syms,
  442.        &$1.referenced_syms,
  443.        &$3.referenced_syms);
  444. }
  445. | expression '&' expression
  446. {
  447. $$.value = $1.value & $3.value;
  448. symlist_merge(&$$.referenced_syms,
  449.        &$1.referenced_syms,
  450.        &$3.referenced_syms);
  451. }
  452. | expression '+' expression
  453. {
  454. $$.value = $1.value + $3.value;
  455. symlist_merge(&$$.referenced_syms,
  456.        &$1.referenced_syms,
  457.        &$3.referenced_syms);
  458. }
  459. | expression '-' expression
  460. {
  461. $$.value = $1.value - $3.value;
  462. symlist_merge(&($$.referenced_syms),
  463.        &($1.referenced_syms),
  464.        &($3.referenced_syms));
  465. }
  466. | expression '*' expression
  467. {
  468. $$.value = $1.value * $3.value;
  469. symlist_merge(&($$.referenced_syms),
  470.        &($1.referenced_syms),
  471.        &($3.referenced_syms));
  472. }
  473. | expression '/' expression
  474. {
  475. $$.value = $1.value / $3.value;
  476. symlist_merge(&($$.referenced_syms),
  477.        &($1.referenced_syms),
  478.        &($3.referenced_syms));
  479. }
  480. |  expression T_EXPR_LSHIFT expression
  481. {
  482. $$.value = $1.value << $3.value;
  483. symlist_merge(&$$.referenced_syms,
  484.        &$1.referenced_syms,
  485.        &$3.referenced_syms);
  486. }
  487. |  expression T_EXPR_RSHIFT expression
  488. {
  489. $$.value = $1.value >> $3.value;
  490. symlist_merge(&$$.referenced_syms,
  491.        &$1.referenced_syms,
  492.        &$3.referenced_syms);
  493. }
  494. | '(' expression ')'
  495. {
  496. $$ = $2;
  497. }
  498. | '~' expression
  499. {
  500. $$ = $2;
  501. $$.value = (~$$.value) & 0xFF;
  502. }
  503. | '-' expression %prec UMINUS
  504. {
  505. $$ = $2;
  506. $$.value = -$$.value;
  507. }
  508. | T_NUMBER
  509. {
  510. $$.value = $1;
  511. SLIST_INIT(&$$.referenced_syms);
  512. }
  513. | T_SYMBOL
  514. {
  515. symbol_t *symbol;
  516. symbol = $1;
  517. switch (symbol->type) {
  518. case ALIAS:
  519. symbol = $1->info.ainfo->parent;
  520. case REGISTER:
  521. case SCBLOC:
  522. case SRAMLOC:
  523. $$.value = symbol->info.rinfo->address;
  524. break;
  525. case MASK:
  526. case BIT:
  527. $$.value = symbol->info.minfo->mask;
  528. break;
  529. case DOWNLOAD_CONST:
  530. case CONST:
  531. $$.value = symbol->info.cinfo->value;
  532. break;
  533. case UNINITIALIZED:
  534. default:
  535. {
  536. snprintf(errbuf, sizeof(errbuf),
  537.  "Undefined symbol %s referenced",
  538.  symbol->name);
  539. stop(errbuf, EX_DATAERR);
  540. /* NOTREACHED */
  541. break;
  542. }
  543. }
  544. SLIST_INIT(&$$.referenced_syms);
  545. symlist_add(&$$.referenced_syms, symbol, SYMLIST_INSERT_HEAD);
  546. }
  547. ;
  548. constant:
  549. T_CONST T_SYMBOL numerical_value
  550. {
  551. if ($2->type != UNINITIALIZED) {
  552. stop("Re-definition of symbol as a constant",
  553.      EX_DATAERR);
  554. /* NOTREACHED */
  555. }
  556. $2->type = CONST;
  557. initialize_symbol($2);
  558. $2->info.cinfo->value = $3;
  559. }
  560. | T_CONST T_SYMBOL T_DOWNLOAD
  561. {
  562. if ($1) {
  563. stop("Invalid downloaded constant declaration",
  564.      EX_DATAERR);
  565. /* NOTREACHED */
  566. }
  567. if ($2->type != UNINITIALIZED) {
  568. stop("Re-definition of symbol as a downloaded constant",
  569.      EX_DATAERR);
  570. /* NOTREACHED */
  571. }
  572. $2->type = DOWNLOAD_CONST;
  573. initialize_symbol($2);
  574. $2->info.cinfo->value = download_constant_count++;
  575. }
  576. ;
  577. macrodefn_prologue:
  578. T_DEFINE T_SYMBOL
  579. {
  580. if ($2->type != UNINITIALIZED) {
  581. stop("Re-definition of symbol as a macro",
  582.      EX_DATAERR);
  583. /* NOTREACHED */
  584. }
  585. cur_symbol = $2;
  586. cur_symbol->type = MACRO;
  587. initialize_symbol(cur_symbol);
  588. }
  589. ;
  590. macrodefn:
  591. macrodefn_prologue T_MACROBODY
  592. {
  593. add_macro_body($2);
  594. }
  595. | macrodefn_prologue '(' macro_arglist ')' T_MACROBODY
  596. {
  597. add_macro_body($5);
  598. cur_symbol->info.macroinfo->narg = $3;
  599. }
  600. ;
  601. macro_arglist:
  602. {
  603. /* Macros can take no arguments */
  604. $$ = 0;
  605. }
  606. | T_ARG
  607. {
  608. $$ = 1;
  609. add_macro_arg($1, 0);
  610. }
  611. | macro_arglist ',' T_ARG
  612. {
  613. if ($1 == 0) {
  614. stop("Comma without preceeding argument in arg list",
  615.      EX_DATAERR);
  616. /* NOTREACHED */
  617. }
  618. $$ = $1 + 1;
  619. add_macro_arg($3, $1);
  620. }
  621. ;
  622. numerical_value:
  623. T_NUMBER
  624. {
  625. $$ = $1;
  626. }
  627. | '-' T_NUMBER
  628. {
  629. $$ = -$2;
  630. }
  631. ;
  632. scratch_ram:
  633. T_SRAM '{'
  634. {
  635. snprintf(errbuf, sizeof(errbuf), "%s%d", SRAM_SYMNAME,
  636.  num_srams);
  637. cur_symbol = symtable_get(SRAM_SYMNAME);
  638. cur_symtype = SRAMLOC;
  639. cur_symbol->type = SRAMLOC;
  640. initialize_symbol(cur_symbol);
  641. }
  642. reg_address
  643. {
  644. sram_or_scb_offset = cur_symbol->info.rinfo->address;
  645. }
  646. size
  647. {
  648. scb_or_sram_symbol = cur_symbol;
  649. }
  650. scb_or_sram_attributes
  651. '}'
  652. {
  653. cur_symbol = NULL;
  654. scb_or_sram_symbol = NULL;
  655. }
  656. ;
  657. scb:
  658. T_SCB '{'
  659. {
  660. cur_symbol = symtable_get(SCB_SYMNAME);
  661. cur_symtype = SCBLOC;
  662. if (cur_symbol->type != UNINITIALIZED) {
  663. stop("Only one SRAM definition allowed",
  664.      EX_SOFTWARE);
  665. /* NOTREACHED */
  666. }
  667. cur_symbol->type = SCBLOC;
  668. initialize_symbol(cur_symbol);
  669. /* 64 bytes of SCB space */
  670. cur_symbol->info.rinfo->size = 64;
  671. }
  672. reg_address
  673. {
  674. sram_or_scb_offset = cur_symbol->info.rinfo->address;
  675. }
  676. size
  677. {
  678. scb_or_sram_symbol = cur_symbol;
  679. }
  680. scb_or_sram_attributes
  681. '}'
  682. {
  683. cur_symbol = NULL;
  684. scb_or_sram_symbol = NULL;
  685. }
  686. ;
  687. scb_or_sram_attributes:
  688. /* NULL definition is okay */
  689. | modes
  690. | scb_or_sram_reg_list
  691. | modes scb_or_sram_reg_list
  692. ;
  693. scb_or_sram_reg_list:
  694. reg_definition
  695. | scb_or_sram_reg_list reg_definition
  696. ;
  697. reg_symbol:
  698. T_SYMBOL
  699. {
  700. process_register(&$1);
  701. $$.symbol = $1;
  702. $$.offset = 0;
  703. }
  704. | T_SYMBOL '[' T_SYMBOL ']'
  705. {
  706. process_register(&$1);
  707. if ($3->type != CONST) {
  708. stop("register offset must be a constant", EX_DATAERR);
  709. /* NOTREACHED */
  710. }
  711. if (($3->info.cinfo->value + 1) > $1->info.rinfo->size) {
  712. stop("Accessing offset beyond range of register",
  713.      EX_DATAERR);
  714. /* NOTREACHED */
  715. }
  716. $$.symbol = $1;
  717. $$.offset = $3->info.cinfo->value;
  718. }
  719. | T_SYMBOL '[' T_NUMBER ']'
  720. {
  721. process_register(&$1);
  722. if (($3 + 1) > $1->info.rinfo->size) {
  723. stop("Accessing offset beyond range of register",
  724.      EX_DATAERR);
  725. /* NOTREACHED */
  726. }
  727. $$.symbol = $1;
  728. $$.offset = $3;
  729. }
  730. | T_A
  731. {
  732. if (accumulator.symbol == NULL) {
  733. stop("No accumulator has been defined", EX_DATAERR);
  734. /* NOTREACHED */
  735. }
  736. $$.symbol = accumulator.symbol;
  737. $$.offset = 0;
  738. }
  739. ;
  740. destination:
  741. reg_symbol
  742. {
  743. test_writable_symbol($1.symbol);
  744. $$ = $1;
  745. }
  746. ;
  747. immediate:
  748. expression
  749. { $$ = $1; }
  750. ;
  751. immediate_or_a:
  752. expression
  753. {
  754. if ($1.value == 0 && is_download_const(&$1) == 0) {
  755. snprintf(errbuf, sizeof(errbuf),
  756.  "nExpression evaluates to 0 and thus "
  757.  "references the accumulator.n "
  758.  "If this is the desired effect, use 'A' "
  759.  "instead.n");
  760. stop(errbuf, EX_DATAERR);
  761. }
  762. $$ = $1;
  763. }
  764. | T_A
  765. {
  766. SLIST_INIT(&$$.referenced_syms);
  767. $$.value = 0;
  768. }
  769. ;
  770. source:
  771. reg_symbol
  772. {
  773. test_readable_symbol($1.symbol);
  774. $$ = $1;
  775. }
  776. ;
  777. opt_source:
  778. {
  779. $$.symbol = NULL;
  780. $$.offset = 0;
  781. }
  782. | ',' source
  783. { $$ = $2; }
  784. ;
  785. ret:
  786. { $$ = 0; }
  787. | T_RET
  788. { $$ = 1; }
  789. ;
  790. set_src_mode:
  791. T_SET_SRC_MODE T_NUMBER ';'
  792. {
  793. src_mode = $2;
  794. }
  795. ;
  796. set_dst_mode:
  797. T_SET_DST_MODE T_NUMBER ';'
  798. {
  799. dst_mode = $2;
  800. }
  801. ;
  802. critical_section_start:
  803. T_BEGIN_CS ';'
  804. {
  805. critical_section_t *cs;
  806. if (in_critical_section != FALSE) {
  807. stop("Critical Section within Critical Section",
  808.      EX_DATAERR);
  809. /* NOTREACHED */
  810. }
  811. cs = cs_alloc();
  812. cs->begin_addr = instruction_ptr;
  813. in_critical_section = TRUE;
  814. }
  815. critical_section_end:
  816. T_END_CS ';'
  817. {
  818. critical_section_t *cs;
  819. if (in_critical_section == FALSE) {
  820. stop("Unballanced 'end_cs'", EX_DATAERR);
  821. /* NOTREACHED */
  822. }
  823. cs = TAILQ_LAST(&cs_tailq, cs_tailq);
  824. cs->end_addr = instruction_ptr;
  825. in_critical_section = FALSE;
  826. }
  827. export:
  828. { $$ = 0; }
  829. | T_EXPORT
  830. { $$ = 1; }
  831. ;
  832. label:
  833. export T_SYMBOL ':'
  834. {
  835. if ($2->type != UNINITIALIZED) {
  836. stop("Program label multiply defined", EX_DATAERR);
  837. /* NOTREACHED */
  838. }
  839. $2->type = LABEL;
  840. initialize_symbol($2);
  841. $2->info.linfo->address = instruction_ptr;
  842. $2->info.linfo->exported = $1;
  843. }
  844. ;
  845. address:
  846. T_SYMBOL
  847. {
  848. $$.symbol = $1;
  849. $$.offset = 0;
  850. }
  851. | T_SYMBOL '+' T_NUMBER
  852. {
  853. $$.symbol = $1;
  854. $$.offset = $3;
  855. }
  856. | T_SYMBOL '-' T_NUMBER
  857. {
  858. $$.symbol = $1;
  859. $$.offset = -$3;
  860. }
  861. | '.'
  862. {
  863. $$.symbol = NULL;
  864. $$.offset = 0;
  865. }
  866. | '.' '+' T_NUMBER
  867. {
  868. $$.symbol = NULL;
  869. $$.offset = $3;
  870. }
  871. | '.' '-' T_NUMBER
  872. {
  873. $$.symbol = NULL;
  874. $$.offset = -$3;
  875. }
  876. ;
  877. conditional:
  878. T_IF T_CEXPR '{'
  879. {
  880. scope_t *new_scope;
  881. add_conditional($2);
  882. new_scope = scope_alloc();
  883. new_scope->type = SCOPE_IF;
  884. new_scope->begin_addr = instruction_ptr;
  885. new_scope->func_num = $2->info.condinfo->func_num;
  886. }
  887. | T_ELSE T_IF T_CEXPR '{'
  888. {
  889. scope_t *new_scope;
  890. scope_t *scope_context;
  891. scope_t *last_scope;
  892. /*
  893.  * Ensure that the previous scope is either an
  894.  * if or and else if.
  895.  */
  896. scope_context = SLIST_FIRST(&scope_stack);
  897. last_scope = TAILQ_LAST(&scope_context->inner_scope,
  898. scope_tailq);
  899. if (last_scope == NULL
  900.  || last_scope->type == T_ELSE) {
  901. stop("'else if' without leading 'if'", EX_DATAERR);
  902. /* NOTREACHED */
  903. }
  904. add_conditional($3);
  905. new_scope = scope_alloc();
  906. new_scope->type = SCOPE_ELSE_IF;
  907. new_scope->begin_addr = instruction_ptr;
  908. new_scope->func_num = $3->info.condinfo->func_num;
  909. }
  910. | T_ELSE '{'
  911. {
  912. scope_t *new_scope;
  913. scope_t *scope_context;
  914. scope_t *last_scope;
  915. /*
  916.  * Ensure that the previous scope is either an
  917.  * if or and else if.
  918.  */
  919. scope_context = SLIST_FIRST(&scope_stack);
  920. last_scope = TAILQ_LAST(&scope_context->inner_scope,
  921. scope_tailq);
  922. if (last_scope == NULL
  923.  || last_scope->type == SCOPE_ELSE) {
  924. stop("'else' without leading 'if'", EX_DATAERR);
  925. /* NOTREACHED */
  926. }
  927. new_scope = scope_alloc();
  928. new_scope->type = SCOPE_ELSE;
  929. new_scope->begin_addr = instruction_ptr;
  930. }
  931. ;
  932. conditional:
  933. '}'
  934. {
  935. scope_t *scope_context;
  936. scope_context = SLIST_FIRST(&scope_stack);
  937. if (scope_context->type == SCOPE_ROOT) {
  938. stop("Unexpected '}' encountered", EX_DATAERR);
  939. /* NOTREACHED */
  940. }
  941. scope_context->end_addr = instruction_ptr;
  942. /* Pop the scope */
  943. SLIST_REMOVE_HEAD(&scope_stack, scope_stack_links);
  944. process_scope(scope_context);
  945. if (SLIST_FIRST(&scope_stack) == NULL) {
  946. stop("Unexpected '}' encountered", EX_DATAERR);
  947. /* NOTREACHED */
  948. }
  949. }
  950. ;
  951. f1_opcode:
  952. T_AND { $$ = AIC_OP_AND; }
  953. | T_XOR { $$ = AIC_OP_XOR; }
  954. | T_ADD { $$ = AIC_OP_ADD; }
  955. | T_ADC { $$ = AIC_OP_ADC; }
  956. ;
  957. code:
  958. f1_opcode destination ',' immediate_or_a opt_source ret ';'
  959. {
  960. format_1_instr($1, &$2, &$4, &$5, $6);
  961. }
  962. ;
  963. code:
  964. T_OR reg_symbol ',' immediate_or_a opt_source ret ';'
  965. {
  966. format_1_instr(AIC_OP_OR, &$2, &$4, &$5, $6);
  967. }
  968. ;
  969. code:
  970. T_INC destination opt_source ret ';'
  971. {
  972. expression_t immed;
  973. make_expression(&immed, 1);
  974. format_1_instr(AIC_OP_ADD, &$2, &immed, &$3, $4);
  975. }
  976. ;
  977. code:
  978. T_DEC destination opt_source ret ';'
  979. {
  980. expression_t immed;
  981. make_expression(&immed, -1);
  982. format_1_instr(AIC_OP_ADD, &$2, &immed, &$3, $4);
  983. }
  984. ;
  985. code:
  986. T_CLC ret ';'
  987. {
  988. expression_t immed;
  989. make_expression(&immed, -1);
  990. format_1_instr(AIC_OP_ADD, &none, &immed, &allzeros, $2);
  991. }
  992. | T_CLC T_MVI destination ',' immediate_or_a ret ';'
  993. {
  994. format_1_instr(AIC_OP_ADD, &$3, &$5, &allzeros, $6);
  995. }
  996. ;
  997. code:
  998. T_STC ret ';'
  999. {
  1000. expression_t immed;
  1001. make_expression(&immed, 1);
  1002. format_1_instr(AIC_OP_ADD, &none, &immed, &allones, $2);
  1003. }
  1004. | T_STC destination ret ';'
  1005. {
  1006. expression_t immed;
  1007. make_expression(&immed, 1);
  1008. format_1_instr(AIC_OP_ADD, &$2, &immed, &allones, $3);
  1009. }
  1010. ;
  1011. code:
  1012. T_BMOV destination ',' source ',' immediate ret ';'
  1013. {
  1014. format_1_instr(AIC_OP_BMOV, &$2, &$6, &$4, $7);
  1015. }
  1016. ;
  1017. code:
  1018. T_MOV destination ',' source ret ';'
  1019. {
  1020. expression_t immed;
  1021. make_expression(&immed, 1);
  1022. format_1_instr(AIC_OP_BMOV, &$2, &immed, &$4, $5);
  1023. }
  1024. ;
  1025. code:
  1026. T_MVI destination ',' immediate_or_a ret ';'
  1027. {
  1028. format_1_instr(AIC_OP_OR, &$2, &$4, &allzeros, $5);
  1029. }
  1030. ;
  1031. code:
  1032. T_NOT destination opt_source ret ';'
  1033. {
  1034. expression_t immed;
  1035. make_expression(&immed, 0xff);
  1036. format_1_instr(AIC_OP_XOR, &$2, &immed, &$3, $4);
  1037. }
  1038. ;
  1039. code:
  1040. T_CLR destination ret ';'
  1041. {
  1042. expression_t immed;
  1043. make_expression(&immed, 0xff);
  1044. format_1_instr(AIC_OP_AND, &$2, &immed, &allzeros, $3);
  1045. }
  1046. ;
  1047. code:
  1048. T_NOP ret ';'
  1049. {
  1050. expression_t immed;
  1051. make_expression(&immed, 0xff);
  1052. format_1_instr(AIC_OP_AND, &none, &immed, &allzeros, $2);
  1053. }
  1054. ;
  1055. code:
  1056. T_RET ';'
  1057. {
  1058. expression_t immed;
  1059. make_expression(&immed, 0xff);
  1060. format_1_instr(AIC_OP_AND, &none, &immed, &allzeros, TRUE);
  1061. }
  1062. ;
  1063. /*
  1064.  * This grammer differs from the one in the aic7xxx
  1065.  * reference manual since the grammer listed there is
  1066.  * ambiguous and causes a shift/reduce conflict.
  1067.  * It also seems more logical as the "immediate"
  1068.  * argument is listed as the second arg like the
  1069.  * other formats.
  1070.  */
  1071. f2_opcode:
  1072. T_SHL { $$ = AIC_OP_SHL; }
  1073. | T_SHR { $$ = AIC_OP_SHR; }
  1074. | T_ROL { $$ = AIC_OP_ROL; }
  1075. | T_ROR { $$ = AIC_OP_ROR; }
  1076. ;
  1077. code:
  1078. f2_opcode destination ',' expression opt_source ret ';'
  1079. {
  1080. format_2_instr($1, &$2, &$4, &$5, $6);
  1081. }
  1082. ;
  1083. jmp_jc_jnc_call:
  1084. T_JMP { $$ = AIC_OP_JMP; }
  1085. | T_JC { $$ = AIC_OP_JC; }
  1086. | T_JNC { $$ = AIC_OP_JNC; }
  1087. | T_CALL { $$ = AIC_OP_CALL; }
  1088. ;
  1089. jz_jnz:
  1090. T_JZ { $$ = AIC_OP_JZ; }
  1091. | T_JNZ { $$ = AIC_OP_JNZ; }
  1092. ;
  1093. je_jne:
  1094. T_JE { $$ = AIC_OP_JE; }
  1095. | T_JNE { $$ = AIC_OP_JNE; }
  1096. ;
  1097. code:
  1098. jmp_jc_jnc_call address ';'
  1099. {
  1100. expression_t immed;
  1101. make_expression(&immed, 0);
  1102. format_3_instr($1, &sindex, &immed, &$2);
  1103. }
  1104. ;
  1105. code:
  1106. T_OR reg_symbol ',' immediate jmp_jc_jnc_call address ';'
  1107. {
  1108. format_3_instr($5, &$2, &$4, &$6);
  1109. }
  1110. ;
  1111. code:
  1112. T_TEST source ',' immediate_or_a jz_jnz address ';'
  1113. {
  1114. format_3_instr($5, &$2, &$4, &$6);
  1115. }
  1116. ;
  1117. code:
  1118. T_CMP source ',' immediate_or_a je_jne address ';'
  1119. {
  1120. format_3_instr($5, &$2, &$4, &$6);
  1121. }
  1122. ;
  1123. code:
  1124. T_MOV source jmp_jc_jnc_call address ';'
  1125. {
  1126. expression_t immed;
  1127. make_expression(&immed, 0);
  1128. format_3_instr($3, &$2, &immed, &$4);
  1129. }
  1130. ;
  1131. code:
  1132. T_MVI immediate jmp_jc_jnc_call address ';'
  1133. {
  1134. format_3_instr($3, &allzeros, &$2, &$4);
  1135. }
  1136. ;
  1137. %%
  1138. static void
  1139. process_bitmask(int mask_type, symbol_t *sym, int mask)
  1140. {
  1141. /*
  1142.  * Add the current register to its
  1143.  * symbol list, if it already exists,
  1144.  * warn if we are setting it to a
  1145.  * different value, or in the bit to
  1146.  * the "allowed bits" of this register.
  1147.  */
  1148. if (sym->type == UNINITIALIZED) {
  1149. sym->type = mask_type;
  1150. initialize_symbol(sym);
  1151. if (mask_type == BIT) {
  1152. if (mask == 0) {
  1153. stop("Bitmask with no bits set", EX_DATAERR);
  1154. /* NOTREACHED */
  1155. }
  1156. if ((mask & ~(0x01 << (ffs(mask) - 1))) != 0) {
  1157. stop("Bitmask with more than one bit set",
  1158.      EX_DATAERR);
  1159. /* NOTREACHED */
  1160. }
  1161. }
  1162. sym->info.minfo->mask = mask;
  1163. } else if (sym->type != mask_type) {
  1164. stop("Bit definition mirrors a definition of the same "
  1165.      " name, but a different type", EX_DATAERR);
  1166. /* NOTREACHED */
  1167. } else if (mask != sym->info.minfo->mask) {
  1168. stop("Bitmask redefined with a conflicting value", EX_DATAERR);
  1169. /* NOTREACHED */
  1170. }
  1171. /* Fail if this symbol is already listed */
  1172. if (symlist_search(&(sym->info.minfo->symrefs),
  1173.    cur_symbol->name) != NULL) {
  1174. stop("Bitmask defined multiple times for register", EX_DATAERR);
  1175. /* NOTREACHED */
  1176. }
  1177. symlist_add(&(sym->info.minfo->symrefs), cur_symbol,
  1178.     SYMLIST_INSERT_HEAD);
  1179. cur_symbol->info.rinfo->valid_bitmask |= mask;
  1180. cur_symbol->info.rinfo->typecheck_masks = TRUE;
  1181. }
  1182. static void
  1183. initialize_symbol(symbol_t *symbol)
  1184. {
  1185. switch (symbol->type) {
  1186.         case UNINITIALIZED:
  1187. stop("Call to initialize_symbol with type field unset",
  1188.      EX_SOFTWARE);
  1189. /* NOTREACHED */
  1190. break;
  1191.         case REGISTER:
  1192.         case SRAMLOC:
  1193.         case SCBLOC:
  1194. symbol->info.rinfo =
  1195.     (struct reg_info *)malloc(sizeof(struct reg_info));
  1196. if (symbol->info.rinfo == NULL) {
  1197. stop("Can't create register info", EX_SOFTWARE);
  1198. /* NOTREACHED */
  1199. }
  1200. memset(symbol->info.rinfo, 0,
  1201.        sizeof(struct reg_info));
  1202. /*
  1203.  * Default to allowing access in all register modes
  1204.  * or to the mode specified by the SCB or SRAM space
  1205.  * we are in.
  1206.  */
  1207. if (scb_or_sram_symbol != NULL)
  1208. symbol->info.rinfo->modes =
  1209.     scb_or_sram_symbol->info.rinfo->modes;
  1210. else
  1211. symbol->info.rinfo->modes = ~0;
  1212. break;
  1213.         case ALIAS:
  1214. symbol->info.ainfo =
  1215.     (struct alias_info *)malloc(sizeof(struct alias_info));
  1216. if (symbol->info.ainfo == NULL) {
  1217. stop("Can't create alias info", EX_SOFTWARE);
  1218. /* NOTREACHED */
  1219. }
  1220. memset(symbol->info.ainfo, 0,
  1221.        sizeof(struct alias_info));
  1222. break;
  1223.         case MASK:
  1224.         case BIT:
  1225. symbol->info.minfo =
  1226.     (struct mask_info *)malloc(sizeof(struct mask_info));
  1227. if (symbol->info.minfo == NULL) {
  1228. stop("Can't create bitmask info", EX_SOFTWARE);
  1229. /* NOTREACHED */
  1230. }
  1231. memset(symbol->info.minfo, 0, sizeof(struct mask_info));
  1232. SLIST_INIT(&(symbol->info.minfo->symrefs));
  1233. break;
  1234.         case CONST:
  1235.         case DOWNLOAD_CONST:
  1236. symbol->info.cinfo =
  1237.     (struct const_info *)malloc(sizeof(struct const_info));
  1238. if (symbol->info.cinfo == NULL) {
  1239. stop("Can't create alias info", EX_SOFTWARE);
  1240. /* NOTREACHED */
  1241. }
  1242. memset(symbol->info.cinfo, 0,
  1243.        sizeof(struct const_info));
  1244. break;
  1245. case LABEL:
  1246. symbol->info.linfo =
  1247.     (struct label_info *)malloc(sizeof(struct label_info));
  1248. if (symbol->info.linfo == NULL) {
  1249. stop("Can't create label info", EX_SOFTWARE);
  1250. /* NOTREACHED */
  1251. }
  1252. memset(symbol->info.linfo, 0,
  1253.        sizeof(struct label_info));
  1254. break;
  1255. case CONDITIONAL:
  1256. symbol->info.condinfo =
  1257.     (struct cond_info *)malloc(sizeof(struct cond_info));
  1258. if (symbol->info.condinfo == NULL) {
  1259. stop("Can't create conditional info", EX_SOFTWARE);
  1260. /* NOTREACHED */
  1261. }
  1262. memset(symbol->info.condinfo, 0,
  1263.        sizeof(struct cond_info));
  1264. break;
  1265. case MACRO:
  1266. symbol->info.macroinfo = 
  1267.     (struct macro_info *)malloc(sizeof(struct macro_info));
  1268. if (symbol->info.macroinfo == NULL) {
  1269. stop("Can't create macro info", EX_SOFTWARE);
  1270. /* NOTREACHED */
  1271. }
  1272. memset(symbol->info.macroinfo, 0,
  1273.        sizeof(struct macro_info));
  1274. STAILQ_INIT(&symbol->info.macroinfo->args);
  1275. break;
  1276. default:
  1277. stop("Call to initialize_symbol with invalid symbol type",
  1278.      EX_SOFTWARE);
  1279. /* NOTREACHED */
  1280. break;
  1281. }
  1282. }
  1283. static void
  1284. add_macro_arg(const char *argtext, int argnum)
  1285. {
  1286. struct macro_arg *marg;
  1287. int i;
  1288. int retval;
  1289. if (cur_symbol == NULL || cur_symbol->type != MACRO) {
  1290. stop("Invalid current symbol for adding macro arg",
  1291.      EX_SOFTWARE);
  1292. /* NOTREACHED */
  1293. }
  1294. marg = (struct macro_arg *)malloc(sizeof(*marg));
  1295. if (marg == NULL) {
  1296. stop("Can't create macro_arg structure", EX_SOFTWARE);
  1297. /* NOTREACHED */
  1298. }
  1299. marg->replacement_text = NULL;
  1300. retval = snprintf(regex_pattern, sizeof(regex_pattern),
  1301.   "[^-/A-Za-z0-9_](%s)([^-/A-Za-z0-9_]|$)",
  1302.   argtext);
  1303. if (retval >= sizeof(regex_pattern)) {
  1304. stop("Regex text buffer too small for arg",
  1305.      EX_SOFTWARE);
  1306. /* NOTREACHED */
  1307. }
  1308. retval = regcomp(&marg->arg_regex, regex_pattern, REG_EXTENDED);
  1309. if (retval != 0) {
  1310. stop("Regex compilation failed", EX_SOFTWARE);
  1311. /* NOTREACHED */
  1312. }
  1313. STAILQ_INSERT_TAIL(&cur_symbol->info.macroinfo->args, marg, links);
  1314. }
  1315. static void
  1316. add_macro_body(const char *bodytext)
  1317. {
  1318. if (cur_symbol == NULL || cur_symbol->type != MACRO) {
  1319. stop("Invalid current symbol for adding macro arg",
  1320.      EX_SOFTWARE);
  1321. /* NOTREACHED */
  1322. }
  1323. cur_symbol->info.macroinfo->body = strdup(bodytext);
  1324. if (cur_symbol->info.macroinfo->body == NULL) {
  1325. stop("Can't duplicate macro body text", EX_SOFTWARE);
  1326. /* NOTREACHED */
  1327. }
  1328. }
  1329. static void
  1330. process_register(symbol_t **p_symbol)
  1331. {
  1332. symbol_t *symbol = *p_symbol;
  1333. if (symbol->type == UNINITIALIZED) {
  1334. snprintf(errbuf, sizeof(errbuf), "Undefined register %s",
  1335.  symbol->name);
  1336. stop(errbuf, EX_DATAERR);
  1337. /* NOTREACHED */
  1338. } else if (symbol->type == ALIAS) {
  1339. *p_symbol = symbol->info.ainfo->parent;
  1340. } else if ((symbol->type != REGISTER)
  1341. && (symbol->type != SCBLOC)
  1342. && (symbol->type != SRAMLOC)) {
  1343. snprintf(errbuf, sizeof(errbuf),
  1344.  "Specified symbol %s is not a register",
  1345.  symbol->name);
  1346. stop(errbuf, EX_DATAERR);
  1347. }
  1348. }
  1349. static void
  1350. format_1_instr(int opcode, symbol_ref_t *dest, expression_t *immed,
  1351.        symbol_ref_t *src, int ret)
  1352. {
  1353. struct instruction *instr;
  1354. struct ins_format1 *f1_instr;
  1355. if (src->symbol == NULL)
  1356. src = dest;
  1357. /* Test register permissions */
  1358. test_writable_symbol(dest->symbol);
  1359. test_readable_symbol(src->symbol);
  1360. /* Ensure that immediate makes sense for this destination */
  1361. type_check(dest->symbol, immed, opcode);
  1362. /* Allocate sequencer space for the instruction and fill it out */
  1363. instr = seq_alloc();
  1364. f1_instr = &instr->format.format1;
  1365. f1_instr->ret = ret ? 1 : 0;
  1366. f1_instr->opcode = opcode;
  1367. f1_instr->destination = dest->symbol->info.rinfo->address
  1368.       + dest->offset;
  1369. f1_instr->source = src->symbol->info.rinfo->address
  1370.  + src->offset;
  1371. f1_instr->immediate = immed->value;
  1372. if (is_download_const(immed))
  1373. f1_instr->parity = 1;
  1374. else if (dest->symbol == mode_ptr.symbol) {
  1375. u_int src_value;
  1376. u_int dst_value;
  1377. /*
  1378.  * Attempt to update mode information if
  1379.  * we are operating on the mode register.
  1380.  */
  1381. if (src->symbol == allones.symbol)
  1382. src_value = 0xFF;
  1383. else if (src->symbol == allzeros.symbol)
  1384. src_value = 0;
  1385. else if (src->symbol == mode_ptr.symbol)
  1386. src_value = (dst_mode << 4) | src_mode;
  1387. else
  1388. goto cant_update;
  1389. switch (opcode) {
  1390. case AIC_OP_AND:
  1391. dst_value = src_value & immed->value;
  1392. break;
  1393. case AIC_OP_XOR:
  1394. dst_value = src_value ^ immed->value;
  1395. break;
  1396. case AIC_OP_ADD:
  1397. dst_value = (src_value + immed->value) & 0xFF;
  1398. break;
  1399. case AIC_OP_OR:
  1400. dst_value = src_value | immed->value;
  1401. break;
  1402. break;
  1403. case AIC_OP_BMOV:
  1404. dst_value = src_value;
  1405. break;
  1406. default:
  1407. goto cant_update;
  1408. }
  1409. src_mode = dst_value & 0xF;
  1410. dst_mode = (dst_value >> 4) & 0xF;
  1411. cant_update:
  1412. }
  1413. symlist_free(&immed->referenced_syms);
  1414. instruction_ptr++;
  1415. }
  1416. static void
  1417. format_2_instr(int opcode, symbol_ref_t *dest, expression_t *places,
  1418.        symbol_ref_t *src, int ret)
  1419. {
  1420. struct instruction *instr;
  1421. struct ins_format2 *f2_instr;
  1422. uint8_t shift_control;
  1423. if (src->symbol == NULL)
  1424. src = dest;
  1425. /* Test register permissions */
  1426. test_writable_symbol(dest->symbol);
  1427. test_readable_symbol(src->symbol);
  1428. /* Allocate sequencer space for the instruction and fill it out */
  1429. instr = seq_alloc();
  1430. f2_instr = &instr->format.format2;
  1431. f2_instr->ret = ret ? 1 : 0;
  1432. f2_instr->opcode = AIC_OP_ROL;
  1433. f2_instr->destination = dest->symbol->info.rinfo->address
  1434.       + dest->offset;
  1435. f2_instr->source = src->symbol->info.rinfo->address
  1436.  + src->offset;
  1437. if (places->value > 8 || places->value <= 0) {
  1438. stop("illegal shift value", EX_DATAERR);
  1439. /* NOTREACHED */
  1440. }
  1441. switch (opcode) {
  1442. case AIC_OP_SHL:
  1443. if (places->value == 8)
  1444. shift_control = 0xf0;
  1445. else
  1446. shift_control = (places->value << 4) | places->value;
  1447. break;
  1448. case AIC_OP_SHR:
  1449. if (places->value == 8) {
  1450. shift_control = 0xf8;
  1451. } else {
  1452. shift_control = (places->value << 4)
  1453.       | (8 - places->value)
  1454.       | 0x08;
  1455. }
  1456. break;
  1457. case AIC_OP_ROL:
  1458. shift_control = places->value & 0x7;
  1459. break;
  1460. case AIC_OP_ROR:
  1461. shift_control = (8 - places->value) | 0x08;
  1462. break;
  1463. default:
  1464. shift_control = 0; /* Quiet Compiler */
  1465. stop("Invalid shift operation specified", EX_SOFTWARE);
  1466. /* NOTREACHED */
  1467. break;
  1468. };
  1469. f2_instr->shift_control = shift_control;
  1470. symlist_free(&places->referenced_syms);
  1471. instruction_ptr++;
  1472. }
  1473. static void
  1474. format_3_instr(int opcode, symbol_ref_t *src,
  1475.        expression_t *immed, symbol_ref_t *address)
  1476. {
  1477. struct instruction *instr;
  1478. struct ins_format3 *f3_instr;
  1479. int addr;
  1480. /* Test register permissions */
  1481. test_readable_symbol(src->symbol);
  1482. /* Ensure that immediate makes sense for this source */
  1483. type_check(src->symbol, immed, opcode);
  1484. /* Allocate sequencer space for the instruction and fill it out */
  1485. instr = seq_alloc();
  1486. f3_instr = &instr->format.format3;
  1487. if (address->symbol == NULL) {
  1488. /* 'dot' referrence.  Use the current instruction pointer */
  1489. addr = instruction_ptr + address->offset;
  1490. } else if (address->symbol->type == UNINITIALIZED) {
  1491. /* forward reference */
  1492. addr = address->offset;
  1493. instr->patch_label = address->symbol;
  1494. } else
  1495. addr = address->symbol->info.linfo->address + address->offset;
  1496. f3_instr->opcode = opcode;
  1497. f3_instr->address = addr;
  1498. f3_instr->source = src->symbol->info.rinfo->address
  1499.  + src->offset;
  1500. f3_instr->immediate = immed->value;
  1501. if (is_download_const(immed))
  1502. f3_instr->parity = 1;
  1503. symlist_free(&immed->referenced_syms);
  1504. instruction_ptr++;
  1505. }
  1506. static void
  1507. test_readable_symbol(symbol_t *symbol)
  1508. {
  1509. if ((symbol->info.rinfo->modes & (0x1 << src_mode)) == 0) {
  1510. snprintf(errbuf, sizeof(errbuf),
  1511. "Register %s unavailable in source reg mode %d",
  1512. symbol->name, src_mode);
  1513. stop(errbuf, EX_DATAERR);
  1514. }
  1515. if (symbol->info.rinfo->mode == WO) {
  1516. stop("Write Only register specified as source",
  1517.      EX_DATAERR);
  1518. /* NOTREACHED */
  1519. }
  1520. }
  1521. static void
  1522. test_writable_symbol(symbol_t *symbol)
  1523. {
  1524. if ((symbol->info.rinfo->modes & (0x1 << dst_mode)) == 0) {
  1525. snprintf(errbuf, sizeof(errbuf),
  1526. "Register %s unavailable in destination reg mode %d",
  1527. symbol->name, dst_mode);
  1528. stop(errbuf, EX_DATAERR);
  1529. }
  1530. if (symbol->info.rinfo->mode == RO) {
  1531. stop("Read Only register specified as destination",
  1532.      EX_DATAERR);
  1533. /* NOTREACHED */
  1534. }
  1535. }
  1536. static void
  1537. type_check(symbol_t *symbol, expression_t *expression, int opcode)
  1538. {
  1539. symbol_node_t *node;
  1540. int and_op;
  1541. and_op = FALSE;
  1542. if (opcode == AIC_OP_AND || opcode == AIC_OP_JNZ || AIC_OP_JZ)
  1543. and_op = TRUE;
  1544. /*
  1545.  * Make sure that we aren't attempting to write something
  1546.  * that hasn't been defined.  If this is an and operation,
  1547.  * this is a mask, so "undefined" bits are okay.
  1548.  */
  1549. if (and_op == FALSE
  1550.  && (expression->value & ~symbol->info.rinfo->valid_bitmask) != 0) {
  1551. snprintf(errbuf, sizeof(errbuf),
  1552.  "Invalid bit(s) 0x%x in immediate written to %s",
  1553.  expression->value & ~symbol->info.rinfo->valid_bitmask,
  1554.  symbol->name);
  1555. stop(errbuf, EX_DATAERR);
  1556. /* NOTREACHED */
  1557. }
  1558. /*
  1559.  * Now make sure that all of the symbols referenced by the
  1560.  * expression are defined for this register.
  1561.  */
  1562. if (symbol->info.rinfo->typecheck_masks != FALSE) {
  1563. for(node = expression->referenced_syms.slh_first;
  1564.     node != NULL;
  1565.     node = node->links.sle_next) {
  1566. if ((node->symbol->type == MASK
  1567.   || node->symbol->type == BIT)
  1568.  && symlist_search(&node->symbol->info.minfo->symrefs,
  1569.    symbol->name) == NULL) {
  1570. snprintf(errbuf, sizeof(errbuf),
  1571.  "Invalid bit or mask %s "
  1572.  "for register %s",
  1573.  node->symbol->name, symbol->name);
  1574. stop(errbuf, EX_DATAERR);
  1575. /* NOTREACHED */
  1576. }
  1577. }
  1578. }
  1579. }
  1580. static void
  1581. make_expression(expression_t *immed, int value)
  1582. {
  1583. SLIST_INIT(&immed->referenced_syms);
  1584. immed->value = value & 0xff;
  1585. }
  1586. static void
  1587. add_conditional(symbol_t *symbol)
  1588. {
  1589. static int numfuncs;
  1590. if (numfuncs == 0) {
  1591. /* add a special conditional, "0" */
  1592. symbol_t *false_func;
  1593. false_func = symtable_get("0");
  1594. if (false_func->type != UNINITIALIZED) {
  1595. stop("Conditional expression '0' "
  1596.      "conflicts with a symbol", EX_DATAERR);
  1597. /* NOTREACHED */
  1598. }
  1599. false_func->type = CONDITIONAL;
  1600. initialize_symbol(false_func);
  1601. false_func->info.condinfo->func_num = numfuncs++;
  1602. symlist_add(&patch_functions, false_func, SYMLIST_INSERT_HEAD);
  1603. }
  1604. /* This condition has occurred before */
  1605. if (symbol->type == CONDITIONAL)
  1606. return;
  1607. if (symbol->type != UNINITIALIZED) {
  1608. stop("Conditional expression conflicts with a symbol",
  1609.      EX_DATAERR);
  1610. /* NOTREACHED */
  1611. }
  1612. symbol->type = CONDITIONAL;
  1613. initialize_symbol(symbol);
  1614. symbol->info.condinfo->func_num = numfuncs++;
  1615. symlist_add(&patch_functions, symbol, SYMLIST_INSERT_HEAD);
  1616. }
  1617. static void
  1618. add_version(const char *verstring)
  1619. {
  1620. const char prefix[] = " * ";
  1621. int newlen;
  1622. int oldlen;
  1623. newlen = strlen(verstring) + strlen(prefix);
  1624. oldlen = 0;
  1625. if (versions != NULL)
  1626. oldlen = strlen(versions);
  1627. versions = realloc(versions, newlen + oldlen + 2);
  1628. if (versions == NULL)
  1629. stop("Can't allocate version string", EX_SOFTWARE);
  1630. strcpy(&versions[oldlen], prefix);
  1631. strcpy(&versions[oldlen + strlen(prefix)], verstring);
  1632. versions[newlen + oldlen] = 'n';
  1633. versions[newlen + oldlen + 1] = '';
  1634. }
  1635. void
  1636. yyerror(const char *string)
  1637. {
  1638. stop(string, EX_DATAERR);
  1639. }
  1640. static int
  1641. is_download_const(expression_t *immed)
  1642. {
  1643. if ((immed->referenced_syms.slh_first != NULL)
  1644.  && (immed->referenced_syms.slh_first->symbol->type == DOWNLOAD_CONST))
  1645. return (TRUE);
  1646. return (FALSE);
  1647. }