| OpenCOBOL 1.1pre-rel | 
00001 /* 00002 * Copyright (C) 2001-2009 Keisuke Nishida 00003 * Copyright (C) 2007-2009 Roger While 00004 * 00005 * This program is free software; you can redistribute it and/or modify 00006 * it under the terms of the GNU General Public License as published by 00007 * the Free Software Foundation; either version 2, or (at your option) 00008 * any later version. 00009 * 00010 * This program is distributed in the hope that it will be useful, 00011 * but WITHOUT ANY WARRANTY; without even the implied warranty of 00012 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 00013 * GNU General Public License for more details. 00014 * 00015 * You should have received a copy of the GNU General Public License 00016 * along with this software; see the file COPYING. If not, write to 00017 * the Free Software Foundation, 51 Franklin Street, Fifth Floor 00018 * Boston, MA 02110-1301 USA 00019 */ 00020 00021 #include "config.h" 00022 00023 #include <stdio.h> 00024 #include <stdlib.h> 00025 #include <string.h> 00026 #include <ctype.h> 00027 #include <time.h> 00028 #ifdef HAVE_SYS_TIME_H 00029 #include <sys/time.h> 00030 #endif 00031 #ifdef _WIN32 00032 #define WINDOWS_LEAN_AND_MEAN 00033 #include <windows.h> 00034 #endif 00035 00036 #ifdef HAVE_LOCALE_H 00037 #include <locale.h> 00038 #endif 00039 00040 #include "cobc.h" 00041 #include "tree.h" 00042 00043 struct system_table { 00044 const char *syst_name; 00045 const int syst_params; 00046 }; 00047 00048 struct expr_node { 00049 /* The token of this node. 00050 * 'x' - values (cb_tree) 00051 * '+', '-', '*', '/', '^' - arithmetic operators 00052 * '=', '~', '<', '>', '[', ']' - relational operators 00053 * '!', '&', '|' - logical operators 00054 * '(', ')' - parentheses 00055 */ 00056 int token; 00057 /* The value itself if this node is a value */ 00058 cb_tree value; 00059 }; 00060 00061 #define START_STACK_SIZE 32 00062 #define TOKEN(offset) (expr_stack[expr_index + offset].token) 00063 #define VALUE(offset) (expr_stack[expr_index + offset].value) 00064 00065 #define dpush(x) decimal_stack = cb_cons (x, decimal_stack) 00066 00067 #define cb_emit(x) \ 00068 current_statement->body = cb_list_add (current_statement->body, x) 00069 #define cb_emit_list(l) \ 00070 current_statement->body = cb_list_append (current_statement->body, l) 00071 00072 /* Global variables */ 00073 00074 size_t sending_id = 0; 00075 size_t suppress_warn = 0; 00076 00077 /* Local variables */ 00078 00079 static cb_tree decimal_stack = NULL; 00080 00081 static const char *inspect_func; 00082 static cb_tree inspect_data; 00083 00084 static int expr_op; /* last operator */ 00085 static cb_tree expr_lh; /* last left hand */ 00086 00087 static int expr_index; /* stack index */ 00088 static int expr_stack_size; /* stack max size */ 00089 static struct expr_node *expr_stack; /* expr node stack */ 00090 00091 static char expr_prio[256]; 00092 00093 static const struct system_table system_tab[] = { 00094 #undef COB_SYSTEM_GEN 00095 #define COB_SYSTEM_GEN(x, y, z) { x, y }, 00096 #include "libcob/system.def" 00097 { NULL, 0 } 00098 }; 00099 00100 static const char *const bin_set_funcs[] = { 00101 NULL, 00102 "cob_setswp_u16_binary", 00103 "cob_setswp_u24_binary", 00104 "cob_setswp_u32_binary", 00105 "cob_setswp_u40_binary", 00106 "cob_setswp_u48_binary", 00107 "cob_setswp_u56_binary", 00108 "cob_setswp_u64_binary", 00109 NULL, 00110 "cob_setswp_s16_binary", 00111 "cob_setswp_s24_binary", 00112 "cob_setswp_s32_binary", 00113 "cob_setswp_s40_binary", 00114 "cob_setswp_s48_binary", 00115 "cob_setswp_s56_binary", 00116 "cob_setswp_s64_binary" 00117 }; 00118 00119 static const char *const bin_compare_funcs[] = { 00120 "cob_cmp_u8_binary", 00121 "cob_cmp_u16_binary", 00122 "cob_cmp_u24_binary", 00123 "cob_cmp_u32_binary", 00124 "cob_cmp_u40_binary", 00125 "cob_cmp_u48_binary", 00126 "cob_cmp_u56_binary", 00127 "cob_cmp_u64_binary", 00128 "cob_cmp_s8_binary", 00129 "cob_cmp_s16_binary", 00130 "cob_cmp_s24_binary", 00131 "cob_cmp_s32_binary", 00132 "cob_cmp_s40_binary", 00133 "cob_cmp_s48_binary", 00134 "cob_cmp_s56_binary", 00135 "cob_cmp_s64_binary", 00136 "cob_cmp_u8_binary", 00137 "cob_cmpswp_u16_binary", 00138 "cob_cmpswp_u24_binary", 00139 "cob_cmpswp_u32_binary", 00140 "cob_cmpswp_u40_binary", 00141 "cob_cmpswp_u48_binary", 00142 "cob_cmpswp_u56_binary", 00143 "cob_cmpswp_u64_binary", 00144 "cob_cmp_s8_binary", 00145 "cob_cmpswp_s16_binary", 00146 "cob_cmpswp_s24_binary", 00147 "cob_cmpswp_s32_binary", 00148 "cob_cmpswp_s40_binary", 00149 "cob_cmpswp_s48_binary", 00150 "cob_cmpswp_s56_binary", 00151 "cob_cmpswp_s64_binary" 00152 }; 00153 00154 static const char *const bin_add_funcs[] = { 00155 "cob_add_u8_binary", 00156 "cob_add_u16_binary", 00157 "cob_add_u24_binary", 00158 "cob_add_u32_binary", 00159 "cob_add_u40_binary", 00160 "cob_add_u48_binary", 00161 "cob_add_u56_binary", 00162 "cob_add_u64_binary", 00163 "cob_add_s8_binary", 00164 "cob_add_s16_binary", 00165 "cob_add_s24_binary", 00166 "cob_add_s32_binary", 00167 "cob_add_s40_binary", 00168 "cob_add_s48_binary", 00169 "cob_add_s56_binary", 00170 "cob_add_s64_binary", 00171 "cob_add_u8_binary", 00172 "cob_addswp_u16_binary", 00173 "cob_addswp_u24_binary", 00174 "cob_addswp_u32_binary", 00175 "cob_addswp_u40_binary", 00176 "cob_addswp_u48_binary", 00177 "cob_addswp_u56_binary", 00178 "cob_addswp_u64_binary", 00179 "cob_add_s8_binary", 00180 "cob_addswp_s16_binary", 00181 "cob_addswp_s24_binary", 00182 "cob_addswp_s32_binary", 00183 "cob_addswp_s40_binary", 00184 "cob_addswp_s48_binary", 00185 "cob_addswp_s56_binary", 00186 "cob_addswp_s64_binary" 00187 }; 00188 00189 static const char *const bin_sub_funcs[] = { 00190 "cob_sub_u8_binary", 00191 "cob_sub_u16_binary", 00192 "cob_sub_u24_binary", 00193 "cob_sub_u32_binary", 00194 "cob_sub_u40_binary", 00195 "cob_sub_u48_binary", 00196 "cob_sub_u56_binary", 00197 "cob_sub_u64_binary", 00198 "cob_sub_s8_binary", 00199 "cob_sub_s16_binary", 00200 "cob_sub_s24_binary", 00201 "cob_sub_s32_binary", 00202 "cob_sub_s40_binary", 00203 "cob_sub_s48_binary", 00204 "cob_sub_s56_binary", 00205 "cob_sub_s64_binary", 00206 "cob_sub_u8_binary", 00207 "cob_subswp_u16_binary", 00208 "cob_subswp_u24_binary", 00209 "cob_subswp_u32_binary", 00210 "cob_subswp_u40_binary", 00211 "cob_subswp_u48_binary", 00212 "cob_subswp_u56_binary", 00213 "cob_subswp_u64_binary", 00214 "cob_sub_s8_binary", 00215 "cob_subswp_s16_binary", 00216 "cob_subswp_s24_binary", 00217 "cob_subswp_s32_binary", 00218 "cob_subswp_s40_binary", 00219 "cob_subswp_s48_binary", 00220 "cob_subswp_s56_binary", 00221 "cob_subswp_s64_binary" 00222 }; 00223 00224 static const char *const align_bin_compare_funcs[] = { 00225 "cob_cmp_u8_binary", 00226 "cob_cmp_align_u16_binary", 00227 "cob_cmp_u24_binary", 00228 "cob_cmp_align_u32_binary", 00229 "cob_cmp_u40_binary", 00230 "cob_cmp_u48_binary", 00231 "cob_cmp_u56_binary", 00232 "cob_cmp_align_u64_binary", 00233 "cob_cmp_s8_binary", 00234 "cob_cmp_align_s16_binary", 00235 "cob_cmp_s24_binary", 00236 "cob_cmp_align_s32_binary", 00237 "cob_cmp_s40_binary", 00238 "cob_cmp_s48_binary", 00239 "cob_cmp_s56_binary", 00240 "cob_cmp_align_s64_binary", 00241 "cob_cmp_u8_binary", 00242 "cob_cmpswp_align_u16_binary", 00243 "cob_cmpswp_u24_binary", 00244 "cob_cmpswp_align_u32_binary", 00245 "cob_cmpswp_u40_binary", 00246 "cob_cmpswp_u48_binary", 00247 "cob_cmpswp_u56_binary", 00248 "cob_cmpswp_align_u64_binary", 00249 "cob_cmp_s8_binary", 00250 "cob_cmpswp_align_s16_binary", 00251 "cob_cmpswp_s24_binary", 00252 "cob_cmpswp_align_s32_binary", 00253 "cob_cmpswp_s40_binary", 00254 "cob_cmpswp_s48_binary", 00255 "cob_cmpswp_s56_binary", 00256 "cob_cmpswp_align_s64_binary" 00257 }; 00258 00259 static const char *const align_bin_add_funcs[] = { 00260 "cob_add_u8_binary", 00261 "cob_add_align_u16_binary", 00262 "cob_add_u24_binary", 00263 "cob_add_align_u32_binary", 00264 "cob_add_u40_binary", 00265 "cob_add_u48_binary", 00266 "cob_add_u56_binary", 00267 "cob_add_align_u64_binary", 00268 "cob_add_s8_binary", 00269 "cob_add_align_s16_binary", 00270 "cob_add_s24_binary", 00271 "cob_add_align_s32_binary", 00272 "cob_add_s40_binary", 00273 "cob_add_s48_binary", 00274 "cob_add_s56_binary", 00275 "cob_add_align_s64_binary", 00276 "cob_add_u8_binary", 00277 "cob_addswp_u16_binary", 00278 "cob_addswp_u24_binary", 00279 "cob_addswp_u32_binary", 00280 "cob_addswp_u40_binary", 00281 "cob_addswp_u48_binary", 00282 "cob_addswp_u56_binary", 00283 "cob_addswp_u64_binary", 00284 "cob_add_s8_binary", 00285 "cob_addswp_s16_binary", 00286 "cob_addswp_s24_binary", 00287 "cob_addswp_s32_binary", 00288 "cob_addswp_s40_binary", 00289 "cob_addswp_s48_binary", 00290 "cob_addswp_s56_binary", 00291 "cob_addswp_s64_binary" 00292 }; 00293 00294 static const char *const align_bin_sub_funcs[] = { 00295 "cob_sub_u8_binary", 00296 "cob_sub_align_u16_binary", 00297 "cob_sub_u24_binary", 00298 "cob_sub_align_u32_binary", 00299 "cob_sub_u40_binary", 00300 "cob_sub_u48_binary", 00301 "cob_sub_u56_binary", 00302 "cob_sub_align_u64_binary", 00303 "cob_sub_s8_binary", 00304 "cob_sub_align_s16_binary", 00305 "cob_sub_s24_binary", 00306 "cob_sub_align_s32_binary", 00307 "cob_sub_s40_binary", 00308 "cob_sub_s48_binary", 00309 "cob_sub_s56_binary", 00310 "cob_sub_align_s64_binary", 00311 "cob_sub_u8_binary", 00312 "cob_subswp_u16_binary", 00313 "cob_subswp_u24_binary", 00314 "cob_subswp_u32_binary", 00315 "cob_subswp_u40_binary", 00316 "cob_subswp_u48_binary", 00317 "cob_subswp_u56_binary", 00318 "cob_subswp_u64_binary", 00319 "cob_sub_s8_binary", 00320 "cob_subswp_s16_binary", 00321 "cob_subswp_s24_binary", 00322 "cob_subswp_s32_binary", 00323 "cob_subswp_s40_binary", 00324 "cob_subswp_s48_binary", 00325 "cob_subswp_s56_binary", 00326 "cob_subswp_s64_binary" 00327 }; 00328 00329 /* functions */ 00330 00331 static size_t 00332 cb_validate_one (cb_tree x) 00333 { 00334 cb_tree y; 00335 00336 if (x == cb_error_node) { 00337 return 1; 00338 } 00339 if (!x) { 00340 return 0; 00341 } 00342 if (CB_REFERENCE_P (x)) { 00343 y = cb_ref (x); 00344 if (y == cb_error_node) { 00345 return 1; 00346 } 00347 if (CB_FIELD_P (y) && CB_FIELD (y)->level == 88) { 00348 cb_error_x (x, _("Invalid use of 88 level item")); 00349 return 1; 00350 } 00351 } 00352 return 0; 00353 } 00354 00355 static size_t 00356 cb_validate_list (cb_tree l) 00357 { 00358 for (; l; l = CB_CHAIN (l)) { 00359 if (cb_validate_one (CB_VALUE (l))) { 00360 return 1; 00361 } 00362 } 00363 return 0; 00364 } 00365 00366 static cb_tree 00367 cb_check_group_name (cb_tree x) 00368 { 00369 cb_tree y; 00370 00371 if (x == cb_error_node) { 00372 return cb_error_node; 00373 } 00374 00375 if (CB_REFERENCE_P (x)) { 00376 y = cb_ref (x); 00377 if (y == cb_error_node) { 00378 return cb_error_node; 00379 } 00380 if (CB_FIELD_P (y) && CB_FIELD (y)->children != NULL && 00381 CB_REFERENCE (x)->offset == NULL) { 00382 return x; 00383 } 00384 } 00385 00386 cb_error_x (x, _("'%s' is not group name"), cb_name (x)); 00387 return cb_error_node; 00388 } 00389 00390 static cb_tree 00391 cb_check_numeric_name (cb_tree x) 00392 { 00393 if (x == cb_error_node) { 00394 return cb_error_node; 00395 } 00396 00397 if (CB_REFERENCE_P (x) 00398 && CB_FIELD_P (cb_ref (x)) 00399 && CB_TREE_CATEGORY (x) == CB_CATEGORY_NUMERIC) { 00400 return x; 00401 } 00402 00403 cb_error_x (x, _("'%s' is not a numeric name"), cb_name (x)); 00404 return cb_error_node; 00405 } 00406 00407 static cb_tree 00408 cb_check_numeric_edited_name (cb_tree x) 00409 { 00410 if (x == cb_error_node) { 00411 return cb_error_node; 00412 } 00413 00414 if (CB_REFERENCE_P (x) 00415 && CB_FIELD_P (cb_ref (x)) 00416 && (CB_TREE_CATEGORY (x) == CB_CATEGORY_NUMERIC 00417 || CB_TREE_CATEGORY (x) == CB_CATEGORY_NUMERIC_EDITED)) { 00418 return x; 00419 } 00420 00421 cb_error_x (x, _("'%s' is not numeric or numeric-edited name"), cb_name (x)); 00422 return cb_error_node; 00423 } 00424 00425 cb_tree 00426 cb_check_numeric_value (cb_tree x) 00427 { 00428 if (x == cb_error_node) { 00429 return cb_error_node; 00430 } 00431 00432 if (CB_TREE_CATEGORY (x) == CB_CATEGORY_NUMERIC) { 00433 return x; 00434 } 00435 00436 cb_error_x (x, _("'%s' is not a numeric value"), cb_name (x)); 00437 return cb_error_node; 00438 } 00439 00440 static cb_tree 00441 cb_check_integer_value (cb_tree x) 00442 { 00443 struct cb_literal *l; 00444 struct cb_field *f; 00445 cb_tree y; 00446 00447 if (x == cb_error_node) { 00448 return cb_error_node; 00449 } 00450 00451 if (CB_TREE_CATEGORY (x) != CB_CATEGORY_NUMERIC) { 00452 goto invalid; 00453 } 00454 00455 switch (CB_TREE_TAG (x)) { 00456 case CB_TAG_CONST: 00457 if (x != cb_zero) { 00458 goto invalid; 00459 } 00460 return x; 00461 case CB_TAG_LITERAL: 00462 l = CB_LITERAL (x); 00463 if (l->sign < 0 || l->scale > 0) { 00464 goto invliteral; 00465 } 00466 return x; 00467 case CB_TAG_REFERENCE: 00468 y = cb_ref (x); 00469 if (y == cb_error_node) { 00470 return cb_error_node; 00471 } 00472 f = CB_FIELD (y); 00473 if (f->pic->scale > 0) { 00474 goto invalid; 00475 } 00476 return x; 00477 case CB_TAG_BINARY_OP: 00478 /* TODO: need to check */ 00479 return x; 00480 case CB_TAG_INTRINSIC: 00481 /* TODO: need to check */ 00482 return x; 00483 default: 00484 invalid: 00485 cb_error_x (x, _("'%s' is not an integer value"), cb_name (x)); 00486 return cb_error_node; 00487 } 00488 invliteral: 00489 cb_error_x (x, _("A positive numeric integer is required here")); 00490 return cb_error_node; 00491 } 00492 00493 void 00494 cb_build_registers (void) 00495 { 00496 #if !defined(__linux__) && !defined(__CYGWIN__) && defined(HAVE_TIMEZONE) 00497 long contz; 00498 #endif 00499 time_t t; 00500 char buff[48]; 00501 00502 /* RETURN-CODE */ 00503 if (!current_program->nested_level) { 00504 current_program->cb_return_code = 00505 cb_build_index (cb_build_reference ("RETURN-CODE"), 00506 cb_zero, 0, NULL); 00507 cb_field (current_program->cb_return_code)->flag_is_global = 1; 00508 } 00509 00510 /* SORT-RETURN */ 00511 current_program->cb_sort_return = 00512 cb_build_index (cb_build_reference ("SORT-RETURN"), cb_zero, 0, NULL); 00513 cb_field (current_program->cb_sort_return)->flag_no_init = 1; 00514 00515 /* NUMBER-OF-CALL-PARAMETERS */ 00516 current_program->cb_call_params = 00517 cb_build_index (cb_build_reference ("NUMBER-OF-CALL-PARAMETERS"), cb_zero, 0, NULL); 00518 cb_field (current_program->cb_call_params)->flag_no_init = 1; 00519 00520 /* TALLY */ 00521 /* 01 TALLY GLOBAL PICTURE 9(9) USAGE COMP-5 VALUE ZERO. */ 00522 /* TALLY/EXAMINE not standard/supported */ 00523 00524 t = time (NULL); 00525 00526 /* WHEN-COMPILED */ 00527 memset (buff, 0, sizeof (buff)); 00528 strftime (buff, 17, "%m/%d/%y%H.%M.%S", localtime (&t)); 00529 cb_build_constant (cb_build_reference ("WHEN-COMPILED"), 00530 cb_build_alphanumeric_literal ((ucharptr)buff, 16)); 00531 00532 /* FUNCTION WHEN-COMPILED */ 00533 memset (buff, 0, sizeof (buff)); 00534 #if defined(__linux__) || defined(__CYGWIN__) 00535 strftime (buff, 22, "%Y%m%d%H%M%S00%z", localtime (&t)); 00536 #elif defined(HAVE_TIMEZONE) 00537 strftime (buff, 17, "%Y%m%d%H%M%S00", localtime (&t)); 00538 if (timezone <= 0) { 00539 contz = -timezone; 00540 buff[16] = '+'; 00541 } else { 00542 contz = timezone; 00543 buff[16] = '-'; 00544 } 00545 sprintf (&buff[17], "%2.2ld%2.2ld", contz / 3600, contz % 60); 00546 #else 00547 strftime (buff, 22, "%Y%m%d%H%M%S0000000", localtime (&t)); 00548 #endif 00549 cb_intr_whencomp = cb_build_alphanumeric_literal ((ucharptr)buff, 21); 00550 00551 /* FUNCTION PI */ 00552 memset (buff, 0, sizeof (buff)); 00553 strcpy (buff, "31415926535897932384626433832795029"); 00554 cb_intr_pi = cb_build_numeric_literal (0, (ucharptr)buff, 34); 00555 00556 /* FUNCTION E */ 00557 memset (buff, 0, sizeof (buff)); 00558 strcpy (buff, "27182818284590452353602874713526625"); 00559 cb_intr_e = cb_build_numeric_literal (0, (ucharptr)buff, 34); 00560 } 00561 00562 char * 00563 cb_encode_program_id (const char *name) 00564 { 00565 unsigned char *p; 00566 const unsigned char *s; 00567 unsigned char buff[COB_SMALL_BUFF]; 00568 00569 p = buff; 00570 s = (const unsigned char *)name; 00571 /* encode the initial digit */ 00572 if (isdigit (*s)) { 00573 p += sprintf ((char *)p, "_%02X", *s++); 00574 } 00575 /* encode invalid letters */ 00576 for (; *s; s++) { 00577 if (isalnum (*s) || *s == '_') { 00578 *p++ = *s; 00579 } else if (*s == '-') { 00580 *p++ = '_'; 00581 *p++ = '_'; 00582 } else { 00583 p += sprintf ((char *)p, "_%02X", *s); 00584 } 00585 } 00586 *p = 0; 00587 return strdup ((char *)buff); 00588 } 00589 00590 const char * 00591 cb_build_program_id (cb_tree name, cb_tree alt_name) 00592 { 00593 const char *s; 00594 00595 /* This needs some more thought, should we generate an entry 00596 point per program source name ? 00597 if (alt_name) { 00598 s = (char *)CB_LITERAL (alt_name)->data; 00599 } else if (CB_LITERAL_P (name)) { 00600 s = (char *)CB_LITERAL (name)->data; 00601 } else { 00602 s = (char *)CB_NAME (name); 00603 } 00604 00605 if (!cb_flag_main && strcmp (s, source_name)) { 00606 cb_warning (_("Source name '%s' differs from PROGRAM-ID '%s'"), 00607 source_name, s); 00608 current_program->source_name = strdup (source_name); 00609 } 00610 End comment out */ 00611 00612 if (alt_name) { 00613 current_program->orig_source_name = strdup ((char *)CB_LITERAL (alt_name)->data); 00614 s = (char *)CB_LITERAL (alt_name)->data; 00615 } else if (CB_LITERAL_P (name)) { 00616 current_program->orig_source_name = strdup ((char *)CB_LITERAL (name)->data); 00617 s = cb_encode_program_id ((char *)CB_LITERAL (name)->data); 00618 } else { 00619 current_program->orig_source_name = strdup (CB_NAME (name)); 00620 s = cb_encode_program_id (CB_NAME (name)); 00621 } 00622 if (cobc_check_valid_name (current_program->orig_source_name)) { 00623 cb_error (_("PROGRAM-ID '%s' invalid"), current_program->orig_source_name); 00624 } 00625 return s; 00626 } 00627 00628 void 00629 cb_define_switch_name (cb_tree name, cb_tree sname, cb_tree flag, cb_tree ref) 00630 { 00631 cb_tree switch_id; 00632 cb_tree value; 00633 00634 if (name == cb_error_node) { 00635 return; 00636 } 00637 if (sname == cb_error_node) { 00638 return; 00639 } 00640 if (CB_SYSTEM_NAME (sname)->category != CB_SWITCH_NAME) { 00641 cb_error_x (ref, _("Switch-name is expected '%s'"), CB_NAME (ref)); 00642 } else { 00643 switch_id = cb_int (CB_SYSTEM_NAME (sname)->token); 00644 value = cb_build_funcall_1 ("cob_get_switch", switch_id); 00645 if (flag == cb_int0) { 00646 value = cb_build_negation (value); 00647 } 00648 cb_build_constant (name, value); 00649 } 00650 } 00651 00652 cb_tree 00653 cb_build_section_name (cb_tree name, int sect_or_para) 00654 { 00655 cb_tree x; 00656 00657 if (name == cb_error_node) { 00658 return cb_error_node; 00659 } 00660 00661 if (CB_REFERENCE (name)->word->count > 0) { 00662 x = CB_VALUE (CB_REFERENCE (name)->word->items); 00663 /* Used as a non-label name or used as a section name. 00664 Duplicate paragraphs are allowed if not referenced; 00665 Checked in typeck.c */ 00666 if (!CB_LABEL_P (x) || sect_or_para == 0 00667 || (sect_or_para && CB_LABEL_P (x) && CB_LABEL (x)->is_section)) { 00668 redefinition_error (name); 00669 return cb_error_node; 00670 } 00671 } 00672 00673 return name; 00674 } 00675 00676 cb_tree 00677 cb_build_assignment_name (struct cb_file *cfile, cb_tree name) 00678 { 00679 const char *s; 00680 const char *p; 00681 00682 if (name == cb_error_node) { 00683 return cb_error_node; 00684 } 00685 00686 switch (CB_TREE_TAG (name)) { 00687 case CB_TAG_LITERAL: 00688 if (strcmp ((char *)(CB_LITERAL(name)->data), "$#@DUMMY@#$") == 0) { 00689 cfile->special = 2; 00690 } 00691 return name; 00692 00693 case CB_TAG_REFERENCE: 00694 s = CB_REFERENCE (name)->word->name; 00695 if (strcasecmp (s, "KEYBOARD") == 0) { 00696 s = "#DUMMY#"; 00697 cfile->special = 1; 00698 return cb_build_alphanumeric_literal ((ucharptr)s, strlen (s)); 00699 } 00700 switch (cb_assign_clause) { 00701 case CB_ASSIGN_COBOL2002: 00702 /* TODO */ 00703 return cb_error_node; 00704 00705 case CB_ASSIGN_MF: 00706 if (cfile->external_assign) { 00707 p = strrchr (s, '-'); 00708 if (p) { 00709 s = p + 1; 00710 } 00711 return cb_build_alphanumeric_literal ((ucharptr)s, strlen (s)); 00712 } 00713 current_program->reference_list = 00714 cb_list_add (current_program->reference_list, name); 00715 return name; 00716 00717 case CB_ASSIGN_IBM: 00718 /* check organization */ 00719 if (strncmp (s, "S-", 2) == 0 || 00720 strncmp (s, "AS-", 3) == 0) { 00721 goto org; 00722 } 00723 /* skip the device label if exists */ 00724 if ((p = strchr (s, '-')) != NULL) { 00725 s = p + 1; 00726 } 00727 /* check organization again */ 00728 if (strncmp (s, "S-", 2) == 0 || 00729 strncmp (s, "AS-", 3) == 0) { 00730 org: 00731 /* skip it for now */ 00732 s = strchr (s, '-') + 1; 00733 } 00734 /* convert the name into literal */ 00735 return cb_build_alphanumeric_literal ((ucharptr)s, strlen (s)); 00736 } 00737 00738 default: 00739 return cb_error_node; 00740 } 00741 } 00742 00743 cb_tree 00744 cb_build_index (cb_tree x, cb_tree values, int indexed_by, struct cb_field *qual) 00745 { 00746 struct cb_field *f; 00747 00748 f = CB_FIELD (cb_build_field (x)); 00749 f->usage = CB_USAGE_INDEX; 00750 cb_validate_field (f); 00751 if (values) { 00752 f->values = cb_list_init (values); 00753 } 00754 if (qual) { 00755 f->index_qual = qual; 00756 } 00757 f->flag_indexed_by = indexed_by; 00758 current_program->working_storage = cb_field_add (current_program->working_storage, f); 00759 return x; 00760 } 00761 00762 cb_tree 00763 cb_build_identifier (cb_tree x) 00764 { 00765 struct cb_reference *r; 00766 struct cb_field *f; 00767 struct cb_field *p; 00768 const char *name; 00769 cb_tree v; 00770 cb_tree e1; 00771 cb_tree e2; 00772 cb_tree l; 00773 cb_tree sub; 00774 int offset; 00775 int length; 00776 int n; 00777 00778 if (x == cb_error_node) { 00779 return cb_error_node; 00780 } 00781 00782 r = CB_REFERENCE (x); 00783 name = r->word->name; 00784 00785 /* resolve reference */ 00786 v = cb_ref (x); 00787 if (v == cb_error_node) { 00788 return cb_error_node; 00789 } 00790 00791 /* check if it is a data name */ 00792 if (!CB_FIELD_P (v)) { 00793 if (r->subs) { 00794 cb_error_x (x, _("'%s' cannot be subscripted"), name); 00795 return cb_error_node; 00796 } 00797 if (r->offset) { 00798 cb_error_x (x, _("'%s' cannot be reference modified"), name); 00799 return cb_error_node; 00800 } 00801 return x; 00802 } 00803 f = CB_FIELD (v); 00804 00805 /* BASED check */ 00806 if (CB_EXCEPTION_ENABLE (COB_EC_BOUND_PTR)) { 00807 for (p = f; p->parent; p = p->parent) { 00808 ; 00809 } 00810 if (current_statement) { 00811 if (p->flag_item_based || 00812 (f->storage == CB_STORAGE_LINKAGE && 00813 !p->flag_is_pdiv_parm)) { 00814 current_statement->null_check = cb_build_funcall_2 ( 00815 "cob_check_based", 00816 cb_build_address (cb_build_field_reference (p, NULL)), 00817 cb_build_string0 ((ucharptr)name)); 00818 } 00819 } 00820 } 00821 00822 /* check the number of subscripts */ 00823 if (!r->all && cb_list_length (r->subs) != f->indexes) { 00824 switch (f->indexes) { 00825 case 0: 00826 cb_error_x (x, _("'%s' cannot be subscripted"), name); 00827 return cb_error_node; 00828 case 1: 00829 cb_error_x (x, _("'%s' requires 1 subscript"), name); 00830 return cb_error_node; 00831 default: 00832 cb_error_x (x, _("'%s' requires %d subscripts"), name, f->indexes); 00833 return cb_error_node; 00834 } 00835 } 00836 00837 /* subscript check */ 00838 if (!r->all && r->subs) { 00839 l = r->subs; 00840 for (p = f; p; p = p->parent) { 00841 if (p->flag_occurs) { 00842 sub = cb_check_integer_value (CB_VALUE (l)); 00843 00844 l = CB_CHAIN (l); 00845 00846 if (sub == cb_error_node) { 00847 continue; 00848 } 00849 00850 /* compile-time check */ 00851 if (CB_LITERAL_P (sub)) { 00852 n = cb_get_int (sub); 00853 if (n < 1 || n > p->occurs_max) { 00854 cb_error_x (x, _("Subscript of '%s' out of bounds: %d"), 00855 name, n); 00856 } 00857 } 00858 00859 /* run-time check */ 00860 if (CB_EXCEPTION_ENABLE (COB_EC_BOUND_SUBSCRIPT)) { 00861 if (p->occurs_depending) { 00862 e1 = cb_build_funcall_4 ("cob_check_odo", 00863 cb_build_cast_integer (p->occurs_depending), 00864 cb_int (p->occurs_min), 00865 cb_int (p->occurs_max), 00866 cb_build_string0 00867 ((ucharptr)(cb_field (p->occurs_depending)->name))); 00868 e2 = cb_build_funcall_4 ("cob_check_subscript", 00869 cb_build_cast_integer (sub), 00870 cb_int1, 00871 cb_build_cast_integer (p->occurs_depending), 00872 cb_build_string0 ((ucharptr)name)); 00873 r->check = cb_list_add (r->check, e1); 00874 r->check = cb_list_add (r->check, e2); 00875 } else { 00876 if (!CB_LITERAL_P (sub)) { 00877 e1 = cb_build_funcall_4 ("cob_check_subscript", 00878 cb_build_cast_integer (sub), 00879 cb_int1, 00880 cb_int (p->occurs_max), 00881 cb_build_string0 ((ucharptr)name)); 00882 r->check = cb_list_add (r->check, e1); 00883 } 00884 } 00885 } 00886 } 00887 } 00888 } 00889 00890 /* reference modification check */ 00891 if (r->offset) { 00892 /* compile-time check */ 00893 if (CB_LITERAL_P (r->offset)) { 00894 offset = cb_get_int (r->offset); 00895 if (offset < 1 || offset > f->size) { 00896 cb_error_x (x, _("Offset of '%s' out of bounds: %d"), name, offset); 00897 } else if (r->length && CB_LITERAL_P (r->length)) { 00898 length = cb_get_int (r->length); 00899 if (length < 1 || length > f->size - offset + 1) { 00900 cb_error_x (x, _("Length of '%s' out of bounds: %d"), 00901 name, length); 00902 } 00903 } 00904 } 00905 00906 /* run-time check */ 00907 if (CB_EXCEPTION_ENABLE (COB_EC_BOUND_REF_MOD)) { 00908 if (!CB_LITERAL_P (r->offset) 00909 || (r->length && !CB_LITERAL_P (r->length))) { 00910 e1 = cb_build_funcall_4 ("cob_check_ref_mod", 00911 cb_build_cast_integer (r->offset), 00912 r->length ? cb_build_cast_integer (r->length) : 00913 cb_int1, cb_int (f->size), 00914 cb_build_string0 ((ucharptr)f->name)); 00915 r->check = cb_list_add (r->check, e1); 00916 } 00917 } 00918 } 00919 00920 if (f->storage == CB_STORAGE_CONSTANT) { 00921 return CB_VALUE (f->values); 00922 } 00923 00924 return x; 00925 } 00926 00927 static cb_tree 00928 cb_build_length_1 (cb_tree x) 00929 { 00930 struct cb_field *f; 00931 cb_tree e; 00932 cb_tree size; 00933 00934 f = CB_FIELD (cb_ref (x)); 00935 00936 if (cb_field_variable_size (f) == NULL) { 00937 /* constant size */ 00938 return cb_int (cb_field_size (x)); 00939 } else { 00940 /* variable size */ 00941 e = NULL; 00942 for (f = f->children; f; f = f->sister) { 00943 size = cb_build_length_1 (cb_build_field_reference (f, x)); 00944 if (f->occurs_depending) { 00945 size = cb_build_binary_op (size, '*', f->occurs_depending); 00946 } else if (f->occurs_max > 1) { 00947 size = cb_build_binary_op (size, '*', cb_int (f->occurs_max)); 00948 } 00949 e = e ? cb_build_binary_op (e, '+', size) : size; 00950 } 00951 return e; 00952 } 00953 } 00954 00955 cb_tree 00956 cb_build_const_length (cb_tree x) 00957 { 00958 struct cb_field *f; 00959 char buff[64]; 00960 00961 if (x == cb_error_node) { 00962 return cb_error_node; 00963 } 00964 if (CB_REFERENCE_P (x) && cb_ref (x) == cb_error_node) { 00965 return cb_error_node; 00966 } 00967 00968 memset (buff, 0, sizeof (buff)); 00969 f = CB_FIELD (cb_ref (x)); 00970 if (f->flag_any_length) { 00971 cb_error (_("ANY LENGTH item not allowed here")); 00972 return cb_error_node; 00973 } 00974 if (f->level == 88) { 00975 cb_error (_("88 level item not allowed here")); 00976 return cb_error_node; 00977 } 00978 if (!f->flag_is_verified) { 00979 cb_validate_field (f); 00980 } 00981 sprintf (buff, "%d", f->memory_size); 00982 return cb_build_numeric_literal (0, (ucharptr)buff, 0); 00983 } 00984 00985 cb_tree 00986 cb_build_length (cb_tree x) 00987 { 00988 struct cb_field *f; 00989 struct cb_literal *l; 00990 cb_tree temp; 00991 char buff[64]; 00992 00993 if (x == cb_error_node) { 00994 return cb_error_node; 00995 } 00996 if (CB_REFERENCE_P (x) && cb_ref (x) == cb_error_node) { 00997 return cb_error_node; 00998 } 00999 01000 memset (buff, 0, sizeof (buff)); 01001 if (CB_LITERAL_P (x)) { 01002 l = CB_LITERAL (x); 01003 sprintf (buff, "%d", (int)l->size); 01004 return cb_build_numeric_literal (0, (ucharptr)buff, 0); 01005 } 01006 if (CB_REF_OR_FIELD_P (x)) { 01007 f = CB_FIELD (cb_ref (x)); 01008 if (f->flag_any_length) { 01009 return cb_build_any_intrinsic (cb_list_init (x)); 01010 } 01011 if (cb_field_variable_size (f) == NULL) { 01012 sprintf (buff, "%d", cb_field_size (x)); 01013 return cb_build_numeric_literal (0, (ucharptr)buff, 0); 01014 } 01015 } 01016 if (CB_INTRINSIC_P (x)) { 01017 return cb_build_any_intrinsic (cb_list_init (x)); 01018 } 01019 temp = cb_build_index (cb_build_filler (), NULL, 0, NULL); 01020 CB_FIELD (cb_ref (temp))->usage = CB_USAGE_LENGTH; 01021 CB_FIELD (cb_ref (temp))->count++; 01022 cb_emit (cb_build_assign (temp, cb_build_length_1 (x))); 01023 return temp; 01024 } 01025 01026 cb_tree 01027 cb_build_address (cb_tree x) 01028 { 01029 if (x == cb_error_node || 01030 (CB_REFERENCE_P (x) && cb_ref (x) == cb_error_node)) { 01031 return cb_error_node; 01032 } 01033 01034 return cb_build_cast_address (x); 01035 } 01036 01037 cb_tree 01038 cb_build_ppointer (cb_tree x) 01039 { 01040 struct cb_field *f; 01041 01042 if (x == cb_error_node || 01043 (CB_REFERENCE_P (x) && cb_ref (x) == cb_error_node)) { 01044 return cb_error_node; 01045 } 01046 01047 if (CB_REFERENCE_P (x)) { 01048 f = cb_field (cb_ref(x)); 01049 f->count++; 01050 } 01051 return cb_build_cast_ppointer (x); 01052 } 01053 01054 /* validate program */ 01055 01056 static int 01057 get_value (cb_tree x) 01058 { 01059 if (x == cb_space) { 01060 return ' '; 01061 } else if (x == cb_zero) { 01062 return '0'; 01063 } else if (x == cb_quote) { 01064 return '"'; 01065 } else if (x == cb_norm_low) { 01066 return 0; 01067 } else if (x == cb_norm_high) { 01068 return 255; 01069 } else if (x == cb_null) { 01070 return 0; 01071 } else if (CB_TREE_CLASS (x) == CB_CLASS_NUMERIC) { 01072 return cb_get_int (x) - 1; 01073 } else { 01074 return CB_LITERAL (x)->data[0]; 01075 } 01076 } 01077 01078 void 01079 cb_validate_program_environment (struct cb_program *prog) 01080 { 01081 cb_tree x; 01082 cb_tree y; 01083 cb_tree l; 01084 cb_tree ls; 01085 struct cb_alphabet_name *ap; 01086 unsigned char *data; 01087 size_t dupls; 01088 size_t unvals; 01089 size_t count; 01090 int lower; 01091 int upper; 01092 int size; 01093 int n; 01094 int i; 01095 int lastval; 01096 int values[256]; 01097 01098 /* Check ALPHABET clauses */ 01099 for (l = current_program->alphabet_name_list; l; l = CB_CHAIN (l)) { 01100 ap = CB_ALPHABET_NAME (CB_VALUE (l)); 01101 if (ap->type != CB_ALPHABET_CUSTOM) { 01102 continue; 01103 } 01104 ap->low_val_char = 0; 01105 ap->high_val_char = 255; 01106 dupls = 0; 01107 unvals = 0; 01108 count = 0; 01109 lastval = 0; 01110 for (n = 0; n < 256; n++) { 01111 values[n] = -1; 01112 } 01113 for (y = ap->custom_list; y; y = CB_CHAIN (y)) { 01114 if (count > 255) { 01115 unvals = 1; 01116 break; 01117 } 01118 x = CB_VALUE (y); 01119 if (CB_PAIR_P (x)) { 01120 /* X THRU Y */ 01121 lower = get_value (CB_PAIR_X (x)); 01122 upper = get_value (CB_PAIR_Y (x)); 01123 lastval = upper; 01124 if (!count) { 01125 ap->low_val_char = lower; 01126 } 01127 if (lower < 0 || lower > 255) { 01128 unvals = 1; 01129 continue; 01130 } 01131 if (upper < 0 || upper > 255) { 01132 unvals = 1; 01133 continue; 01134 } 01135 if (lower <= upper) { 01136 for (i = lower; i <= upper; i++) { 01137 if (values[i] != -1) { 01138 dupls = 1; 01139 } 01140 values[i] = i; 01141 count++; 01142 } 01143 } else { 01144 for (i = lower; i >= upper; i--) { 01145 if (values[i] != -1) { 01146 dupls = 1; 01147 } 01148 values[i] = i; 01149 count++; 01150 } 01151 } 01152 } else if (CB_LIST_P (x)) { 01153 /* X ALSO Y ... */ 01154 if (!count) { 01155 ap->low_val_char = get_value (CB_VALUE (x)); 01156 } 01157 for (ls = x; ls; ls = CB_CHAIN (ls)) { 01158 n = get_value (CB_VALUE (ls)); 01159 if (!CB_CHAIN (ls)) { 01160 lastval = n; 01161 } 01162 if (n < 0 || n > 255) { 01163 unvals = 1; 01164 continue; 01165 } 01166 if (values[n] != -1) { 01167 dupls = 1; 01168 } 01169 values[n] = n; 01170 count++; 01171 } 01172 } else { 01173 /* literal */ 01174 if (CB_TREE_CLASS (x) == CB_CLASS_NUMERIC) { 01175 n = get_value (x); 01176 lastval = n; 01177 if (!count) { 01178 ap->low_val_char = n; 01179 } 01180 if (n < 0 || n > 255) { 01181 unvals = 1; 01182 continue; 01183 } 01184 if (values[n] != -1) { 01185 dupls = 1; 01186 } 01187 values[n] = n; 01188 count++; 01189 } else if (CB_LITERAL_P (x)) { 01190 size = (int)CB_LITERAL (x)->size; 01191 data = CB_LITERAL (x)->data; 01192 if (!count) { 01193 ap->low_val_char = data[0]; 01194 } 01195 lastval = data[size - 1]; 01196 for (i = 0; i < size; i++) { 01197 n = data[i]; 01198 if (values[n] != -1) { 01199 dupls = 1; 01200 } 01201 values[n] = n; 01202 count++; 01203 } 01204 } else { 01205 n = get_value (x); 01206 lastval = n; 01207 if (!count) { 01208 ap->low_val_char = n; 01209 } 01210 if (n < 0 || n > 255) { 01211 unvals = 1; 01212 continue; 01213 } 01214 if (values[n] != -1) { 01215 dupls = 1; 01216 } 01217 values[n] = n; 01218 count++; 01219 } 01220 } 01221 } 01222 if (dupls || unvals) { 01223 if (dupls) { 01224 cb_error_x (l, _("Duplicate character values in alphabet '%s'"), 01225 cb_name (CB_VALUE(l))); 01226 } 01227 if (unvals) { 01228 cb_error_x (l, _("Invalid character values in alphabet '%s'"), 01229 cb_name (CB_VALUE(l))); 01230 } 01231 ap->low_val_char = 0; 01232 ap->high_val_char = 255; 01233 continue; 01234 } 01235 /* Calculate HIGH-VALUE */ 01236 /* If all 256 values have been specified, HIGH-VALUE is the last one */ 01237 /* Otherwise if HIGH-VALUE has been specified, find the highest */ 01238 /* value that has not been used */ 01239 if (count == 256) { 01240 ap->high_val_char = lastval; 01241 } else if (values[255] != -1) { 01242 for (n = 254; n >= 0; n--) { 01243 if (values[n] == -1) { 01244 ap->high_val_char = n; 01245 break; 01246 } 01247 } 01248 } 01249 } 01250 /* Rest HIGH/LOW-VALUES */ 01251 cb_low = cb_norm_low; 01252 cb_high = cb_norm_high; 01253 /* resolve the program collating sequence */ 01254 if (!prog->collating_sequence) { 01255 return; 01256 } 01257 x = cb_ref (prog->collating_sequence); 01258 /* RXWRXW 01259 if (x == cb_error_node) { 01260 prog->collating_sequence = NULL; 01261 return; 01262 } 01263 */ 01264 if (!CB_ALPHABET_NAME_P (x)) { 01265 cb_error_x (prog->collating_sequence, _("'%s' not alphabet name"), 01266 cb_name (prog->collating_sequence)); 01267 prog->collating_sequence = NULL; 01268 return; 01269 } 01270 if (CB_ALPHABET_NAME (x)->type != CB_ALPHABET_CUSTOM) { 01271 return; 01272 } 01273 if (CB_ALPHABET_NAME (x)->low_val_char) { 01274 cb_low = cb_build_alphanumeric_literal ((ucharptr)"\0", 1); 01275 CB_LITERAL(cb_low)->data[0] = CB_ALPHABET_NAME (x)->low_val_char; 01276 CB_LITERAL(cb_low)->all = 1; 01277 } 01278 if (CB_ALPHABET_NAME (x)->high_val_char != 255){ 01279 cb_high = cb_build_alphanumeric_literal ((ucharptr)"\0", 1); 01280 CB_LITERAL(cb_high)->data[0] = CB_ALPHABET_NAME (x)->high_val_char; 01281 CB_LITERAL(cb_high)->all = 1; 01282 } 01283 } 01284 01285 void 01286 cb_validate_program_data (struct cb_program *prog) 01287 { 01288 cb_tree l; 01289 cb_tree x; 01290 cb_tree assign; 01291 struct cb_field *p; 01292 struct cb_file *f; 01293 unsigned char *c; 01294 01295 for (l = current_program->file_list; l; l = CB_CHAIN (l)) { 01296 f = CB_FILE (CB_VALUE (l)); 01297 if (!f->finalized) { 01298 finalize_file (f, NULL); 01299 } 01300 } 01301 /* build undeclared assignment name now */ 01302 if (cb_assign_clause == CB_ASSIGN_MF) { 01303 for (l = current_program->file_list; l; l = CB_CHAIN (l)) { 01304 assign = CB_FILE (CB_VALUE (l))->assign; 01305 if (!assign) { 01306 continue; 01307 } 01308 if (CB_REFERENCE_P (assign)) { 01309 for (x = current_program->file_list; x; x = CB_CHAIN (x)) { 01310 if (!strcmp (CB_FILE (CB_VALUE (x))->name, 01311 CB_REFERENCE (assign)->word->name)) { 01312 redefinition_error (assign); 01313 } 01314 } 01315 p = check_level_78 (CB_REFERENCE (assign)->word->name); 01316 if (p) { 01317 c = (unsigned char *)CB_LITERAL(CB_VALUE(p->values))->data; 01318 assign = CB_TREE (build_literal (CB_CATEGORY_ALPHANUMERIC, c, strlen ((char *)c))); 01319 CB_FILE (CB_VALUE (l))->assign = assign; 01320 } 01321 } 01322 if (CB_REFERENCE_P (assign) && CB_REFERENCE (assign)->word->count == 0) { 01323 if (cb_warn_implicit_define) { 01324 cb_warning (_("'%s' will be implicitly defined"), CB_NAME (assign)); 01325 } 01326 x = cb_build_implicit_field (assign, COB_SMALL_BUFF); 01327 p = current_program->working_storage; 01328 CB_FIELD (x)->count++; 01329 if (p) { 01330 while (p->sister) { 01331 p = p->sister; 01332 } 01333 p->sister = CB_FIELD (x); 01334 } else { 01335 current_program->working_storage = CB_FIELD (x); 01336 } 01337 } 01338 if (CB_REFERENCE_P (assign)) { 01339 x = cb_ref (assign); 01340 if (CB_FIELD_P (x) && CB_FIELD (x)->level == 88) { 01341 cb_error_x (assign, _("ASSIGN data item '%s' invalid"), CB_NAME (assign)); 01342 } 01343 } 01344 } 01345 } 01346 01347 if (prog->cursor_pos) { 01348 x = cb_ref (prog->cursor_pos); 01349 if (x == cb_error_node) { 01350 prog->cursor_pos = NULL; 01351 } else if (CB_FIELD(x)->size != 6 && CB_FIELD(x)->size != 4) { 01352 cb_error_x (prog->cursor_pos, _("'%s' CURSOR is not 4 or 6 characters long"), 01353 cb_name (prog->cursor_pos)); 01354 prog->cursor_pos = NULL; 01355 } 01356 } 01357 if (prog->crt_status) { 01358 x = cb_ref (prog->crt_status); 01359 if (x == cb_error_node) { 01360 prog->crt_status = NULL; 01361 } else if (CB_FIELD(x)->size != 4) { 01362 cb_error_x (prog->crt_status, _("'%s' CRT STATUS is not 4 characters long"), 01363 cb_name (prog->crt_status)); 01364 prog->crt_status = NULL; 01365 } 01366 } else { 01367 l = cb_build_reference ("COB-CRT-STATUS"); 01368 p = CB_FIELD (cb_build_field (l)); 01369 p->usage = CB_USAGE_DISPLAY; 01370 p->pic = CB_PICTURE (cb_build_picture ("9(4)")); 01371 cb_validate_field (p); 01372 p->flag_no_init = 1; 01373 /* Do not initialize/bump ref count here 01374 p->values = cb_list_init (cb_zero); 01375 p->count++; 01376 */ 01377 current_program->working_storage = 01378 cb_field_add (current_program->working_storage, p); 01379 prog->crt_status = l; 01380 /* RXWRXW - Maybe better 01381 prog->crt_status = cb_build_index (cb_build_reference ("COB-CRT-STATUS"), cb_zero, 0, NULL); 01382 */ 01383 } 01384 01385 /* resolve all references so far */ 01386 for (l = cb_list_reverse (prog->reference_list); l; l = CB_CHAIN (l)) { 01387 cb_ref (CB_VALUE (l)); 01388 } 01389 for (l = current_program->file_list; l; l = CB_CHAIN (l)) { 01390 f = CB_FILE (CB_VALUE (l)); 01391 if (f->record_depending && f->record_depending != cb_error_node) { 01392 x = f->record_depending; 01393 if (cb_ref (x) != cb_error_node) { 01394 /* RXW - This breaks old legacy programs 01395 if (CB_REF_OR_FIELD_P(x)) { 01396 p = cb_field (x); 01397 switch (p->storage) { 01398 case CB_STORAGE_WORKING: 01399 case CB_STORAGE_LOCAL: 01400 case CB_STORAGE_LINKAGE: 01401 break; 01402 default: 01403 cb_error (_("RECORD DEPENDING item must be in WORKING/LOCAL/LINKAGE section")); 01404 } 01405 } else { 01406 */ 01407 if (!CB_REFERENCE_P(x) && !CB_FIELD_P(x)) { 01408 cb_error (_("Invalid RECORD DEPENDING item")); 01409 } 01410 } 01411 } 01412 } 01413 } 01414 01415 void 01416 cb_validate_program_body (struct cb_program *prog) 01417 { 01418 /* resolve all labels */ 01419 cb_tree l; 01420 cb_tree x; 01421 cb_tree v; 01422 01423 for (l = cb_list_reverse (prog->label_list); l; l = CB_CHAIN (l)) { 01424 x = CB_VALUE (l); 01425 v = cb_ref (x); 01426 if (CB_LABEL_P (v)) { 01427 CB_LABEL (v)->need_begin = 1; 01428 if (CB_REFERENCE (x)->length) { 01429 CB_LABEL (v)->need_return = 1; 01430 } 01431 } else if (v != cb_error_node) { 01432 cb_error_x (x, _("'%s' not procedure name"), cb_name (x)); 01433 } 01434 } 01435 01436 prog->file_list = cb_list_reverse (prog->file_list); 01437 prog->exec_list = cb_list_reverse (prog->exec_list); 01438 } 01439 01440 /* 01441 * Expressions 01442 */ 01443 01444 static void 01445 cb_expr_init (void) 01446 { 01447 static int initialized = 0; 01448 01449 if (initialized == 0) { 01450 /* init priority talble */ 01451 expr_prio['x'] = 0; 01452 expr_prio['^'] = 1; 01453 expr_prio['*'] = 2; 01454 expr_prio['/'] = 2; 01455 expr_prio['+'] = 3; 01456 expr_prio['-'] = 3; 01457 expr_prio['='] = 4; 01458 expr_prio['~'] = 4; 01459 expr_prio['<'] = 4; 01460 expr_prio['>'] = 4; 01461 expr_prio['['] = 4; 01462 expr_prio[']'] = 4; 01463 expr_prio['!'] = 5; 01464 expr_prio['&'] = 6; 01465 expr_prio['|'] = 7; 01466 expr_prio[')'] = 8; 01467 expr_prio['('] = 9; 01468 expr_prio[0] = 10; 01469 /* init stack */ 01470 expr_stack_size = START_STACK_SIZE; 01471 expr_stack = cobc_malloc (sizeof (struct expr_node) * START_STACK_SIZE); 01472 expr_stack[0].token = 0; /* dummy */ 01473 expr_stack[1].token = 0; /* dummy */ 01474 expr_stack[2].token = 0; /* dummy */ 01475 initialized = 1; 01476 } 01477 01478 expr_op = 0; 01479 expr_lh = NULL; 01480 expr_index = 3; 01481 } 01482 01483 static int 01484 expr_reduce (int token) 01485 { 01486 /* Example: 01487 * index: -3 -2 -1 0 01488 * token: 'x' '*' 'x' '+' ... 01489 */ 01490 01491 int op; 01492 01493 while (expr_prio[TOKEN (-2)] <= expr_prio[token]) { 01494 /* Reduce the expression depending on the last operator */ 01495 op = TOKEN (-2); 01496 switch (op) { 01497 case 'x': 01498 return 0; 01499 01500 case '+': 01501 case '-': 01502 case '*': 01503 case '/': 01504 case '^': 01505 /* Arithmetic operators: 'x' op 'x' */ 01506 if (TOKEN (-1) != 'x' || TOKEN (-3) != 'x') { 01507 return -1; 01508 } 01509 TOKEN (-3) = 'x'; 01510 VALUE (-3) = cb_build_binary_op (VALUE (-3), op, VALUE (-1)); 01511 expr_index -= 2; 01512 break; 01513 01514 case '!': 01515 /* Negation: '!' 'x' */ 01516 if (TOKEN (-1) != 'x') { 01517 return -1; 01518 } 01519 /* 'x' '=' 'x' '|' '!' 'x' */ 01520 if (expr_lh) { 01521 if (CB_TREE_CLASS (VALUE (-1)) != CB_CLASS_BOOLEAN) { 01522 VALUE (-1) = cb_build_binary_op (expr_lh, expr_op, VALUE (-1)); 01523 } 01524 } 01525 TOKEN (-2) = 'x'; 01526 VALUE (-2) = cb_build_negation (VALUE (-1)); 01527 expr_index -= 1; 01528 break; 01529 01530 case '&': 01531 case '|': 01532 /* Logical AND/OR: 'x' op 'x' */ 01533 if (TOKEN (-1) != 'x' || TOKEN (-3) != 'x') { 01534 return -1; 01535 } 01536 /* 'x' '=' 'x' '|' 'x' */ 01537 if (expr_lh) { 01538 if (CB_TREE_CLASS (VALUE (-1)) != CB_CLASS_BOOLEAN) { 01539 VALUE (-1) = cb_build_binary_op (expr_lh, expr_op, VALUE (-1)); 01540 } 01541 if (CB_TREE_CLASS (VALUE (-3)) != CB_CLASS_BOOLEAN) { 01542 VALUE (-3) = cb_build_binary_op (expr_lh, expr_op, VALUE (-3)); 01543 } 01544 } 01545 /* warning for complex expressions without explicit parentheses 01546 (i.e., "a OR b AND c" or "a AND b OR c") */ 01547 if (cb_warn_parentheses && op == '|') { 01548 if ((CB_BINARY_OP_P (VALUE (-3)) 01549 && CB_BINARY_OP (VALUE (-3))->op == '&') 01550 || (CB_BINARY_OP_P (VALUE (-1)) 01551 && CB_BINARY_OP (VALUE (-1))->op == '&')) { 01552 cb_warning (_("Suggest parentheses around AND within OR")); 01553 } 01554 } 01555 TOKEN (-3) = 'x'; 01556 VALUE (-3) = cb_build_binary_op (VALUE (-3), op, VALUE (-1)); 01557 expr_index -= 2; 01558 break; 01559 01560 case '(': 01561 case ')': 01562 return 0; 01563 01564 default: 01565 /* Relational operators */ 01566 if (TOKEN (-1) != 'x') { 01567 return -1; 01568 } 01569 switch (TOKEN (-3)) { 01570 case 'x': 01571 /* Simple condition: 'x' op 'x' */ 01572 if (VALUE (-3) == cb_error_node || VALUE (-1) == cb_error_node) { 01573 VALUE (-3) = cb_error_node; 01574 } else { 01575 expr_lh = VALUE (-3); 01576 if (CB_REF_OR_FIELD_P (expr_lh)) { 01577 if (cb_field (expr_lh)->level == 88) { 01578 VALUE (-3) = cb_error_node; 01579 return -1; 01580 } 01581 } 01582 if (CB_REF_OR_FIELD_P (VALUE(-1))) { 01583 if (cb_field (VALUE(-1))->level == 88) { 01584 VALUE (-3) = cb_error_node; 01585 return -1; 01586 } 01587 } 01588 expr_op = op; 01589 TOKEN (-3) = 'x'; 01590 if (CB_TREE_CLASS (VALUE (-1)) != CB_CLASS_BOOLEAN) { 01591 VALUE (-3) = cb_build_binary_op (expr_lh, op, VALUE (-1)); 01592 } else { 01593 VALUE (-3) = VALUE (-1); 01594 } 01595 } 01596 expr_index -= 2; 01597 break; 01598 case '&': 01599 case '|': 01600 /* Complex condition: 'x' '=' 'x' '|' op 'x' */ 01601 if (VALUE (-1) == cb_error_node) { 01602 VALUE (-2) = cb_error_node; 01603 } else { 01604 expr_op = op; 01605 TOKEN (-2) = 'x'; 01606 if (CB_TREE_CLASS (VALUE (-1)) != CB_CLASS_BOOLEAN) { 01607 VALUE (-2) = cb_build_binary_op (expr_lh, op, VALUE (-1)); 01608 } else { 01609 VALUE (-2) = VALUE (-1); 01610 } 01611 } 01612 expr_index -= 1; 01613 break; 01614 default: 01615 return -1; 01616 } 01617 break; 01618 } 01619 } 01620 01621 /* handle special case "op OR x AND" */ 01622 if (token == '&' && TOKEN (-2) == '|' && CB_TREE_CLASS (VALUE (-1)) != CB_CLASS_BOOLEAN) { 01623 TOKEN (-1) = 'x'; 01624 VALUE (-1) = cb_build_binary_op (expr_lh, expr_op, VALUE (-1)); 01625 } 01626 01627 return 0; 01628 } 01629 01630 static void 01631 cb_expr_shift_sign (const int op) 01632 { 01633 int have_not = 0; 01634 01635 if (TOKEN (-1) == '!') { 01636 have_not = 1; 01637 expr_index--; 01638 } 01639 expr_reduce ('='); 01640 if (TOKEN (-1) == 'x') { 01641 VALUE (-1) = cb_build_binary_op (VALUE (-1), op, cb_zero); 01642 if (have_not) { 01643 VALUE (-1) = cb_build_negation (VALUE (-1)); 01644 } 01645 } 01646 } 01647 01648 static void 01649 cb_expr_shift_class (const char *name) 01650 { 01651 int have_not = 0; 01652 01653 if (TOKEN (-1) == '!') { 01654 have_not = 1; 01655 expr_index--; 01656 } 01657 expr_reduce ('='); 01658 if (TOKEN (-1) == 'x') { 01659 VALUE (-1) = cb_build_funcall_1 (name, VALUE (-1)); 01660 if (have_not) { 01661 VALUE (-1) = cb_build_negation (VALUE (-1)); 01662 } 01663 } 01664 } 01665 01666 static void 01667 cb_expr_shift (int token, cb_tree value) 01668 { 01669 switch (token) { 01670 case 'x': 01671 /* sign ZERO condition */ 01672 if (value == cb_zero) { 01673 if (TOKEN (-1) == 'x' || TOKEN (-1) == '!') { 01674 cb_expr_shift_sign ('='); 01675 return; 01676 } 01677 } 01678 01679 /* class condition */ 01680 if (CB_REFERENCE_P (value) 01681 && CB_CLASS_NAME_P (cb_ref (value))) { 01682 cb_expr_shift_class (CB_CLASS_NAME (cb_ref (value))->cname); 01683 return; 01684 } 01685 01686 /* unary sign */ 01687 if ((TOKEN (-1) == '+' || TOKEN (-1) == '-') && TOKEN (-2) != 'x') { 01688 if (TOKEN (-1) == '-') { 01689 value = cb_build_binary_op (cb_zero, '-', value); 01690 } 01691 expr_index -= 1; 01692 } 01693 break; 01694 01695 case '(': 01696 /* 'x' op '(' --> '(' 'x' op */ 01697 switch (TOKEN (-1)) { 01698 case '=': 01699 case '~': 01700 case '<': 01701 case '>': 01702 case '[': 01703 case ']': 01704 expr_op = TOKEN (-1); 01705 if (TOKEN (-2) == 'x') { 01706 expr_lh = VALUE (-2); 01707 } 01708 } 01709 break; 01710 01711 case ')': 01712 /* enclose by parentheses */ 01713 expr_reduce (token); 01714 if (TOKEN (-2) == '(') { 01715 value = cb_build_parenthesis (VALUE (-1)); 01716 expr_index -= 2; 01717 cb_expr_shift ('x', value); 01718 return; 01719 } 01720 break; 01721 01722 default: 01723 /* '<' '|' '=' --> '[' */ 01724 /* '>' '|' '=' --> ']' */ 01725 if (token == '=' && TOKEN (-1) == '|' && (TOKEN (-2) == '<' || TOKEN (-2) == '>')) { 01726 token = (TOKEN (-2) == '<') ? '[' : ']'; 01727 expr_index -= 2; 01728 } 01729 01730 /* '!' '=' --> '~', etc. */ 01731 if (TOKEN (-1) == '!') { 01732 switch (token) { 01733 case '=': 01734 token = '~'; 01735 expr_index--; 01736 break; 01737 case '~': 01738 token = '='; 01739 expr_index--; 01740 break; 01741 case '<': 01742 token = ']'; 01743 expr_index--; 01744 break; 01745 case '>': 01746 token = '['; 01747 expr_index--; 01748 break; 01749 case '[': 01750 token = '>'; 01751 expr_index--; 01752 break; 01753 case ']': 01754 token = '<'; 01755 expr_index--; 01756 break; 01757 } 01758 } 01759 break; 01760 } 01761 01762 /* reduce */ 01763 expr_reduce (token); 01764 01765 /* allocate sufficient stack memory */ 01766 if (expr_index >= expr_stack_size) { 01767 expr_stack_size *= 2; 01768 expr_stack = cobc_realloc (expr_stack, sizeof (struct expr_node) * expr_stack_size); 01769 } 01770 01771 /* put on the stack */ 01772 TOKEN (0) = token; 01773 VALUE (0) = value; 01774 expr_index++; 01775 } 01776 01777 static void 01778 expr_expand (cb_tree *x) 01779 { 01780 struct cb_binary_op *p; 01781 01782 start: 01783 /* remove parenthesis */ 01784 if (CB_BINARY_OP_P (*x)) { 01785 p = CB_BINARY_OP (*x); 01786 if (p->op == '@') { 01787 *x = p->x; 01788 goto start; 01789 } 01790 expr_expand (&p->x); 01791 if (p->y) { 01792 expr_expand (&p->y); 01793 } 01794 } 01795 } 01796 01797 static cb_tree 01798 cb_expr_finish (void) 01799 { 01800 expr_reduce (0); /* reduce all */ 01801 01802 if (expr_index != 4) { 01803 cb_error (_("Invalid expression")); 01804 return cb_error_node; 01805 } 01806 01807 if (!expr_stack[3].value) { 01808 cb_error (_("Invalid expression")); 01809 return cb_error_node; 01810 } 01811 expr_expand (&expr_stack[3].value); 01812 if (expr_stack[3].token != 'x') { 01813 cb_error (_("Invalid expression")); 01814 return cb_error_node; 01815 } 01816 return expr_stack[3].value; 01817 } 01818 01819 cb_tree 01820 cb_build_expr (cb_tree list) 01821 { 01822 cb_tree l; 01823 /* RXW 01824 cb_tree x; 01825 */ 01826 int op; 01827 01828 cb_expr_init (); 01829 01830 for (l = list; l; l = CB_CHAIN (l)) { 01831 op = CB_PURPOSE_INT (l); 01832 switch (op) { 01833 case '9': /* NUMERIC */ 01834 cb_expr_shift_class ("cob_is_numeric"); 01835 break; 01836 case 'A': /* ALPHABETIC */ 01837 cb_expr_shift_class ("cob_is_alpha"); 01838 break; 01839 case 'L': /* ALPHABETIC_LOWER */ 01840 cb_expr_shift_class ("cob_is_lower"); 01841 break; 01842 case 'U': /* ALPHABETIC_UPPER */ 01843 cb_expr_shift_class ("cob_is_upper"); 01844 break; 01845 case 'P': /* POSITIVE */ 01846 cb_expr_shift_sign ('>'); 01847 break; 01848 case 'N': /* NEGATIVE */ 01849 cb_expr_shift_sign ('<'); 01850 break; 01851 case 'O': /* OMITTED */ 01852 current_statement->null_check = NULL; 01853 cb_expr_shift_class ("cob_is_omitted"); 01854 break; 01855 /* RXW 01856 case 'x': 01857 if (CB_VALUE (l) && CB_REFERENCE_P (CB_VALUE (l))) { 01858 x = CB_CHAIN (l); 01859 if (x && cb_field (CB_VALUE (l))->level == 88) { 01860 switch (CB_PURPOSE_INT (x)) { 01861 case '&': 01862 case '|': 01863 case '(': 01864 case ')': 01865 break; 01866 default: 01867 cb_error (_("Invalid condition")); 01868 break; 01869 } 01870 } 01871 } 01872 cb_expr_shift (op, CB_VALUE (l)); 01873 break; 01874 */ 01875 default: 01876 cb_expr_shift (op, CB_VALUE (l)); 01877 break; 01878 } 01879 } 01880 01881 return cb_expr_finish (); 01882 } 01883 01884 /* 01885 * Numerical operation 01886 */ 01887 01888 static cb_tree 01889 build_store_option (cb_tree x, cb_tree round_opt) 01890 { 01891 int opt = 0; 01892 01893 if (round_opt == cb_int1) { 01894 opt |= COB_STORE_ROUND; 01895 } 01896 01897 switch (CB_FIELD (cb_ref (x))->usage) { 01898 case CB_USAGE_COMP_5: 01899 case CB_USAGE_COMP_X: 01900 if (current_statement->handler1) { 01901 opt |= COB_STORE_KEEP_ON_OVERFLOW; 01902 } 01903 break; 01904 default: 01905 if (!cb_binary_truncate) { 01906 if (current_statement->handler1) { 01907 opt |= COB_STORE_KEEP_ON_OVERFLOW; 01908 } 01909 break; 01910 } 01911 01912 /* RXW Fixme - It seems as though we have NEVER implemented TRUNC, 01913 Code has always been wrong. Hmm. The following statement would 01914 activate what was intended but ... 01915 What should we do here? 01916 if (current_statement->handler1) { 01917 */ 01918 if (current_statement->handler_id) { 01919 opt |= COB_STORE_KEEP_ON_OVERFLOW; 01920 } else if (cb_binary_truncate) { 01921 opt |= COB_STORE_TRUNC_ON_OVERFLOW; 01922 } 01923 break; 01924 } 01925 01926 return cb_int (opt); 01927 } 01928 01929 static cb_tree 01930 decimal_alloc (void) 01931 { 01932 cb_tree x; 01933 01934 x = cb_build_decimal (current_program->decimal_index); 01935 current_program->decimal_index++; 01936 if (current_program->decimal_index > current_program->decimal_index_max) { 01937 current_program->decimal_index_max = current_program->decimal_index; 01938 } 01939 return x; 01940 } 01941 01942 static void 01943 decimal_free (void) 01944 { 01945 current_program->decimal_index--; 01946 } 01947 01948 static void 01949 decimal_compute (const int op, cb_tree x, cb_tree y) 01950 { 01951 const char *func; 01952 01953 switch (op) { 01954 case '+': 01955 func = "cob_decimal_add"; 01956 break; 01957 case '-': 01958 func = "cob_decimal_sub"; 01959 break; 01960 case '*': 01961 func = "cob_decimal_mul"; 01962 break; 01963 case '/': 01964 func = "cob_decimal_div"; 01965 break; 01966 case '^': 01967 func = "cob_decimal_pow"; 01968 break; 01969 default: 01970 fprintf (stderr, "Unexpected operation %d\n", op); 01971 ABORT (); 01972 } 01973 dpush (cb_build_funcall_2 (func, x, y)); 01974 } 01975 01976 static void 01977 decimal_expand (cb_tree d, cb_tree x) 01978 { 01979 struct cb_literal *l; 01980 struct cb_field *f; 01981 struct cb_binary_op *p; 01982 cb_tree t; 01983 01984 switch (CB_TREE_TAG (x)) { 01985 case CB_TAG_CONST: 01986 if (x == cb_zero) { 01987 dpush (cb_build_funcall_2 ("cob_decimal_set_int", d, cb_int0)); 01988 current_program->gen_decset = 1; 01989 } else { 01990 fprintf (stderr, "Unexpected constant expansion\n"); 01991 ABORT (); 01992 } 01993 break; 01994 case CB_TAG_LITERAL: 01995 /* set d, N */ 01996 l = CB_LITERAL (x); 01997 if (l->size < 10 && l->scale == 0) { 01998 dpush (cb_build_funcall_2 ("cob_decimal_set_int", d, cb_build_cast_integer (x))); 01999 current_program->gen_decset = 1; 02000 } else { 02001 dpush (cb_build_funcall_2 ("cob_decimal_set_field", d, x)); 02002 } 02003 break; 02004 case CB_TAG_REFERENCE: 02005 /* set d, X */ 02006 f = cb_field (x); 02007 /* check numeric */ 02008 if (CB_EXCEPTION_ENABLE (COB_EC_DATA_INCOMPATIBLE)) { 02009 if (f->usage == CB_USAGE_DISPLAY || f->usage == CB_USAGE_PACKED) { 02010 dpush (cb_build_funcall_2 ("cob_check_numeric", 02011 x, cb_build_string0 ((ucharptr)(f->name)))); 02012 } 02013 } 02014 02015 if (cb_fits_int (x)) { 02016 if (f->pic->have_sign) { 02017 dpush (cb_build_funcall_2 ("cob_decimal_set_int", d, cb_build_cast_integer (x))); 02018 current_program->gen_decset = 1; 02019 } else { 02020 dpush (cb_build_funcall_2 ("cob_decimal_set_uint", d, cb_build_cast_integer (x))); 02021 current_program->gen_udecset = 1; 02022 } 02023 } else { 02024 dpush (cb_build_funcall_2 ("cob_decimal_set_field", d, x)); 02025 } 02026 break; 02027 case CB_TAG_BINARY_OP: 02028 /* set d, X 02029 * set t, Y 02030 * OP d, t */ 02031 p = CB_BINARY_OP (x); 02032 t = decimal_alloc (); 02033 decimal_expand (d, p->x); 02034 decimal_expand (t, p->y); 02035 decimal_compute (p->op, d, t); 02036 decimal_free (); 02037 break; 02038 case CB_TAG_INTRINSIC: 02039 dpush (cb_build_funcall_2 ("cob_decimal_set_field", d, x)); 02040 break; 02041 default: 02042 fprintf (stderr, "Unexpected tree tag %d\n", CB_TREE_TAG (x)); 02043 ABORT (); 02044 } 02045 } 02046 02047 static void 02048 decimal_assign (cb_tree x, cb_tree d, cb_tree round_opt) 02049 { 02050 dpush (cb_build_funcall_3 ("cob_decimal_get_field", d, x, build_store_option (x, round_opt))); 02051 } 02052 02053 static cb_tree 02054 build_decimal_assign (cb_tree vars, int op, cb_tree val) 02055 { 02056 cb_tree l; 02057 cb_tree t; 02058 cb_tree s1 = NULL; 02059 cb_tree d; 02060 02061 d = decimal_alloc (); 02062 02063 /* set d, VAL */ 02064 decimal_expand (d, val); 02065 02066 if (op == 0) { 02067 for (l = vars; l; l = CB_CHAIN (l)) { 02068 /* set VAR, d */ 02069 decimal_assign (CB_VALUE (l), d, CB_PURPOSE (l)); 02070 s1 = cb_list_add (s1, cb_list_reverse (decimal_stack)); 02071 decimal_stack = NULL; 02072 } 02073 } else { 02074 t = decimal_alloc (); 02075 for (l = vars; l; l = CB_CHAIN (l)) { 02076 /* set t, VAR 02077 * OP t, d 02078 * set VAR, t 02079 */ 02080 decimal_expand (t, CB_VALUE (l)); 02081 decimal_compute (op, t, d); 02082 decimal_assign (CB_VALUE (l), t, CB_PURPOSE (l)); 02083 s1 = cb_list_add (s1, cb_list_reverse (decimal_stack)); 02084 decimal_stack = NULL; 02085 } 02086 decimal_free (); 02087 } 02088 02089 decimal_free (); 02090 return s1; 02091 } 02092 02093 void 02094 cb_emit_arithmetic (cb_tree vars, int op, cb_tree val) 02095 { 02096 cb_tree l; 02097 struct cb_field *f; 02098 02099 val = cb_check_numeric_value (val); 02100 if (op) { 02101 cb_list_map (cb_check_numeric_name, vars); 02102 } else { 02103 cb_list_map (cb_check_numeric_edited_name, vars); 02104 } 02105 02106 if (cb_validate_one (val)) { 02107 return; 02108 } 02109 if (cb_validate_list (vars)) { 02110 return; 02111 } 02112 02113 if (!CB_BINARY_OP_P (val)) { 02114 if (op == '+' || op == '-') { 02115 if (CB_EXCEPTION_ENABLE (COB_EC_DATA_INCOMPATIBLE) && 02116 (CB_REF_OR_FIELD_P (val))) { 02117 f = cb_field (val); 02118 if (f->usage == CB_USAGE_DISPLAY || 02119 f->usage == CB_USAGE_PACKED) { 02120 cb_emit (cb_build_funcall_2 ("cob_check_numeric", 02121 val, 02122 cb_build_string0 ((ucharptr)(f->name)))); 02123 } 02124 } 02125 for (l = vars; l; l = CB_CHAIN (l)) { 02126 if (CB_EXCEPTION_ENABLE (COB_EC_DATA_INCOMPATIBLE) && 02127 (CB_REF_OR_FIELD_P (CB_VALUE(l)))) { 02128 f = cb_field (CB_VALUE(l)); 02129 if (f->usage == CB_USAGE_DISPLAY || 02130 f->usage == CB_USAGE_PACKED) { 02131 cb_emit (cb_build_funcall_2 ("cob_check_numeric", 02132 CB_VALUE(l), 02133 cb_build_string0 ((ucharptr)(f->name)))); 02134 } 02135 } 02136 if (op == '+') { 02137 CB_VALUE (l) = cb_build_add (CB_VALUE (l), val, CB_PURPOSE (l)); 02138 } else { 02139 CB_VALUE (l) = cb_build_sub (CB_VALUE (l), val, CB_PURPOSE (l)); 02140 } 02141 } 02142 cb_emit_list (vars); 02143 return; 02144 } 02145 } 02146 02147 cb_emit (build_decimal_assign (vars, op, val)); 02148 } 02149 02150 /* 02151 * Condition 02152 */ 02153 02154 static cb_tree 02155 build_cond_88 (cb_tree x) 02156 { 02157 struct cb_field *f; 02158 cb_tree l; 02159 cb_tree t; 02160 cb_tree c1 = NULL; 02161 cb_tree c2; 02162 02163 f = cb_field (x); 02164 /* refer to parent's data storage */ 02165 x = cb_build_field_reference (f->parent, x); 02166 f->parent->count++; 02167 02168 /* build condition */ 02169 for (l = f->values; l; l = CB_CHAIN (l)) { 02170 t = CB_VALUE (l); 02171 if (CB_PAIR_P (t)) { 02172 /* VALUE THRU VALUE */ 02173 c2 = cb_build_binary_op (cb_build_binary_op (CB_PAIR_X (t), '[', x), 02174 '&', cb_build_binary_op (x, '[', CB_PAIR_Y (t))); 02175 } else { 02176 /* VALUE */ 02177 c2 = cb_build_binary_op (x, '=', t); 02178 } 02179 if (c1 == NULL) { 02180 c1 = c2; 02181 } else { 02182 c1 = cb_build_binary_op (c1, '|', c2); 02183 } 02184 } 02185 return c1; 02186 } 02187 02188 static cb_tree 02189 cb_build_optim_cond (struct cb_binary_op *p) 02190 { 02191 struct cb_field *f; 02192 struct cb_field *fy; 02193 const char *s; 02194 size_t n; 02195 02196 if (CB_REF_OR_FIELD_P (p->y)) { 02197 fy = cb_field (p->y); 02198 if (!fy->pic->have_sign && (fy->usage == CB_USAGE_BINARY || 02199 fy->usage == CB_USAGE_COMP_5 || 02200 fy->usage == CB_USAGE_COMP_X)) { 02201 return cb_build_funcall_2 ("cob_cmp_uint", p->x, cb_build_cast_integer (p->y)); 02202 } 02203 } 02204 if (CB_REF_OR_FIELD_P (p->x)) { 02205 f = cb_field (p->x); 02206 if (!f->pic->scale && f->usage == CB_USAGE_PACKED) { 02207 if (f->pic->digits < 10) { 02208 return cb_build_funcall_2 ("cob_cmp_packed_int", 02209 p->x, 02210 cb_build_cast_integer (p->y)); 02211 } else { 02212 return cb_build_funcall_2 ("cob_cmp_packed", 02213 p->x, 02214 cb_build_cast_integer (p->y)); 02215 } 02216 } 02217 if (!f->pic->scale && f->usage == CB_USAGE_DISPLAY && 02218 !f->flag_sign_leading && !f->flag_sign_separate) { 02219 if (cb_fits_int (p->x)) { 02220 if (!f->pic->have_sign) { 02221 return cb_build_funcall_3 ("cob_cmp_numdisp", 02222 cb_build_cast_address (p->x), 02223 cb_int (f->size), 02224 cb_build_cast_integer (p->y)); 02225 } else { 02226 return cb_build_funcall_3 ("cob_cmp_sign_numdisp", 02227 cb_build_cast_address (p->x), 02228 cb_int (f->size), 02229 cb_build_cast_integer (p->y)); 02230 } 02231 } else if (cb_fits_long_long (p->x)) { 02232 if (!f->pic->have_sign) { 02233 return cb_build_funcall_3 ("cob_cmp_long_numdisp", 02234 cb_build_cast_address (p->x), 02235 cb_int (f->size), 02236 cb_build_cast_integer (p->y)); 02237 } else { 02238 return cb_build_funcall_3 ("cob_cmp_long_sign_numdisp", 02239 cb_build_cast_address (p->x), 02240 cb_int (f->size), 02241 cb_build_cast_integer (p->y)); 02242 } 02243 } 02244 } 02245 if (!f->pic->scale && (f->usage == CB_USAGE_BINARY || 02246 f->usage == CB_USAGE_COMP_5 || 02247 f->usage == CB_USAGE_INDEX || 02248 f->usage == CB_USAGE_COMP_X)) { 02249 n = (f->size - 1) + (8 * (f->pic->have_sign ? 1 : 0)) + 02250 (16 * (f->flag_binary_swap ? 1 : 0)); 02251 #if defined(COB_NON_ALIGNED) && !defined(_MSC_VER) 02252 switch (f->size) { 02253 case 2: 02254 #ifdef COB_SHORT_BORK 02255 s = bin_compare_funcs[n]; 02256 break; 02257 #endif 02258 case 4: 02259 case 8: 02260 if (f->storage != CB_STORAGE_LINKAGE && 02261 f->indexes == 0 && (f->offset % f->size) == 0) { 02262 s = align_bin_compare_funcs[n]; 02263 } else { 02264 s = bin_compare_funcs[n]; 02265 } 02266 break; 02267 default: 02268 s = bin_compare_funcs[n]; 02269 break; 02270 } 02271 #else 02272 s = bin_compare_funcs[n]; 02273 #endif 02274 if (s) { 02275 return cb_build_funcall_2 (s, 02276 cb_build_cast_address (p->x), 02277 cb_build_cast_integer (p->y)); 02278 } 02279 } 02280 } 02281 return cb_build_funcall_2 ("cob_cmp_int", p->x, cb_build_cast_integer (p->y)); 02282 } 02283 02284 static int 02285 cb_chk_num_cond (cb_tree x, cb_tree y) 02286 { 02287 struct cb_field *fx; 02288 struct cb_field *fy; 02289 02290 if (!CB_REFERENCE_P (x) && !CB_FIELD_P (x)) { 02291 return 0; 02292 } 02293 if (!CB_REFERENCE_P (y) && !CB_FIELD_P (y)) { 02294 return 0; 02295 } 02296 if (CB_TREE_CATEGORY (x) != CB_CATEGORY_NUMERIC) { 02297 return 0; 02298 } 02299 if (CB_TREE_CATEGORY (y) != CB_CATEGORY_NUMERIC) { 02300 return 0; 02301 } 02302 if (CB_TREE_CLASS (x) != CB_CLASS_NUMERIC) { 02303 return 0; 02304 } 02305 if (CB_TREE_CLASS (y) != CB_CLASS_NUMERIC) { 02306 return 0; 02307 } 02308 fx = cb_field (x); 02309 fy = cb_field (y); 02310 if (fx->usage != CB_USAGE_DISPLAY) { 02311 return 0; 02312 } 02313 if (fy->usage != CB_USAGE_DISPLAY) { 02314 return 0; 02315 } 02316 if (fx->pic->have_sign || fy->pic->have_sign) { 02317 return 0; 02318 } 02319 if (fx->size != fy->size) { 02320 return 0; 02321 } 02322 if (fx->pic->scale != fy->pic->scale) { 02323 return 0; 02324 } 02325 return 1; 02326 } 02327 02328 static int 02329 cb_chk_alpha_cond (cb_tree x) 02330 { 02331 if (current_program->alphabet_name_list) { 02332 return 0; 02333 } 02334 if (CB_LITERAL_P (x)) { 02335 return 1; 02336 } 02337 if (!CB_REFERENCE_P (x) && !CB_FIELD_P (x)) { 02338 return 0; 02339 } 02340 if (CB_TREE_CATEGORY (x) != CB_CATEGORY_ALPHANUMERIC && 02341 CB_TREE_CATEGORY (x) != CB_CATEGORY_ALPHABETIC) { 02342 return 0; 02343 } 02344 if (cb_field_variable_size (cb_field (x))) { 02345 return 0; 02346 } 02347 if (cb_field_size (x) < 0) { 02348 return 0; 02349 } 02350 return 1; 02351 } 02352 02353 cb_tree 02354 cb_build_cond (cb_tree x) 02355 { 02356 int size1; 02357 int size2; 02358 struct cb_field *f; 02359 struct cb_binary_op *p; 02360 cb_tree d1; 02361 cb_tree d2; 02362 02363 switch (CB_TREE_TAG (x)) { 02364 case CB_TAG_CONST: 02365 case CB_TAG_FUNCALL: 02366 return x; 02367 case CB_TAG_REFERENCE: 02368 if (!CB_FIELD_P (cb_ref (x))) { 02369 return cb_build_cond (cb_ref (x)); 02370 } 02371 02372 f = cb_field (x); 02373 02374 /* level 88 condition */ 02375 if (f->level == 88) { 02376 /* We need to build a 88 condition at every occurrence 02377 instead of once at the beginning because a 88 item 02378 may be subscripted (i.e., it is not a constant tree). */ 02379 return cb_build_cond (build_cond_88 (x)); 02380 } 02381 02382 cb_error_x (x, _("Invalid expression")); 02383 return cb_error_node; 02384 case CB_TAG_BINARY_OP: 02385 p = CB_BINARY_OP (x); 02386 switch (p->op) { 02387 case '!': 02388 return cb_build_negation (cb_build_cond (p->x)); 02389 case '&': 02390 case '|': 02391 return cb_build_binary_op (cb_build_cond (p->x), p->op, cb_build_cond (p->y)); 02392 default: 02393 if (CB_INDEX_P (p->x) || CB_INDEX_P (p->y) 02394 || CB_TREE_CLASS (p->x) == CB_CLASS_POINTER 02395 || CB_TREE_CLASS (p->y) == CB_CLASS_POINTER) { 02396 x = cb_build_binary_op (p->x, '-', p->y); 02397 } else if (CB_BINARY_OP_P (p->x) || CB_BINARY_OP_P (p->y)) { 02398 /* decimal comparison */ 02399 d1 = decimal_alloc (); 02400 d2 = decimal_alloc (); 02401 02402 decimal_expand (d1, p->x); 02403 decimal_expand (d2, p->y); 02404 dpush (cb_build_funcall_2 ("cob_decimal_cmp", d1, d2)); 02405 decimal_free (); 02406 decimal_free (); 02407 x = cb_list_reverse (decimal_stack); 02408 decimal_stack = NULL; 02409 } else { 02410 if (cb_chk_num_cond (p->x, p->y)) { 02411 size1 = cb_field_size (p->x); 02412 x = cb_build_funcall_3 ("memcmp", 02413 cb_build_cast_address (p->x), 02414 cb_build_cast_address (p->y), 02415 cb_int (size1)); 02416 break; 02417 } 02418 if (CB_TREE_CLASS (p->x) == CB_CLASS_NUMERIC 02419 && CB_TREE_CLASS (p->y) == CB_CLASS_NUMERIC 02420 && cb_fits_int (p->y)) { 02421 x = cb_build_optim_cond (p); 02422 break; 02423 } 02424 02425 /* field comparison */ 02426 if ((CB_REF_OR_FIELD_P (p->x)) 02427 && (CB_TREE_CATEGORY (p->x) == CB_CATEGORY_ALPHANUMERIC || 02428 CB_TREE_CATEGORY (p->x) == CB_CATEGORY_ALPHABETIC) 02429 && (cb_field_size (p->x) == 1) 02430 && (!current_program->alphabet_name_list) 02431 && (p->y == cb_space || p->y == cb_low || 02432 p->y == cb_high || p->y == cb_zero)) { 02433 x = cb_build_funcall_2 ("$G", p->x, p->y); 02434 break; 02435 } 02436 if (cb_chk_alpha_cond (p->x) && cb_chk_alpha_cond (p->y)) { 02437 size1 = cb_field_size (p->x); 02438 size2 = cb_field_size (p->y); 02439 } else { 02440 size1 = 0; 02441 size2 = 0; 02442 } 02443 if (size1 == 1 && size2 == 1) { 02444 x = cb_build_funcall_2 ("$G", p->x, p->y); 02445 } else if (size1 != 0 && size1 == size2) { 02446 x = cb_build_funcall_3 ("memcmp", 02447 cb_build_cast_address (p->x), 02448 cb_build_cast_address (p->y), 02449 cb_int (size1)); 02450 } else { 02451 if (CB_TREE_CLASS (p->x) == CB_CLASS_NUMERIC && p->y == cb_zero) { 02452 x = cb_build_optim_cond (p); 02453 } else { 02454 x = cb_build_funcall_2 ("cob_cmp", p->x, p->y); 02455 } 02456 } 02457 } 02458 } 02459 return cb_build_binary_op (x, p->op, p->y); 02460 default: 02461 cb_error_x (x, _("Invalid expression")); 02462 return cb_error_node; 02463 } 02464 /* NOT REACHED */ 02465 return x; 02466 } 02467 02468 /* 02469 * ADD/SUBTRACT CORRESPONDING 02470 */ 02471 02472 static cb_tree 02473 cb_build_optim_add (cb_tree v, cb_tree n) 02474 { 02475 size_t z; 02476 const char *s; 02477 struct cb_field *f; 02478 02479 if (CB_REF_OR_FIELD_P (v)) { 02480 f = cb_field (v); 02481 if (!f->pic->scale && (f->usage == CB_USAGE_BINARY || 02482 f->usage == CB_USAGE_COMP_5 || 02483 f->usage == CB_USAGE_COMP_X)) { 02484 z = (f->size - 1) + (8 * (f->pic->have_sign ? 1 : 0)) + 02485 (16 * (f->flag_binary_swap ? 1 : 0)); 02486 #if defined(COB_NON_ALIGNED) && !defined(_MSC_VER) 02487 switch (f->size) { 02488 case 2: 02489 #ifdef COB_SHORT_BORK 02490 s = bin_add_funcs[z]; 02491 break; 02492 #endif 02493 case 4: 02494 case 8: 02495 if (f->storage != CB_STORAGE_LINKAGE && 02496 f->indexes == 0 && (f->offset % f->size) == 0) { 02497 s = align_bin_add_funcs[z]; 02498 } else { 02499 s = bin_add_funcs[z]; 02500 } 02501 break; 02502 default: 02503 s = bin_add_funcs[z]; 02504 break; 02505 } 02506 #else 02507 if (f->usage == CB_USAGE_COMP_5) { 02508 switch (f->size) { 02509 case 1: 02510 case 2: 02511 case 4: 02512 case 8: 02513 return cb_build_assign (v, cb_build_binary_op (v, '+', n)); 02514 } 02515 } 02516 s = bin_add_funcs[z]; 02517 #endif 02518 if (s) { 02519 return cb_build_funcall_2 (s, 02520 cb_build_cast_address (v), 02521 cb_build_cast_integer (n)); 02522 } 02523 } else if (!f->pic->scale && f->usage == CB_USAGE_PACKED && 02524 f->pic->digits < 10) { 02525 return cb_build_funcall_2 ("cob_add_packed_int", 02526 v, cb_build_cast_integer (n)); 02527 } 02528 02529 } 02530 return cb_build_funcall_2 ("cob_add_int", v, cb_build_cast_integer (n)); 02531 } 02532 02533 static cb_tree 02534 cb_build_optim_sub (cb_tree v, cb_tree n) 02535 { 02536 size_t z; 02537 const char *s; 02538 struct cb_field *f; 02539 02540 if (CB_REF_OR_FIELD_P (v)) { 02541 f = cb_field (v); 02542 if (!f->pic->scale && (f->usage == CB_USAGE_BINARY || 02543 f->usage == CB_USAGE_COMP_5 || 02544 f->usage == CB_USAGE_COMP_X)) { 02545 z = (f->size - 1) + (8 * (f->pic->have_sign ? 1 : 0)) + 02546 (16 * (f->flag_binary_swap ? 1 : 0)); 02547 #if defined(COB_NON_ALIGNED) && !defined(_MSC_VER) 02548 switch (f->size) { 02549 case 2: 02550 #ifdef COB_SHORT_BORK 02551 s = bin_sub_funcs[z]; 02552 break; 02553 #endif 02554 case 4: 02555 case 8: 02556 if (f->storage != CB_STORAGE_LINKAGE && 02557 f->indexes == 0 && (f->offset % f->size) == 0) { 02558 s = align_bin_sub_funcs[z]; 02559 } else { 02560 s = bin_sub_funcs[z]; 02561 } 02562 break; 02563 default: 02564 s = bin_sub_funcs[z]; 02565 break; 02566 } 02567 #else 02568 if (f->usage == CB_USAGE_COMP_5) { 02569 switch (f->size) { 02570 case 1: 02571 case 2: 02572 case 4: 02573 case 8: 02574 return cb_build_assign (v, cb_build_binary_op (v, '-', n)); 02575 } 02576 } 02577 s = bin_sub_funcs[z]; 02578 #endif 02579 if (s) { 02580 return cb_build_funcall_2 (s, 02581 cb_build_cast_address (v), 02582 cb_build_cast_integer (n)); 02583 } 02584 } 02585 02586 } 02587 return cb_build_funcall_2 ("cob_sub_int", v, cb_build_cast_integer (n)); 02588 } 02589 02590 cb_tree 02591 cb_build_add (cb_tree v, cb_tree n, cb_tree round_opt) 02592 { 02593 cb_tree opt; 02594 struct cb_field *f; 02595 02596 #ifdef COB_NON_ALIGNED 02597 if (CB_INDEX_P (v)) { 02598 return cb_build_move (cb_build_binary_op (v, '+', n), v); 02599 } 02600 if (CB_TREE_CLASS (v) == CB_CLASS_POINTER) { 02601 current_program->gen_ptrmanip = 1; 02602 return cb_build_funcall_3 ("cob_pointer_manip", v, n, cb_int0); 02603 } 02604 #else 02605 if (CB_INDEX_P (v) || CB_TREE_CLASS (v) == CB_CLASS_POINTER) { 02606 return cb_build_move (cb_build_binary_op (v, '+', n), v); 02607 } 02608 #endif 02609 02610 if (CB_REF_OR_FIELD_P (v)) { 02611 f = cb_field (v); 02612 f->count++; 02613 } 02614 if (CB_REF_OR_FIELD_P (n)) { 02615 f = cb_field (n); 02616 f->count++; 02617 } 02618 if (round_opt == cb_high) { 02619 if (cb_fits_int (n)) { 02620 return cb_build_optim_add (v, n); 02621 } else { 02622 return cb_build_funcall_3 ("cob_add", v, n, cb_int0); 02623 } 02624 } 02625 opt = build_store_option (v, round_opt); 02626 if (opt == cb_int0 && cb_fits_int (n)) { 02627 return cb_build_optim_add (v, n); 02628 } 02629 return cb_build_funcall_3 ("cob_add", v, n, opt); 02630 } 02631 02632 cb_tree 02633 cb_build_sub (cb_tree v, cb_tree n, cb_tree round_opt) 02634 { 02635 cb_tree opt; 02636 struct cb_field *f; 02637 02638 #ifdef COB_NON_ALIGNED 02639 if (CB_INDEX_P (v)) { 02640 return cb_build_move (cb_build_binary_op (v, '-', n), v); 02641 } 02642 if (CB_TREE_CLASS (v) == CB_CLASS_POINTER) { 02643 current_program->gen_ptrmanip = 1; 02644 return cb_build_funcall_3 ("cob_pointer_manip", v, n, cb_int1); 02645 } 02646 #else 02647 if (CB_INDEX_P (v) || CB_TREE_CLASS (v) == CB_CLASS_POINTER) { 02648 return cb_build_move (cb_build_binary_op (v, '-', n), v); 02649 } 02650 #endif 02651 02652 if (CB_REF_OR_FIELD_P (v)) { 02653 f = cb_field (v); 02654 f->count++; 02655 } 02656 if (CB_REF_OR_FIELD_P (n)) { 02657 f = cb_field (n); 02658 f->count++; 02659 } 02660 opt = build_store_option (v, round_opt); 02661 if (opt == cb_int0 && cb_fits_int (n)) { 02662 return cb_build_optim_sub (v, n); 02663 } 02664 return cb_build_funcall_3 ("cob_sub", v, n, opt); 02665 } 02666 02667 static void 02668 emit_corresponding (cb_tree (*func) (cb_tree f1, cb_tree f2, cb_tree f3), 02669 cb_tree x1, cb_tree x2, cb_tree opt) 02670 { 02671 struct cb_field *f1, *f2; 02672 cb_tree t1; 02673 cb_tree t2; 02674 02675 for (f1 = cb_field (x1)->children; f1; f1 = f1->sister) { 02676 if (!f1->redefines && !f1->flag_occurs) { 02677 for (f2 = cb_field (x2)->children; f2; f2 = f2->sister) { 02678 if (!f2->redefines && !f2->flag_occurs) { 02679 if (strcmp (f1->name, f2->name) == 0) { 02680 t1 = cb_build_field_reference (f1, x1); 02681 t2 = cb_build_field_reference (f2, x2); 02682 if (f1->children && f2->children) { 02683 emit_corresponding (func, t1, t2, opt); 02684 } else { 02685 cb_emit (func (t1, t2, opt)); 02686 } 02687 } 02688 } 02689 } 02690 } 02691 } 02692 } 02693 02694 void 02695 cb_emit_corresponding (cb_tree (*func) (cb_tree f1, cb_tree f2, cb_tree f3), 02696 cb_tree x1, cb_tree x2, cb_tree opt) 02697 { 02698 x1 = cb_check_group_name (x1); 02699 x2 = cb_check_group_name (x2); 02700 02701 if (cb_validate_one (x1)) { 02702 return; 02703 } 02704 if (cb_validate_one (x2)) { 02705 return; 02706 } 02707 02708 emit_corresponding (func, x1, x2, opt); 02709 } 02710 02711 static void 02712 emit_move_corresponding (cb_tree x1, cb_tree x2) 02713 { 02714 struct cb_field *f1, *f2; 02715 cb_tree t1; 02716 cb_tree t2; 02717 02718 for (f1 = cb_field (x1)->children; f1; f1 = f1->sister) { 02719 if (!f1->redefines && !f1->flag_occurs) { 02720 for (f2 = cb_field (x2)->children; f2; f2 = f2->sister) { 02721 if (!f2->redefines && !f2->flag_occurs) { 02722 if (strcmp (f1->name, f2->name) == 0) { 02723 t1 = cb_build_field_reference (f1, x1); 02724 t2 = cb_build_field_reference (f2, x2); 02725 if (f1->children && f2->children) { 02726 emit_move_corresponding (t1, t2); 02727 } else { 02728 cb_emit (cb_build_move (t1, t2)); 02729 } 02730 } 02731 } 02732 } 02733 } 02734 } 02735 } 02736 02737 void 02738 cb_emit_move_corresponding (cb_tree x1, cb_tree x2) 02739 { 02740 cb_tree l; 02741 cb_tree v; 02742 02743 x1 = cb_check_group_name (x1); 02744 if (cb_validate_one (x1)) { 02745 return; 02746 } 02747 for (l = x2; l; l = CB_CHAIN(l)) { 02748 v = CB_VALUE(l); 02749 v = cb_check_group_name (v); 02750 if (cb_validate_one (v)) { 02751 return; 02752 } 02753 emit_move_corresponding (x1, v); 02754 } 02755 } 02756 02757 static void 02758 output_screen_from (struct cb_field *p, const size_t sisters) 02759 { 02760 int type; 02761 02762 if (sisters && p->sister) { 02763 output_screen_from (p->sister, 1); 02764 } 02765 if (p->children) { 02766 output_screen_from (p->children, 1); 02767 } 02768 02769 type = (p->children ? COB_SCREEN_TYPE_GROUP : 02770 p->values ? COB_SCREEN_TYPE_VALUE : 02771 (p->size > 0) ? COB_SCREEN_TYPE_FIELD : COB_SCREEN_TYPE_ATTRIBUTE); 02772 if (type == COB_SCREEN_TYPE_FIELD && p->screen_from) { 02773 cb_emit (cb_build_funcall_2 ("cob_move", p->screen_from, CB_TREE (p))); 02774 } 02775 } 02776 02777 static void 02778 output_screen_to (struct cb_field *p, const size_t sisters) 02779 { 02780 int type; 02781 02782 if (sisters && p->sister) { 02783 output_screen_to (p->sister, 1); 02784 } 02785 if (p->children) { 02786 output_screen_to (p->children, 1); 02787 } 02788 02789 type = (p->children ? COB_SCREEN_TYPE_GROUP : 02790 p->values ? COB_SCREEN_TYPE_VALUE : 02791 (p->size > 0) ? COB_SCREEN_TYPE_FIELD : COB_SCREEN_TYPE_ATTRIBUTE); 02792 if (type == COB_SCREEN_TYPE_FIELD && p->screen_to) { 02793 cb_emit (cb_build_funcall_2 ("cob_move", CB_TREE (p), p->screen_to)); 02794 } 02795 } 02796 02797 /* 02798 * ACCEPT statement 02799 */ 02800 02801 void 02802 cb_emit_accept (cb_tree var, cb_tree pos, cb_tree fgc, cb_tree bgc, 02803 cb_tree scroll, int dispattrs) 02804 { 02805 cb_tree line; 02806 cb_tree column; 02807 02808 if (cb_validate_one (var)) { 02809 return; 02810 } 02811 if (cb_validate_one (pos)) { 02812 return; 02813 } 02814 if (cb_validate_one (fgc)) { 02815 return; 02816 } 02817 if (cb_validate_one (bgc)) { 02818 return; 02819 } 02820 if (cb_validate_one (scroll)) { 02821 return; 02822 } 02823 if (current_program->flag_screen) { 02824 /* Bump ref count to force CRT STATUS field generation */ 02825 cb_field (current_program->crt_status)->count++; 02826 if ((CB_REF_OR_FIELD_P (var)) && 02827 CB_FIELD (cb_ref (var))->storage == CB_STORAGE_SCREEN) { 02828 output_screen_from (CB_FIELD (cb_ref (var)), 0); 02829 gen_screen_ptr = 1; 02830 if (pos) { 02831 if (CB_PAIR_P (pos)) { 02832 line = CB_PAIR_X (pos); 02833 column = CB_PAIR_Y (pos); 02834 cb_emit (cb_build_funcall_3 ("cob_screen_accept", 02835 var, line, column)); 02836 } else { 02837 cb_emit (cb_build_funcall_3 ("cob_screen_accept", 02838 var, pos, NULL)); 02839 } 02840 } else { 02841 cb_emit (cb_build_funcall_3 ("cob_screen_accept", 02842 var, NULL, NULL)); 02843 } 02844 gen_screen_ptr = 0; 02845 output_screen_to (CB_FIELD (cb_ref (var)), 0); 02846 } else { 02847 if (pos || fgc || bgc) { 02848 if (!pos) { 02849 cb_emit (cb_build_funcall_7 ("cob_field_accept", 02850 var, NULL, NULL, fgc, bgc, 02851 scroll, cb_int (dispattrs))); 02852 } else if (CB_PAIR_P (pos)) { 02853 line = CB_PAIR_X (pos); 02854 column = CB_PAIR_Y (pos); 02855 cb_emit (cb_build_funcall_7 ("cob_field_accept", 02856 var, line, column, fgc, bgc, 02857 scroll, cb_int (dispattrs))); 02858 } else { 02859 cb_emit (cb_build_funcall_7 ("cob_field_accept", 02860 var, pos, NULL, fgc, bgc, 02861 scroll, cb_int (dispattrs))); 02862 } 02863 } else { 02864 cb_emit (cb_build_funcall_7 ("cob_field_accept", 02865 var, NULL, NULL, fgc, bgc, 02866 scroll, cb_int (dispattrs))); 02867 } 02868 } 02869 } else if (pos || fgc || bgc || scroll) { 02870 /* Bump ref count to force CRT STATUS field generation */ 02871 cb_field (current_program->crt_status)->count++; 02872 if (!pos) { 02873 cb_emit (cb_build_funcall_7 ("cob_field_accept", 02874 var, NULL, NULL, fgc, bgc, scroll, 02875 cb_int (dispattrs))); 02876 } else if (CB_PAIR_P (pos)) { 02877 line = CB_PAIR_X (pos); 02878 column = CB_PAIR_Y (pos); 02879 cb_emit (cb_build_funcall_7 ("cob_field_accept", 02880 var, line, column, fgc, bgc, scroll, 02881 cb_int (dispattrs))); 02882 } else { 02883 cb_emit (cb_build_funcall_7 ("cob_field_accept", 02884 var, pos, NULL, fgc, bgc, scroll, 02885 cb_int (dispattrs))); 02886 } 02887 } else { 02888 cb_emit (cb_build_funcall_1 ("cob_accept", var)); 02889 } 02890 } 02891 02892 void 02893 cb_emit_accept_line_or_col (cb_tree var, const int l_or_c) 02894 { 02895 if (cb_validate_one (var)) { 02896 return; 02897 } 02898 cb_emit (cb_build_funcall_2 ("cob_screen_line_col", var, cb_int (l_or_c))); 02899 } 02900 02901 void 02902 cb_emit_accept_date (cb_tree var) 02903 { 02904 if (cb_validate_one (var)) { 02905 return; 02906 } 02907 cb_emit (cb_build_funcall_1 ("cob_accept_date", var)); 02908 } 02909 02910 void 02911 cb_emit_accept_date_yyyymmdd (cb_tree var) 02912 { 02913 if (cb_validate_one (var)) { 02914 return; 02915 } 02916 cb_emit (cb_build_funcall_1 ("cob_accept_date_yyyymmdd", var)); 02917 } 02918 02919 void 02920 cb_emit_accept_day (cb_tree var) 02921 { 02922 if (cb_validate_one (var)) { 02923 return; 02924 } 02925 cb_emit (cb_build_funcall_1 ("cob_accept_day", var)); 02926 } 02927 02928 void 02929 cb_emit_accept_day_yyyyddd (cb_tree var) 02930 { 02931 if (cb_validate_one (var)) { 02932 return; 02933 } 02934 cb_emit (cb_build_funcall_1 ("cob_accept_day_yyyyddd", var)); 02935 } 02936 02937 void 02938 cb_emit_accept_day_of_week (cb_tree var) 02939 { 02940 if (cb_validate_one (var)) { 02941 return; 02942 } 02943 cb_emit (cb_build_funcall_1 ("cob_accept_day_of_week", var)); 02944 } 02945 02946 void 02947 cb_emit_accept_time (cb_tree var) 02948 { 02949 if (cb_validate_one (var)) { 02950 return; 02951 } 02952 cb_emit (cb_build_funcall_1 ("cob_accept_time", var)); 02953 } 02954 02955 void 02956 cb_emit_accept_command_line (cb_tree var) 02957 { 02958 if (cb_validate_one (var)) { 02959 return; 02960 } 02961 cb_emit (cb_build_funcall_1 ("cob_accept_command_line", var)); 02962 } 02963 02964 void 02965 cb_emit_get_environment (cb_tree envvar, cb_tree envval) 02966 { 02967 if (cb_validate_one (envvar)) { 02968 return; 02969 } 02970 if (cb_validate_one (envval)) { 02971 return; 02972 } 02973 cb_emit (cb_build_funcall_2 ("cob_get_environment", envvar, envval)); 02974 } 02975 02976 void 02977 cb_emit_accept_environment (cb_tree var) 02978 { 02979 if (cb_validate_one (var)) { 02980 return; 02981 } 02982 cb_emit (cb_build_funcall_1 ("cob_accept_environment", var)); 02983 } 02984 02985 void 02986 cb_emit_accept_arg_number (cb_tree var) 02987 { 02988 if (cb_validate_one (var)) { 02989 return; 02990 } 02991 cb_emit (cb_build_funcall_1 ("cob_accept_arg_number", var)); 02992 } 02993 02994 void 02995 cb_emit_accept_arg_value (cb_tree var) 02996 { 02997 if (cb_validate_one (var)) { 02998 return; 02999 } 03000 cb_emit (cb_build_funcall_1 ("cob_accept_arg_value", var)); 03001 } 03002 03003 void 03004 cb_emit_accept_mnemonic (cb_tree var, cb_tree mnemonic) 03005 { 03006 if (cb_validate_one (var)) { 03007 return; 03008 } 03009 switch (CB_SYSTEM_NAME (cb_ref (mnemonic))->token) { 03010 case CB_DEVICE_CONSOLE: 03011 case CB_DEVICE_SYSIN: 03012 cb_emit (cb_build_funcall_1 ("cob_accept", var)); 03013 break; 03014 default: 03015 cb_error_x (mnemonic, _("Invalid input stream '%s'"), 03016 cb_name (mnemonic)); 03017 break; 03018 } 03019 } 03020 03021 void 03022 cb_emit_accept_name (cb_tree var, cb_tree name) 03023 { 03024 cb_tree sys; 03025 03026 if (cb_validate_one (var)) { 03027 return; 03028 } 03029 if (CB_REFERENCE (name)->word->count == 0) { 03030 sys = lookup_system_name (CB_NAME (name)); 03031 03032 if (sys != cb_error_node) { 03033 switch (CB_SYSTEM_NAME (sys)->token) { 03034 case CB_DEVICE_CONSOLE: 03035 case CB_DEVICE_SYSIN: 03036 cb_warning_x (name, _("'%s' undefined in SPECIAL-NAMES"), CB_NAME (name)); 03037 cb_emit (cb_build_funcall_1 ("cob_accept", var)); 03038 return; 03039 default: 03040 break; 03041 } 03042 } 03043 } 03044 03045 cb_error_x (name, _("'%s' undefined in SPECIAL-NAMES"), CB_NAME (name)); 03046 } 03047 03048 /* 03049 * ALLOCATE statement 03050 */ 03051 03052 void 03053 cb_emit_allocate (cb_tree target1, cb_tree target2, cb_tree size, cb_tree initialize) 03054 { 03055 cb_tree x; 03056 char buff[32]; 03057 03058 if (cb_validate_one (target1)) { 03059 return; 03060 } 03061 if (cb_validate_one (target2)) { 03062 return; 03063 } 03064 if (cb_validate_one (size)) { 03065 return; 03066 } 03067 if (target1) { 03068 if (!(CB_REFERENCE_P(target1) && 03069 cb_field (target1)->flag_item_based)) { 03070 cb_error_x (CB_TREE(current_statement), 03071 _("Target of ALLOCATE is not a BASED item")); 03072 } 03073 } 03074 if (target2) { 03075 if (!(CB_REFERENCE_P(target2) && 03076 CB_TREE_CLASS (target2) == CB_CLASS_POINTER)) { 03077 cb_error_x (CB_TREE(current_statement), 03078 _("Target of RETURNING is not a data pointer")); 03079 } 03080 } 03081 if (size) { 03082 if (CB_TREE_CLASS (size) != CB_CLASS_NUMERIC) { 03083 cb_error_x (CB_TREE(current_statement), 03084 _("The CHARACTERS field of ALLOCATE must be numeric")); 03085 } 03086 } 03087 if (target1) { 03088 sprintf (buff, "%d", cb_field (target1)->memory_size); 03089 x = cb_build_numeric_literal (0, (ucharptr)buff, 0); 03090 cb_emit (cb_build_funcall_3 ("cob_allocate", 03091 cb_build_cast_addr_of_addr (target1), target2, x)); 03092 } else { 03093 cb_emit (cb_build_funcall_3 ("cob_allocate", 03094 NULL, target2, size)); 03095 } 03096 if (initialize && target1) { 03097 current_statement->handler2 = 03098 cb_build_initialize (target1, cb_true, NULL, cb_true, 0); 03099 } 03100 } 03101 03102 03103 /* 03104 * CALL statement 03105 */ 03106 03107 void 03108 cb_emit_call (cb_tree prog, cb_tree using, cb_tree returning, 03109 cb_tree on_exception, cb_tree not_on_exception) 03110 { 03111 cb_tree l; 03112 cb_tree x; 03113 const struct system_table *psyst; 03114 int is_sys_call = 0; 03115 03116 if (CB_INTRINSIC_P (prog)) { 03117 if (CB_INTRINSIC(prog)->intr_tab->category != CB_CATEGORY_ALPHANUMERIC) { 03118 cb_error (_("Only alphanumeric FUNCTION types are allowed here")); 03119 return; 03120 } 03121 } 03122 if (returning) { 03123 if (CB_TREE_CLASS(returning) != CB_CLASS_NUMERIC && 03124 CB_TREE_CLASS(returning) != CB_CLASS_POINTER) { 03125 cb_error (_("Invalid RETURNING field")); 03126 return; 03127 } 03128 } 03129 for (l = using; l; l = CB_CHAIN (l)) { 03130 x = CB_VALUE (l); 03131 if (x == cb_error_node) { 03132 continue; 03133 } 03134 if (CB_CONST_P (x) && x != cb_null) { 03135 cb_error_x (x, _("Figurative constant invalid here")); 03136 } 03137 if ((CB_REFERENCE_P (x) && CB_FIELD_P(CB_REFERENCE(x)->value)) 03138 || CB_FIELD_P (x)) { 03139 if (cb_field (x)->level == 88) { 03140 cb_error_x (x, _("'%s' Not a data name"), CB_NAME (x)); 03141 return; 03142 } 03143 if (cb_warn_call_params && 03144 CB_PURPOSE_INT (l) == CB_CALL_BY_REFERENCE) { 03145 if (cb_field (x)->level != 01 && 03146 cb_field (x)->level != 77) { 03147 cb_warning_x (x, _("'%s' is not 01 or 77 level item"), CB_NAME (x)); 03148 } 03149 } 03150 } 03151 } 03152 03153 if (CB_LITERAL_P(prog)) { 03154 for (psyst = (const struct system_table *)&system_tab[0]; psyst->syst_name; psyst++) { 03155 if (!strcmp((const char *)CB_LITERAL(prog)->data, 03156 (const char *)psyst->syst_name)) { 03157 if (psyst->syst_params > cb_list_length (using)) { 03158 cb_error (_("Wrong number of CALL parameters for '%s'"), 03159 (char *)psyst->syst_name); 03160 return; 03161 } 03162 is_sys_call = 1; 03163 break; 03164 } 03165 } 03166 } 03167 03168 cb_emit (cb_build_call (prog, using, on_exception, not_on_exception, 03169 returning, is_sys_call)); 03170 } 03171 03172 /* 03173 * CANCEL statement 03174 */ 03175 03176 void 03177 cb_emit_cancel (cb_tree prog) 03178 { 03179 if (cb_validate_one (prog)) { 03180 return; 03181 } 03182 cb_emit (cb_build_funcall_1 ("cob_field_cancel", prog)); 03183 } 03184 03185 /* 03186 * CLOSE statement 03187 */ 03188 03189 void 03190 cb_emit_close (cb_tree file, cb_tree opt) 03191 { 03192 if (file == cb_error_node) { 03193 return; 03194 } 03195 file = cb_ref (file); 03196 if (file == cb_error_node) { 03197 return; 03198 } 03199 current_statement->file = file; 03200 if (CB_FILE (file)->organization == COB_ORG_SORT) { 03201 cb_error_x (CB_TREE (current_statement), 03202 _("Operation not allowed on SORT files")); 03203 } 03204 cb_emit (cb_build_funcall_3 ("cob_close", file, opt, 03205 CB_FILE(file)->file_status)); 03206 } 03207 03208 /* 03209 * COMMIT statement 03210 */ 03211 03212 void 03213 cb_emit_commit (void) 03214 { 03215 cb_emit (cb_build_funcall_0 ("cob_commit")); 03216 } 03217 03218 /* 03219 * CONTINUE statement 03220 */ 03221 03222 void 03223 cb_emit_continue (void) 03224 { 03225 cb_emit (cb_build_continue ()); 03226 } 03227 03228 /* 03229 * DELETE statement 03230 */ 03231 03232 void 03233 cb_emit_delete (cb_tree file) 03234 { 03235 if (file == cb_error_node) { 03236 return; 03237 } 03238 file = cb_ref (file); 03239 if (file == cb_error_node) { 03240 return; 03241 } 03242 current_statement->file = file; 03243 if (CB_FILE (file)->organization == COB_ORG_SORT) { 03244 cb_error_x (CB_TREE (current_statement), 03245 _("Operation not allowed on SORT files")); 03246 } 03247 cb_emit (cb_build_funcall_2 ("cob_delete", file, CB_FILE(file)->file_status)); 03248 } 03249 03250 /* 03251 * DISPLAY statement 03252 */ 03253 03254 void 03255 cb_emit_env_name (cb_tree value) 03256 { 03257 if (cb_validate_one (value)) { 03258 return; 03259 } 03260 cb_emit (cb_build_funcall_1 ("cob_display_environment", value)); 03261 } 03262 03263 void 03264 cb_emit_env_value (cb_tree value) 03265 { 03266 if (cb_validate_one (value)) { 03267 return; 03268 } 03269 cb_emit (cb_build_funcall_1 ("cob_display_env_value", value)); 03270 } 03271 03272 void 03273 cb_emit_arg_number (cb_tree value) 03274 { 03275 if (cb_validate_one (value)) { 03276 return; 03277 } 03278 cb_emit (cb_build_funcall_1 ("cob_display_arg_number", value)); 03279 } 03280 03281 void 03282 cb_emit_command_line (cb_tree value) 03283 { 03284 if (cb_validate_one (value)) { 03285 return; 03286 } 03287 cb_emit (cb_build_funcall_1 ("cob_display_command_line", value)); 03288 } 03289 03290 void 03291 cb_emit_display (cb_tree values, cb_tree upon, cb_tree no_adv, cb_tree pos, 03292 cb_tree fgc, cb_tree bgc, cb_tree scroll, int dispattrs) 03293 { 03294 cb_tree l; 03295 cb_tree x; 03296 cb_tree line; 03297 cb_tree column; 03298 cb_tree p; 03299 03300 if (cb_validate_list (values)) { 03301 return; 03302 } 03303 if (cb_validate_one (pos)) { 03304 return; 03305 } 03306 if (cb_validate_one (fgc)) { 03307 return; 03308 } 03309 if (cb_validate_one (bgc)) { 03310 return; 03311 } 03312 if (cb_validate_one (scroll)) { 03313 return; 03314 } 03315 for (l = values; l; l = CB_CHAIN (l)) { 03316 x = CB_VALUE (l); 03317 if (x == cb_error_node) { 03318 return; 03319 } 03320 03321 switch (CB_TREE_TAG (x)) { 03322 case CB_TAG_LITERAL: 03323 case CB_TAG_INTRINSIC: 03324 case CB_TAG_CONST: 03325 case CB_TAG_STRING: 03326 case CB_TAG_INTEGER: 03327 break; 03328 case CB_TAG_REFERENCE: 03329 if (!CB_FIELD_P(CB_REFERENCE(x)->value)) { 03330 cb_error_x (x, _("'%s' is an invalid type for DISPLAY operand"), cb_name (x)); 03331 return; 03332 } 03333 break; 03334 default: 03335 cb_error_x (x, _("Invalid type for DISPLAY operand")); 03336 return; 03337 } 03338 } 03339 if (upon == cb_error_node) { 03340 return; 03341 } 03342 03343 x = CB_VALUE (values); 03344 if ((CB_REF_OR_FIELD_P (x)) && 03345 CB_FIELD (cb_ref (x))->storage == CB_STORAGE_SCREEN) { 03346 output_screen_from (CB_FIELD (cb_ref (x)), 0); 03347 gen_screen_ptr = 1; 03348 if (pos) { 03349 if (CB_PAIR_P (pos)) { 03350 line = CB_PAIR_X (pos); 03351 column = CB_PAIR_Y (pos); 03352 if (line == NULL) { 03353 line = cb_one; 03354 } 03355 if (column == NULL) { 03356 column = cb_one; 03357 } 03358 cb_emit (cb_build_funcall_3 ("cob_screen_display", x, 03359 line, column)); 03360 } else { 03361 cb_emit (cb_build_funcall_3 ("cob_screen_display", x, 03362 pos, NULL)); 03363 } 03364 } else { 03365 cb_emit (cb_build_funcall_3 ("cob_screen_display", x, 03366 NULL, NULL)); 03367 } 03368 gen_screen_ptr = 0; 03369 } else if (pos || fgc || bgc || scroll || dispattrs) { 03370 if (!pos) { 03371 cb_emit (cb_build_funcall_7 ("cob_field_display", 03372 CB_VALUE (values), NULL, NULL, fgc, bgc, 03373 scroll, cb_int (dispattrs))); 03374 } else if (CB_PAIR_P (pos)) { 03375 line = CB_PAIR_X (pos); 03376 column = CB_PAIR_Y (pos); 03377 if (line == NULL) { 03378 line = cb_one; 03379 } 03380 if (column == NULL) { 03381 column = cb_one; 03382 } 03383 cb_emit (cb_build_funcall_7 ("cob_field_display", 03384 CB_VALUE (values), line, column, fgc, bgc, 03385 scroll, cb_int (dispattrs))); 03386 } else { 03387 cb_emit (cb_build_funcall_7 ("cob_field_display", 03388 CB_VALUE (values), pos, NULL, fgc, bgc, 03389 scroll, cb_int (dispattrs))); 03390 } 03391 } else { 03392 /* DISPLAY x ... [UPON device-name] */ 03393 p = cb_build_funcall_3 ("cob_display", upon, no_adv, values); 03394 CB_FUNCALL(p)->varcnt = cb_list_length (values); 03395 cb_emit (p); 03396 for (l = values; l; l = CB_CHAIN (l)) { 03397 x = CB_VALUE (l); 03398 if (CB_FIELD_P (x)) { 03399 CB_FIELD (cb_ref (x))->count++; 03400 } 03401 } 03402 } 03403 } 03404 03405 cb_tree 03406 cb_build_display_upon (cb_tree x) 03407 { 03408 if (x == cb_error_node) { 03409 return cb_error_node; 03410 } 03411 03412 switch (CB_SYSTEM_NAME (cb_ref (x))->token) { 03413 case CB_DEVICE_CONSOLE: 03414 case CB_DEVICE_SYSOUT: 03415 return cb_int0; 03416 case CB_DEVICE_SYSERR: 03417 return cb_int1; 03418 default: 03419 cb_error_x (x, _("Invalid output stream")); 03420 return cb_error_node; 03421 } 03422 } 03423 03424 cb_tree 03425 cb_build_display_upon_direct (cb_tree x) 03426 { 03427 const char *name; 03428 cb_tree sys; 03429 03430 if (x == cb_error_node) { 03431 return cb_error_node; 03432 } 03433 name = CB_NAME (x); 03434 if (CB_REFERENCE (x)->word->count == 0) { 03435 sys = lookup_system_name (CB_NAME (x)); 03436 if (sys != cb_error_node) { 03437 switch (CB_SYSTEM_NAME (sys)->token) { 03438 case CB_DEVICE_CONSOLE: 03439 case CB_DEVICE_SYSOUT: 03440 cb_warning_x (x, _("'%s' undefined in SPECIAL-NAMES"), name); 03441 return cb_int0; 03442 case CB_DEVICE_SYSERR: 03443 cb_warning_x (x, _("'%s' undefined in SPECIAL-NAMES"), name); 03444 return cb_int1; 03445 default: 03446 break; 03447 } 03448 } 03449 } 03450 03451 cb_error_x (x, _("'%s' undefined in SPECIAL-NAMES"), name); 03452 return cb_error_node; 03453 } 03454 03455 /* 03456 * DIVIDE statement 03457 */ 03458 03459 void 03460 cb_emit_divide (cb_tree dividend, cb_tree divisor, cb_tree quotient, cb_tree remainder) 03461 { 03462 if (cb_validate_one (dividend)) { 03463 return; 03464 } 03465 if (cb_validate_one (divisor)) { 03466 return; 03467 } 03468 CB_VALUE (quotient) = cb_check_numeric_edited_name (CB_VALUE (quotient)); 03469 CB_VALUE (remainder) = cb_check_numeric_edited_name (CB_VALUE (remainder)); 03470 03471 if (cb_validate_one (CB_VALUE (quotient))) { 03472 return; 03473 } 03474 if (cb_validate_one (CB_VALUE (remainder))) { 03475 return; 03476 } 03477 03478 cb_emit (cb_build_funcall_4 ("cob_div_quotient", dividend, divisor, 03479 CB_VALUE (quotient), 03480 build_store_option (CB_VALUE (quotient), CB_PURPOSE (quotient)))); 03481 cb_emit (cb_build_funcall_2 ("cob_div_remainder", CB_VALUE (remainder), 03482 build_store_option (CB_VALUE (remainder), cb_int0))); 03483 } 03484 03485 /* 03486 * EVALUATE statement 03487 */ 03488 03489 static cb_tree 03490 evaluate_test (cb_tree s, cb_tree o) 03491 { 03492 int flag; 03493 cb_tree x, y; 03494 cb_tree t; 03495 03496 /* ANY is always true */ 03497 if (o == cb_any) { 03498 return cb_true; 03499 } 03500 03501 /* object TRUE or FALSE */ 03502 if (o == cb_true) { 03503 return s; 03504 } 03505 if (o == cb_false) { 03506 return cb_build_negation (s); 03507 } 03508 03509 flag = CB_PURPOSE_INT (o); 03510 x = CB_PAIR_X (CB_VALUE (o)); 03511 y = CB_PAIR_Y (CB_VALUE (o)); 03512 03513 /* subject TRUE or FALSE */ 03514 if (s == cb_true) { 03515 return flag ? cb_build_negation (x) : x; 03516 } 03517 if (s == cb_false) { 03518 return flag ? x : cb_build_negation (x); 03519 } 03520 03521 /* x THRU y */ 03522 if (y) { 03523 t = cb_build_binary_op (cb_build_binary_op (x, '[', s), 03524 '&', 03525 cb_build_binary_op (s, '[', y)); 03526 03527 return flag ? cb_build_negation (t) : t; 03528 } 03529 03530 if (CB_REFERENCE_P(x) && CB_FIELD_P(CB_REFERENCE(x)->value) && 03531 CB_FIELD(CB_REFERENCE(x)->value)->level == 88) { 03532 cb_error_x (CB_TREE (current_statement), 03533 _("Invalid use of 88 level in WHEN expression")); 03534 return NULL; 03535 } 03536 03537 /* regular comparison */ 03538 if (flag) { 03539 return cb_build_binary_op (s, '~', x); 03540 } else { 03541 return cb_build_binary_op (s, '=', x); 03542 } 03543 } 03544 03545 static cb_tree 03546 build_evaluate (cb_tree subject_list, cb_tree case_list) 03547 { 03548 cb_tree c1 = NULL; 03549 cb_tree c2; 03550 cb_tree c3; 03551 cb_tree subjs; 03552 cb_tree whens; 03553 cb_tree objs; 03554 cb_tree stmt; 03555 03556 if (case_list == NULL) { 03557 return NULL; 03558 } 03559 03560 whens = CB_VALUE (case_list); 03561 stmt = CB_VALUE (whens); 03562 whens = CB_CHAIN (whens); 03563 03564 /* for each WHEN sequence */ 03565 for (; whens; whens = CB_CHAIN (whens)) { 03566 c2 = NULL; 03567 /* single WHEN test */ 03568 for (subjs = subject_list, objs = CB_VALUE (whens); 03569 subjs && objs; subjs = CB_CHAIN (subjs), objs = CB_CHAIN (objs)) { 03570 c3 = evaluate_test (CB_VALUE (subjs), CB_VALUE (objs)); 03571 if (c3 == NULL) { 03572 return NULL; 03573 } 03574 03575 if (c2 == NULL) { 03576 c2 = c3; 03577 } else { 03578 c2 = cb_build_binary_op (c2, '&', c3); 03579 } 03580 } 03581 if (subjs || objs) { 03582 cb_error (_("Wrong number of WHEN parameters")); 03583 } 03584 /* connect multiple WHEN's */ 03585 if (c1 == NULL) { 03586 c1 = c2; 03587 } else { 03588 c1 = cb_build_binary_op (c1, '|', c2); 03589 } 03590 } 03591 03592 if (c1 == NULL) { 03593 return stmt; 03594 } else { 03595 return cb_build_if (cb_build_cond (c1), stmt, 03596 build_evaluate (subject_list, CB_CHAIN (case_list))); 03597 } 03598 } 03599 03600 void 03601 cb_emit_evaluate (cb_tree subject_list, cb_tree case_list) 03602 { 03603 cb_emit (build_evaluate (subject_list, case_list)); 03604 } 03605 03606 /* 03607 * FREE statement 03608 */ 03609 03610 void 03611 cb_emit_free (cb_tree vars) 03612 { 03613 cb_tree l; 03614 struct cb_field *f; 03615 int i; 03616 03617 if (cb_validate_list (vars)) { 03618 return; 03619 } 03620 for (l = vars, i = 1; l; l = CB_CHAIN (l), i++) { 03621 if (CB_TREE_CLASS (CB_VALUE (l)) == CB_CLASS_POINTER) { 03622 if (CB_CAST_P (CB_VALUE (l))) { 03623 f = cb_field (CB_CAST (CB_VALUE(l))->val); 03624 if (!f->flag_item_based) { 03625 cb_error_x (CB_TREE (current_statement), 03626 _("Target %d of FREE, a data address identifier, must address a BASED data item"), i); 03627 } 03628 cb_emit (cb_build_funcall_2 ("cob_free_alloc", 03629 cb_build_cast_address (CB_VALUE (l)), NULL)); 03630 } else { 03631 cb_emit (cb_build_funcall_2 ("cob_free_alloc", 03632 NULL, cb_build_cast_address (CB_VALUE (l)))); 03633 } 03634 } else if (CB_REF_OR_FIELD_P (CB_VALUE (l))) { 03635 f = cb_field (CB_VALUE (l)); 03636 if (!f->flag_item_based) { 03637 cb_error_x (CB_TREE (current_statement), 03638 _("Target %d of FREE, a data address identifier, must address a BASED data item"), i); 03639 } 03640 cb_emit (cb_build_funcall_2 ("cob_free_alloc", 03641 cb_build_cast_addr_of_addr (CB_VALUE (l)), NULL)); 03642 } else { 03643 cb_error_x (CB_TREE (current_statement), 03644 _("Target %d of FREE must be a data pointer"), i); 03645 } 03646 } 03647 } 03648 03649 /* 03650 * GO TO statement 03651 */ 03652 03653 void 03654 cb_emit_goto (cb_tree target, cb_tree depending) 03655 { 03656 if (target == cb_error_node) { 03657 return; 03658 } 03659 if (depending) { 03660 /* GO TO procedure-name ... DEPENDING ON identifier */ 03661 cb_emit (cb_build_goto (target, depending)); 03662 } else { 03663 /* GO TO procedure-name */ 03664 if (target == NULL) { 03665 cb_verify (cb_goto_statement_without_name, "GO TO without procedure-name"); 03666 } else if (CB_CHAIN (target)) { 03667 cb_error (_("GO TO with multiple procedure-names")); 03668 } else { 03669 cb_emit (cb_build_goto (CB_VALUE (target), NULL)); 03670 } 03671 } 03672 } 03673 03674 void 03675 cb_emit_exit (size_t goback) 03676 { 03677 if (goback) { 03678 cb_emit (cb_build_goto (cb_int1, NULL)); 03679 } else { 03680 cb_emit (cb_build_goto (NULL, NULL)); 03681 } 03682 } 03683 03684 /* 03685 * IF statement 03686 */ 03687 03688 void 03689 cb_emit_if (cb_tree cond, cb_tree stmt1, cb_tree stmt2) 03690 { 03691 cb_emit (cb_build_if (cond, stmt1, stmt2)); 03692 } 03693 03694 /* 03695 * INITIALIZE statement 03696 */ 03697 03698 void 03699 cb_emit_initialize (cb_tree vars, cb_tree fillinit, cb_tree value, cb_tree replacing, cb_tree def) 03700 { 03701 cb_tree l; 03702 int fill_init = 1; 03703 03704 if (cb_validate_list (vars)) { 03705 return; 03706 } 03707 if (value == NULL && replacing == NULL) { 03708 def = cb_true; 03709 } 03710 if (fillinit == cb_true) { 03711 fill_init = 0; 03712 } 03713 for (l = vars; l; l = CB_CHAIN (l)) { 03714 cb_emit (cb_build_initialize (CB_VALUE (l), value, replacing, def, fill_init)); 03715 } 03716 } 03717 03718 /* 03719 * INSPECT statement 03720 */ 03721 03722 void 03723 cb_emit_inspect (cb_tree var, cb_tree body, cb_tree replacing, int replconv) 03724 { 03725 switch (CB_TREE_TAG(var)) { 03726 case CB_TAG_REFERENCE: 03727 break; 03728 case CB_TAG_INTRINSIC: 03729 switch (CB_TREE_CATEGORY(var)) { 03730 case CB_CATEGORY_ALPHABETIC: 03731 case CB_CATEGORY_ALPHANUMERIC: 03732 case CB_CATEGORY_NATIONAL: 03733 break; 03734 default: 03735 cb_error (_("Invalid target for INSPECT")); 03736 return; 03737 } 03738 break; 03739 case CB_TAG_LITERAL: 03740 break; 03741 default: 03742 cb_error (_("Invalid target for REPLACING/CONVERTING")); 03743 return; 03744 } 03745 if (replconv && sending_id) { 03746 cb_error (_("Invalid target for REPLACING/CONVERTING")); 03747 } 03748 cb_emit (cb_build_funcall_2 ("cob_inspect_init", var, replacing)); 03749 cb_emit_list (body); 03750 cb_emit (cb_build_funcall_0 ("cob_inspect_finish")); 03751 } 03752 03753 void 03754 cb_init_tarrying (void) 03755 { 03756 inspect_func = NULL; 03757 inspect_data = NULL; 03758 } 03759 03760 cb_tree 03761 cb_build_tarrying_data (cb_tree x) 03762 { 03763 inspect_data = x; 03764 return NULL; 03765 } 03766 03767 cb_tree 03768 cb_build_tarrying_characters (cb_tree l) 03769 { 03770 if (inspect_data == NULL) { 03771 cb_error (_("Data name expected before CHARACTERS")); 03772 } 03773 inspect_func = NULL; 03774 return cb_list_add (l, cb_build_funcall_1 ("cob_inspect_characters", inspect_data)); 03775 } 03776 03777 cb_tree 03778 cb_build_tarrying_all (void) 03779 { 03780 if (inspect_data == NULL) { 03781 cb_error (_("Data name expected before ALL")); 03782 } 03783 inspect_func = "cob_inspect_all"; 03784 return NULL; 03785 } 03786 03787 cb_tree 03788 cb_build_tarrying_leading (void) 03789 { 03790 if (inspect_data == NULL) { 03791 cb_error (_("Data name expected before LEADING")); 03792 } 03793 inspect_func = "cob_inspect_leading"; 03794 return NULL; 03795 } 03796 03797 cb_tree 03798 cb_build_tarrying_trailing (void) 03799 { 03800 if (inspect_data == NULL) { 03801 cb_error (_("Data name expected before TRAILING")); 03802 } 03803 inspect_func = "cob_inspect_trailing"; 03804 return NULL; 03805 } 03806 03807 cb_tree 03808 cb_build_tarrying_value (cb_tree x, cb_tree l) 03809 { 03810 if (inspect_func == NULL) { 03811 cb_error_x (x, _("ALL, LEADING or TRAILING expected before '%s'"), cb_name (x)); 03812 } 03813 return cb_list_add (l, cb_build_funcall_2 (inspect_func, inspect_data, x)); 03814 } 03815 03816 cb_tree 03817 cb_build_replacing_characters (cb_tree x, cb_tree l) 03818 { 03819 return cb_list_add (l, cb_build_funcall_1 ("cob_inspect_characters", x)); 03820 } 03821 03822 cb_tree 03823 cb_build_replacing_all (cb_tree x, cb_tree y, cb_tree l) 03824 { 03825 return cb_list_add (l, cb_build_funcall_2 ("cob_inspect_all", y, x)); 03826 } 03827 03828 cb_tree 03829 cb_build_replacing_leading (cb_tree x, cb_tree y, cb_tree l) 03830 { 03831 return cb_list_add (l, cb_build_funcall_2 ("cob_inspect_leading", y, x)); 03832 } 03833 03834 cb_tree 03835 cb_build_replacing_first (cb_tree x, cb_tree y, cb_tree l) 03836 { 03837 return cb_list_add (l, cb_build_funcall_2 ("cob_inspect_first", y, x)); 03838 } 03839 03840 cb_tree 03841 cb_build_replacing_trailing (cb_tree x, cb_tree y, cb_tree l) 03842 { 03843 return cb_list_add (l, cb_build_funcall_2 ("cob_inspect_trailing", y, x)); 03844 } 03845 03846 cb_tree 03847 cb_build_converting (cb_tree x, cb_tree y, cb_tree l) 03848 { 03849 return cb_list_add (l, cb_build_funcall_2 ("cob_inspect_converting", x, y)); 03850 } 03851 03852 cb_tree 03853 cb_build_inspect_region_start (void) 03854 { 03855 return cb_list_init (cb_build_funcall_0 ("cob_inspect_start")); 03856 } 03857 03858 cb_tree 03859 cb_build_inspect_region (cb_tree l, cb_tree pos, cb_tree x) 03860 { 03861 if (pos == CB_BEFORE) { 03862 return cb_list_add (l, cb_build_funcall_1 ("cob_inspect_before", x)); 03863 } else { 03864 return cb_list_add (l, cb_build_funcall_1 ("cob_inspect_after", x)); 03865 } 03866 } 03867 03868 /* 03869 * MOVE statement 03870 */ 03871 03872 static void 03873 warning_destination (cb_tree x) 03874 { 03875 struct cb_reference *r; 03876 struct cb_field *f; 03877 cb_tree loc; 03878 03879 r = CB_REFERENCE (x); 03880 f = CB_FIELD (r->value); 03881 loc = CB_TREE (f); 03882 03883 if (r->offset) { 03884 return; 03885 } 03886 03887 if (!strcmp (f->name, "RETURN-CODE") || 03888 !strcmp (f->name, "SORT-RETURN") || 03889 !strcmp (f->name, "NUMBER-OF-CALL-PARAMETERS")) { 03890 cb_warning (_("Internal register '%s' defined as BINARY-LONG"), f->name); 03891 } else if (f->pic) { 03892 cb_warning_x (loc, _("'%s' defined here as PIC %s"), f->name, f->pic->orig); 03893 } else { 03894 cb_warning_x (loc, _("'%s' defined here as a group of length %d"), f->name, f->size); 03895 } 03896 } 03897 03898 static int 03899 move_error (cb_tree src, cb_tree dst, const size_t value_flag, const int flag, 03900 const int src_flag, const char *msg) 03901 { 03902 cb_tree loc; 03903 03904 if (suppress_warn) { 03905 return 0; 03906 } 03907 loc = src->source_line ? src : dst; 03908 if (value_flag) { 03909 /* VALUE clause */ 03910 cb_warning_x (loc, msg); 03911 } else { 03912 /* MOVE statement */ 03913 if (flag) { 03914 cb_warning_x (loc, msg); 03915 if (src_flag) { 03916 warning_destination (src); 03917 } 03918 warning_destination (dst); 03919 } 03920 } 03921 03922 return 0; 03923 } 03924 03925 /* count the number of free places in an alphanumeric edited field */ 03926 static int 03927 count_pic_alphanumeric_edited (struct cb_field *field) 03928 { 03929 int count; 03930 int repeat; 03931 unsigned char *p; 03932 03933 count = 0; 03934 for (p = (unsigned char *)(field->pic->str); *p; p += 5) { 03935 if (*p == '9' || *p == 'A' || *p == 'X') { 03936 memcpy ((unsigned char *)&repeat, p + 1, sizeof(int)); 03937 count += repeat; 03938 } 03939 } 03940 return count; 03941 } 03942 03943 int 03944 validate_move (cb_tree src, cb_tree dst, size_t is_value) 03945 { 03946 struct cb_field *f; 03947 struct cb_literal *l; 03948 unsigned char *p; 03949 cb_tree loc; 03950 long long val; 03951 size_t i; 03952 size_t is_numeric_edited = 0; 03953 int src_scale_mod; 03954 int dst_scale_mod; 03955 int dst_size_mod; 03956 int size; 03957 int most_significant; 03958 int least_significant; 03959 03960 loc = src->source_line ? src : dst; 03961 if (CB_REFERENCE_P(dst) && CB_ALPHABET_NAME_P(CB_REFERENCE(dst)->value)) { 03962 goto invalid; 03963 } 03964 if (CB_REFERENCE_P(dst) && CB_FILE_P(CB_REFERENCE(dst)->value)) { 03965 goto invalid; 03966 } 03967 if (CB_TREE_CATEGORY (dst) == CB_CATEGORY_BOOLEAN) { 03968 cb_error_x (loc, _("Invalid destination for MOVE")); 03969 return -1; 03970 } 03971 03972 if (CB_TREE_CLASS (dst) == CB_CLASS_POINTER) { 03973 if (CB_TREE_CLASS (src) == CB_CLASS_POINTER) { 03974 return 0; 03975 } else { 03976 goto invalid; 03977 } 03978 } 03979 03980 f = cb_field (dst); 03981 switch (CB_TREE_TAG (src)) { 03982 case CB_TAG_CONST: 03983 if (src == cb_space) { 03984 if (CB_TREE_CATEGORY (dst) == CB_CATEGORY_NUMERIC 03985 || (CB_TREE_CATEGORY (dst) == CB_CATEGORY_NUMERIC_EDITED && !is_value)) { 03986 goto invalid; 03987 } 03988 } else if (src == cb_zero) { 03989 if (CB_TREE_CATEGORY (dst) == CB_CATEGORY_ALPHABETIC) { 03990 goto invalid; 03991 } 03992 } 03993 break; 03994 case CB_TAG_LITERAL: 03995 /* TODO: ALL literal */ 03996 03997 l = CB_LITERAL (src); 03998 if (CB_TREE_CLASS (src) == CB_CLASS_NUMERIC) { 03999 /* Numeric literal */ 04000 if (l->all) { 04001 goto invalid; 04002 } 04003 most_significant = -999; 04004 least_significant = 999; 04005 04006 /* compute the most significant figure place */ 04007 for (i = 0; i < l->size; i++) { 04008 if (l->data[i] != '0') { 04009 break; 04010 } 04011 } 04012 if (i != l->size) { 04013 most_significant = (int) (l->size - l->scale - i - 1); 04014 } 04015 04016 /* compute the least significant figure place */ 04017 for (i = 0; i < l->size; i++) { 04018 if (l->data[l->size - i - 1] != '0') { 04019 break; 04020 } 04021 } 04022 if (i != l->size) { 04023 least_significant = (int) (-l->scale + i); 04024 } 04025 04026 /* value check */ 04027 switch (CB_TREE_CATEGORY (dst)) { 04028 case CB_CATEGORY_ALPHANUMERIC: 04029 case CB_CATEGORY_ALPHANUMERIC_EDITED: 04030 if (is_value) { 04031 goto expect_alphanumeric; 04032 } 04033 04034 if (l->scale == 0) { 04035 goto expect_alphanumeric; 04036 } else { 04037 goto invalid; 04038 } 04039 case CB_CATEGORY_NUMERIC: 04040 if (f->pic->scale < 0) { 04041 /* check for PIC 9(n)P(m) */ 04042 if (least_significant < -f->pic->scale) { 04043 goto value_mismatch; 04044 } 04045 } else if (f->pic->scale > f->pic->size) { 04046 /* check for PIC P(n)9(m) */ 04047 if (most_significant >= f->pic->size - f->pic->scale) { 04048 goto value_mismatch; 04049 } 04050 } 04051 break; 04052 case CB_CATEGORY_NUMERIC_EDITED: 04053 if (is_value) { 04054 goto expect_alphanumeric; 04055 } 04056 04057 /* TODO */ 04058 break; 04059 default: 04060 if (is_value) { 04061 goto expect_alphanumeric; 04062 } 04063 goto invalid; 04064 } 04065 04066 /* sign check */ 04067 if (l->sign != 0 && !f->pic->have_sign) { 04068 if (is_value) { 04069 cb_error_x (loc, _("Data item not signed")); 04070 return -1; 04071 } 04072 if (cb_warn_constant) { 04073 cb_warning_x (loc, _("Ignoring negative sign")); 04074 } 04075 } 04076 04077 /* size check */ 04078 if (f->flag_real_binary || 04079 ((f->usage == CB_USAGE_COMP_5 || 04080 f->usage == CB_USAGE_COMP_X || 04081 f->usage == CB_USAGE_BINARY) && 04082 f->pic->scale == 0)) { 04083 p = l->data; 04084 for (i = 0; i < l->size; i++) { 04085 if (l->data[i] != '0') { 04086 p = &l->data[i]; 04087 break; 04088 } 04089 } 04090 i = l->size - i; 04091 switch (f->size) { 04092 case 1: 04093 if (i > 18) { 04094 goto numlit_overflow; 04095 } 04096 val = cb_get_long_long (src); 04097 if (f->pic->have_sign) { 04098 if (val < -128LL || 04099 val > 127LL) { 04100 goto numlit_overflow; 04101 } 04102 } else { 04103 if (val > 255LL) { 04104 goto numlit_overflow; 04105 } 04106 } 04107 break; 04108 case 2: 04109 if (i > 18) { 04110 goto numlit_overflow; 04111 } 04112 val = cb_get_long_long (src); 04113 if (f->pic->have_sign) { 04114 if (val < -32768LL || 04115 val > 32767LL) { 04116 goto numlit_overflow; 04117 } 04118 } else { 04119 if (val > 65535LL) { 04120 goto numlit_overflow; 04121 } 04122 } 04123 break; 04124 case 3: 04125 if (i > 18) { 04126 goto numlit_overflow; 04127 } 04128 val = cb_get_long_long (src); 04129 if (f->pic->have_sign) { 04130 if (val < -8388608LL || 04131 val > 8388607LL) { 04132 goto numlit_overflow; 04133 } 04134 } else { 04135 if (val > 16777215LL) { 04136 goto numlit_overflow; 04137 } 04138 } 04139 break; 04140 case 4: 04141 if (i > 18) { 04142 goto numlit_overflow; 04143 } 04144 val = cb_get_long_long (src); 04145 if (f->pic->have_sign) { 04146 if (val < -2147483648LL || 04147 val > 2147483647LL) { 04148 goto numlit_overflow; 04149 } 04150 } else { 04151 if (val > 4294967295LL) { 04152 goto numlit_overflow; 04153 } 04154 } 04155 break; 04156 case 5: 04157 if (i > 18) { 04158 goto numlit_overflow; 04159 } 04160 val = cb_get_long_long (src); 04161 if (f->pic->have_sign) { 04162 if (val < -549755813888LL || 04163 val > 549755813887LL) { 04164 goto numlit_overflow; 04165 } 04166 } else { 04167 if (val > 1099511627775LL) { 04168 goto numlit_overflow; 04169 } 04170 } 04171 break; 04172 case 6: 04173 if (i > 18) { 04174 goto numlit_overflow; 04175 } 04176 val = cb_get_long_long (src); 04177 if (f->pic->have_sign) { 04178 if (val < -140737488355328LL || 04179 val > 140737488355327LL) { 04180 goto numlit_overflow; 04181 } 04182 } else { 04183 if (val > 281474976710655LL) { 04184 goto numlit_overflow; 04185 } 04186 } 04187 break; 04188 case 7: 04189 if (i > 18) { 04190 goto numlit_overflow; 04191 } 04192 val = cb_get_long_long (src); 04193 if (f->pic->have_sign) { 04194 if (val < -36028797018963968LL || 04195 val > 36028797018963967LL) { 04196 goto numlit_overflow; 04197 } 04198 } else { 04199 if (val > 72057594037927935LL) { 04200 goto numlit_overflow; 04201 } 04202 } 04203 break; 04204 default: 04205 if (f->pic->have_sign) { 04206 if (i < 19) { 04207 break; 04208 } 04209 if (i > 19) { 04210 goto numlit_overflow; 04211 } 04212 if (memcmp (p, "9223372036854775807", 19) > 0) { 04213 goto numlit_overflow; 04214 } 04215 } else { 04216 if (i < 20) { 04217 break; 04218 } 04219 if (i > 20) { 04220 goto numlit_overflow; 04221 } 04222 if (memcmp (p, "18446744073709551615", 20) > 0) { 04223 goto numlit_overflow; 04224 } 04225 } 04226 break; 04227 } 04228 return 0; 04229 } 04230 if (least_significant < -f->pic->scale) { 04231 goto size_overflow; 04232 } 04233 if (f->pic->scale > 0) { 04234 size = f->pic->digits - f->pic->scale; 04235 } else { 04236 size = f->pic->digits; 04237 } 04238 if (most_significant >= size) { 04239 goto size_overflow; 04240 } 04241 } else { 04242 /* Alphanumeric literal */ 04243 04244 /* value check */ 04245 switch (CB_TREE_CATEGORY (dst)) { 04246 case CB_CATEGORY_ALPHABETIC: 04247 for (i = 0; i < l->size; i++) { 04248 if (!isalpha (l->data[i]) && !isspace (l->data[i])) { 04249 goto value_mismatch; 04250 } 04251 } 04252 break; 04253 case CB_CATEGORY_NUMERIC: 04254 goto expect_numeric; 04255 case CB_CATEGORY_NUMERIC_EDITED: 04256 if (!is_value) { 04257 goto expect_numeric; 04258 } 04259 04260 /* TODO: validate the value */ 04261 break; 04262 default: 04263 break; 04264 } 04265 04266 /* size check */ 04267 size = cb_field_size (dst); 04268 if (size >= 0 && (int)l->size > size) { 04269 goto size_overflow; 04270 } 04271 } 04272 break; 04273 case CB_TAG_FIELD: 04274 case CB_TAG_REFERENCE: 04275 if (CB_REFERENCE_P(src) && 04276 CB_ALPHABET_NAME_P(CB_REFERENCE(src)->value)) { 04277 break; 04278 } 04279 if (CB_REFERENCE_P(src) && 04280 CB_FILE_P(CB_REFERENCE(src)->value)) { 04281 goto invalid; 04282 } 04283 size = cb_field_size (src); 04284 if (size < 0) { 04285 size = cb_field (src)->size; 04286 } 04287 /* non-elementary move */ 04288 if (cb_field (src)->children || cb_field (dst)->children) { 04289 if (size > cb_field (dst)->size) { 04290 goto size_overflow_1; 04291 } 04292 break; 04293 } 04294 04295 /* elementary move */ 04296 switch (CB_TREE_CATEGORY (src)) { 04297 case CB_CATEGORY_ALPHANUMERIC: 04298 switch (CB_TREE_CATEGORY (dst)) { 04299 case CB_CATEGORY_NUMERIC: 04300 case CB_CATEGORY_NUMERIC_EDITED: 04301 if (size > cb_field (dst)->pic->digits) { 04302 goto size_overflow_2; 04303 } 04304 break; 04305 case CB_CATEGORY_ALPHANUMERIC_EDITED: 04306 if (size > 04307 count_pic_alphanumeric_edited (cb_field (dst))) { 04308 goto size_overflow_1; 04309 } 04310 break; 04311 default: 04312 if (size > cb_field (dst)->size) { 04313 goto size_overflow_1; 04314 } 04315 break; 04316 } 04317 break; 04318 case CB_CATEGORY_ALPHABETIC: 04319 case CB_CATEGORY_ALPHANUMERIC_EDITED: 04320 switch (CB_TREE_CATEGORY (dst)) { 04321 case CB_CATEGORY_NUMERIC: 04322 case CB_CATEGORY_NUMERIC_EDITED: 04323 goto invalid; 04324 case CB_CATEGORY_ALPHANUMERIC_EDITED: 04325 if (size > 04326 count_pic_alphanumeric_edited(cb_field (dst))) { 04327 goto size_overflow_1; 04328 } 04329 break; 04330 default: 04331 if (size > cb_field (dst)->size) { 04332 goto size_overflow_1; 04333 } 04334 break; 04335 } 04336 break; 04337 case CB_CATEGORY_NUMERIC: 04338 case CB_CATEGORY_NUMERIC_EDITED: 04339 switch (CB_TREE_CATEGORY (dst)) { 04340 case CB_CATEGORY_ALPHABETIC: 04341 goto invalid; 04342 case CB_CATEGORY_ALPHANUMERIC_EDITED: 04343 is_numeric_edited = 1; 04344 /* Drop through */ 04345 case CB_CATEGORY_ALPHANUMERIC: 04346 if (is_numeric_edited) { 04347 dst_size_mod = count_pic_alphanumeric_edited (cb_field (dst)); 04348 } else { 04349 dst_size_mod = cb_field (dst)->size; 04350 } 04351 if (CB_TREE_CATEGORY (src) == CB_CATEGORY_NUMERIC 04352 && cb_field (src)->pic->scale > 0) { 04353 if (cb_move_noninteger_to_alphanumeric == CB_ERROR) { 04354 goto invalid; 04355 } 04356 cb_warning_x (loc, _("Move non-integer to alphanumeric")); 04357 break; 04358 } 04359 if (CB_TREE_CATEGORY (src) == CB_CATEGORY_NUMERIC 04360 && cb_field (src)->pic->digits > dst_size_mod) { 04361 goto size_overflow_2; 04362 } 04363 if (CB_TREE_CATEGORY (src) == CB_CATEGORY_NUMERIC_EDITED 04364 && cb_field (src)->size > dst_size_mod) { 04365 goto size_overflow_1; 04366 } 04367 break; 04368 default: 04369 src_scale_mod = cb_field (src)->pic->scale < 0 ? 04370 0 : cb_field (src)->pic->scale; 04371 dst_scale_mod = cb_field (dst)->pic->scale < 0 ? 04372 0 : cb_field (dst)->pic->scale; 04373 if (cb_field (src)->pic->digits - src_scale_mod > 04374 cb_field (dst)->pic->digits - dst_scale_mod || 04375 src_scale_mod > dst_scale_mod) { 04376 goto size_overflow_2; 04377 } 04378 break; 04379 } 04380 break; 04381 default: 04382 cb_error_x (loc, _("Invalid source for MOVE")); 04383 return -1; 04384 } 04385 break; 04386 case CB_TAG_INTEGER: 04387 case CB_TAG_BINARY_OP: 04388 case CB_TAG_INTRINSIC: 04389 /* TODO: check this */ 04390 break; 04391 default: 04392 fprintf (stderr, "Invalid tree tag %d\n", CB_TREE_TAG (src)); 04393 ABORT (); 04394 } 04395 return 0; 04396 04397 invalid: 04398 if (is_value) { 04399 cb_error_x (loc, _("Invalid VALUE clause")); 04400 } else { 04401 cb_error_x (loc, _("Invalid MOVE statement")); 04402 } 04403 return -1; 04404 04405 numlit_overflow: 04406 if (is_value) { 04407 cb_error_x (loc, _("Invalid VALUE clause - literal exceeds data size")); 04408 return -1; 04409 } 04410 if (cb_warn_constant) { 04411 cb_warning_x (loc, _("Numeric literal exceeds data size")); 04412 } 04413 return 0; 04414 04415 expect_numeric: 04416 return move_error (src, dst, is_value, cb_warn_strict_typing, 0, 04417 _("Numeric value is expected")); 04418 04419 expect_alphanumeric: 04420 return move_error (src, dst, is_value, cb_warn_strict_typing, 0, 04421 _("Alphanumeric value is expected")); 04422 04423 value_mismatch: 04424 return move_error (src, dst, is_value, cb_warn_constant, 0, 04425 _("Value does not fit the picture string")); 04426 04427 size_overflow: 04428 return move_error (src, dst, is_value, cb_warn_constant, 0, 04429 _("Value size exceeds data size")); 04430 04431 size_overflow_1: 04432 return move_error (src, dst, is_value, cb_warn_truncate, 1, 04433 _("Sending field larger than receiving field")); 04434 04435 size_overflow_2: 04436 return move_error (src, dst, is_value, cb_warn_truncate, 1, 04437 _("Some digits may be truncated")); 04438 } 04439 04440 static cb_tree 04441 cb_build_memset (cb_tree x, int c) 04442 { 04443 int size = cb_field_size (x); 04444 04445 if (size == 1) { 04446 return cb_build_funcall_2 ("$E", x, cb_int (c)); 04447 } else { 04448 return cb_build_funcall_3 ("memset", 04449 cb_build_cast_address (x), 04450 cb_int (c), cb_build_cast_length (x)); 04451 } 04452 } 04453 04454 static cb_tree 04455 cb_build_move_copy (cb_tree src, cb_tree dst) 04456 { 04457 int size = cb_field_size (dst); 04458 04459 if (size == 1) { 04460 return cb_build_funcall_2 ("$F", dst, src); 04461 } else { 04462 return cb_build_funcall_3 ("memcpy", 04463 cb_build_cast_address (dst), 04464 cb_build_cast_address (src), cb_build_cast_length (dst)); 04465 } 04466 } 04467 04468 static cb_tree 04469 cb_build_move_call (cb_tree src, cb_tree dst) 04470 { 04471 return cb_build_funcall_2 ("cob_move", src, dst); 04472 } 04473 04474 static cb_tree 04475 cb_build_move_num_zero (cb_tree x) 04476 { 04477 struct cb_field *f; 04478 04479 f = cb_field (x); 04480 switch (f->usage) { 04481 case CB_USAGE_BINARY: 04482 case CB_USAGE_COMP_5: 04483 case CB_USAGE_COMP_X: 04484 if (f->flag_binary_swap) { 04485 return cb_build_memset (x, 0); 04486 } 04487 switch (f->size) { 04488 #ifdef COB_NON_ALIGNED 04489 case 1: 04490 return cb_build_assign (x, cb_int0); 04491 case 2: 04492 #ifdef COB_SHORT_BORK 04493 if (f->storage != CB_STORAGE_LINKAGE && f->indexes == 0 && 04494 (f->offset % 4 == 0)) { 04495 return cb_build_assign (x, cb_int0); 04496 } 04497 break; 04498 #endif 04499 case 4: 04500 case 8: 04501 if (f->storage != CB_STORAGE_LINKAGE && f->indexes == 0 && 04502 (f->offset % f->size == 0)) { 04503 return cb_build_assign (x, cb_int0); 04504 } 04505 break; 04506 #else 04507 case 1: 04508 case 2: 04509 case 4: 04510 case 8: 04511 return cb_build_assign (x, cb_int0); 04512 #endif 04513 } 04514 return cb_build_memset (x, 0); 04515 case CB_USAGE_DISPLAY: 04516 return cb_build_memset (x, '0'); 04517 case CB_USAGE_PACKED: 04518 return cb_build_funcall_1 ("cob_set_packed_zero", x); 04519 default: 04520 return cb_build_move_call (cb_zero, x); 04521 } 04522 } 04523 04524 static cb_tree 04525 cb_build_move_space (cb_tree x) 04526 { 04527 switch (CB_TREE_CATEGORY (x)) { 04528 case CB_CATEGORY_NUMERIC: 04529 case CB_CATEGORY_ALPHABETIC: 04530 case CB_CATEGORY_ALPHANUMERIC: 04531 return cb_build_memset (x, ' '); 04532 default: 04533 return cb_build_move_call (cb_space, x); 04534 } 04535 } 04536 04537 static cb_tree 04538 cb_build_move_zero (cb_tree x) 04539 { 04540 switch (CB_TREE_CATEGORY (x)) { 04541 case CB_CATEGORY_NUMERIC: 04542 if (cb_field (x)->flag_blank_zero) { 04543 return cb_build_move_space (x); 04544 } else { 04545 return cb_build_move_num_zero (x); 04546 } 04547 case CB_CATEGORY_ALPHABETIC: 04548 case CB_CATEGORY_ALPHANUMERIC: 04549 return cb_build_memset (x, '0'); 04550 default: 04551 return cb_build_move_call (cb_zero, x); 04552 } 04553 } 04554 04555 static cb_tree 04556 cb_build_move_high (cb_tree x) 04557 { 04558 switch (CB_TREE_CATEGORY (x)) { 04559 case CB_CATEGORY_NUMERIC: 04560 case CB_CATEGORY_ALPHABETIC: 04561 case CB_CATEGORY_ALPHANUMERIC: 04562 if (cb_high == cb_norm_high) { 04563 return cb_build_memset (x, 255); 04564 } else { 04565 return cb_build_move_call (cb_high, x); 04566 } 04567 default: 04568 return cb_build_move_call (cb_high, x); 04569 } 04570 } 04571 04572 static cb_tree 04573 cb_build_move_low (cb_tree x) 04574 { 04575 switch (CB_TREE_CATEGORY (x)) { 04576 case CB_CATEGORY_NUMERIC: 04577 case CB_CATEGORY_ALPHABETIC: 04578 case CB_CATEGORY_ALPHANUMERIC: 04579 if (cb_low == cb_norm_low) { 04580 return cb_build_memset (x, 0); 04581 } else { 04582 return cb_build_move_call (cb_low, x); 04583 } 04584 default: 04585 return cb_build_move_call (cb_low, x); 04586 } 04587 } 04588 04589 static cb_tree 04590 cb_build_move_quote (cb_tree x) 04591 { 04592 switch (CB_TREE_CATEGORY (x)) { 04593 case CB_CATEGORY_NUMERIC: 04594 case CB_CATEGORY_ALPHABETIC: 04595 case CB_CATEGORY_ALPHANUMERIC: 04596 return cb_build_memset (x, '"'); 04597 default: 04598 return cb_build_move_call (cb_quote, x); 04599 } 04600 } 04601 04602 #ifdef COB_EBCDIC_MACHINE 04603 static void 04604 cob_put_sign_ascii (unsigned char *p) 04605 { 04606 switch (*p) { 04607 case '0': 04608 *p = (unsigned char)'p'; 04609 return; 04610 case '1': 04611 *p = (unsigned char)'q'; 04612 return; 04613 case '2': 04614 *p = (unsigned char)'r'; 04615 return; 04616 case '3': 04617 *p = (unsigned char)'s'; 04618 return; 04619 case '4': 04620 *p = (unsigned char)'t'; 04621 return; 04622 case '5': 04623 *p = (unsigned char)'u'; 04624 return; 04625 case '6': 04626 *p = (unsigned char)'v'; 04627 return; 04628 case '7': 04629 *p = (unsigned char)'w'; 04630 return; 04631 case '8': 04632 *p = (unsigned char)'x'; 04633 return; 04634 case '9': 04635 *p = (unsigned char)'y'; 04636 return; 04637 } 04638 } 04639 #endif 04640 04641 static void 04642 cob_put_sign_ebcdic (unsigned char *p, const int sign) 04643 { 04644 if (sign < 0) { 04645 switch (*p) { 04646 case '0': 04647 *p = (unsigned char)'}'; 04648 return; 04649 case '1': 04650 *p = (unsigned char)'J'; 04651 return; 04652 case '2': 04653 *p = (unsigned char)'K'; 04654 return; 04655 case '3': 04656 *p = (unsigned char)'L'; 04657 return; 04658 case '4': 04659 *p = (unsigned char)'M'; 04660 return; 04661 case '5': 04662 *p = (unsigned char)'N'; 04663 return; 04664 case '6': 04665 *p = (unsigned char)'O'; 04666 return; 04667 case '7': 04668 *p = (unsigned char)'P'; 04669 return; 04670 case '8': 04671 *p = (unsigned char)'Q'; 04672 return; 04673 case '9': 04674 *p = (unsigned char)'R'; 04675 return; 04676 default: 04677 /* What to do here */ 04678 *p = (unsigned char)'}'; 04679 return; 04680 } 04681 } 04682 switch (*p) { 04683 case '0': 04684 *p = (unsigned char)'{'; 04685 return; 04686 case '1': 04687 *p = (unsigned char)'A'; 04688 return; 04689 case '2': 04690 *p = (unsigned char)'B'; 04691 return; 04692 case '3': 04693 *p = (unsigned char)'C'; 04694 return; 04695 case '4': 04696 *p = (unsigned char)'D'; 04697 return; 04698 case '5': 04699 *p = (unsigned char)'E'; 04700 return; 04701 case '6': 04702 *p = (unsigned char)'F'; 04703 return; 04704 case '7': 04705 *p = (unsigned char)'G'; 04706 return; 04707 case '8': 04708 *p = (unsigned char)'H'; 04709 return; 04710 case '9': 04711 *p = (unsigned char)'I'; 04712 return; 04713 default: 04714 /* What to do here */ 04715 *p = (unsigned char)'{'; 04716 return; 04717 } 04718 /* NOT REACHED */ 04719 } 04720 04721 static cb_tree 04722 cb_build_move_literal (cb_tree src, cb_tree dst) 04723 { 04724 struct cb_literal *l; 04725 struct cb_field *f; 04726 unsigned char *buff; 04727 unsigned char *p; 04728 enum cb_category cat; 04729 int i; 04730 int diff; 04731 int val; 04732 int n; 04733 unsigned char bbyte; 04734 04735 l = CB_LITERAL (src); 04736 f = cb_field (dst); 04737 cat = CB_TREE_CATEGORY (dst); 04738 04739 if (l->all) { 04740 if (cat == CB_CATEGORY_NUMERIC || cat == CB_CATEGORY_NUMERIC_EDITED) { 04741 return cb_build_move_call (src, dst); 04742 } 04743 if (l->size == 1) { 04744 return cb_build_funcall_3 ("memset", 04745 cb_build_cast_address (dst), 04746 cb_int (l->data[0]), cb_build_cast_length (dst)); 04747 } 04748 bbyte = l->data[0]; 04749 for (i = 0; i < (int)l->size; i++) { 04750 if (bbyte != l->data[i]) { 04751 break; 04752 } 04753 bbyte = l->data[i]; 04754 } 04755 if (i == (int)l->size) { 04756 return cb_build_funcall_3 ("memset", 04757 cb_build_cast_address (dst), 04758 cb_int (l->data[0]), cb_build_cast_length (dst)); 04759 } 04760 if (f->size > 128) { 04761 return cb_build_move_call (src, dst); 04762 } 04763 buff = cobc_malloc ((size_t)f->size); 04764 for (i = 0; i < f->size; i++) { 04765 buff[i] = l->data[i % l->size]; 04766 } 04767 return cb_build_funcall_3 ("memcpy", 04768 cb_build_cast_address (dst), 04769 cb_build_string (buff, f->size), cb_build_cast_length (dst)); 04770 } else if ((cat == CB_CATEGORY_NUMERIC 04771 && f->usage == CB_USAGE_DISPLAY 04772 && f->pic->scale == l->scale && !f->flag_sign_leading && !f->flag_sign_separate) 04773 || ((cat == CB_CATEGORY_ALPHABETIC || cat == CB_CATEGORY_ALPHANUMERIC) 04774 && f->size < (int) (l->size + 16) && !cb_field_variable_size (f))) { 04775 buff = cobc_malloc ((size_t)f->size); 04776 diff = (int) (f->size - l->size); 04777 if (cat == CB_CATEGORY_NUMERIC) { 04778 if (diff <= 0) { 04779 memcpy (buff, l->data - diff, (size_t)f->size); 04780 } else { 04781 memset (buff, '0', (size_t)diff); 04782 memcpy (buff + diff, l->data, (size_t)l->size); 04783 } 04784 if (f->pic->have_sign) { 04785 p = &buff[f->size - 1]; 04786 if (cb_display_sign) { 04787 cob_put_sign_ebcdic (p, l->sign); 04788 } else if (l->sign < 0) { 04789 #ifdef COB_EBCDIC_MACHINE 04790 cob_put_sign_ascii (p); 04791 #else 04792 *p += 0x40; 04793 #endif 04794 } 04795 } 04796 } else { 04797 if (f->flag_justified) { 04798 if (diff <= 0) { 04799 memcpy (buff, l->data - diff, (size_t)f->size); 04800 } else { 04801 memset (buff, ' ', (size_t)diff); 04802 memcpy (buff + diff, l->data, (size_t)l->size); 04803 } 04804 } else { 04805 if (diff <= 0) { 04806 memcpy (buff, l->data, (size_t)f->size); 04807 } else { 04808 memcpy (buff, l->data, (size_t)l->size); 04809 memset (buff + l->size, ' ', (size_t)diff); 04810 } 04811 } 04812 } 04813 bbyte = *buff; 04814 if (f->size == 1) { 04815 free (buff); 04816 return cb_build_funcall_2 ("$E", dst, cb_int (bbyte)); 04817 } 04818 for (i = 0; i < f->size; i++) { 04819 if (bbyte != buff[i]) { 04820 break; 04821 } 04822 } 04823 if (i == f->size) { 04824 free (buff); 04825 return cb_build_funcall_3 ("memset", 04826 cb_build_cast_address (dst), 04827 cb_int (bbyte), cb_build_cast_length (dst)); 04828 } 04829 return cb_build_funcall_3 ("memcpy", 04830 cb_build_cast_address (dst), 04831 cb_build_string (buff, f->size), 04832 cb_build_cast_length (dst)); 04833 } else if (cb_fits_int (src) && f->size <= 8 && 04834 (f->usage == CB_USAGE_BINARY || f->usage == CB_USAGE_COMP_5 || 04835 f->usage == CB_USAGE_COMP_X)) { 04836 val = cb_get_int (src); 04837 n = f->pic->scale - l->scale; 04838 if ((l->size + n) > 9) { 04839 return cb_build_move_call (src, dst); 04840 } 04841 for (; n > 0; n--) { 04842 val *= 10; 04843 } 04844 for (; n < 0; n++) { 04845 val /= 10; 04846 } 04847 if (val == 0) { 04848 return cb_build_move_num_zero (dst); 04849 } 04850 if (f->size == 1) { 04851 return cb_build_assign (dst, cb_int (val)); 04852 } 04853 if (f->flag_binary_swap) { 04854 i = (f->size - 1) + (8 * (f->pic->have_sign ? 1 : 0)); 04855 return cb_build_funcall_2 (bin_set_funcs[i], 04856 cb_build_cast_address (dst), 04857 cb_int (val)); 04858 } 04859 switch (f->size) { 04860 case 2: 04861 #ifdef COB_SHORT_BORK 04862 if (f->storage != CB_STORAGE_LINKAGE && f->indexes == 0 && 04863 (f->offset % 4 == 0)) { 04864 return cb_build_assign (dst, cb_int (val)); 04865 } 04866 break; 04867 #endif 04868 case 4: 04869 case 8: 04870 #ifdef COB_NON_ALIGNED 04871 if (f->storage != CB_STORAGE_LINKAGE && f->indexes == 0 && 04872 (f->offset % f->size == 0)) { 04873 return cb_build_assign (dst, cb_int (val)); 04874 } 04875 break; 04876 #else 04877 return cb_build_assign (dst, cb_int (val)); 04878 #endif 04879 } 04880 return cb_build_move_call (src, dst); 04881 } else if (cb_fits_int (src) && f->usage == CB_USAGE_PACKED) { 04882 if (f->pic->scale < 0) { 04883 return cb_build_move_call (src, dst); 04884 } 04885 val = cb_get_int (src); 04886 n = f->pic->scale - l->scale; 04887 if ((l->size + n) > 9) { 04888 return cb_build_move_call (src, dst); 04889 } 04890 for (; n > 0; n--) { 04891 val *= 10; 04892 } 04893 for (; n < 0; n++) { 04894 val /= 10; 04895 } 04896 if (val == 0) { 04897 return cb_build_move_num_zero (dst); 04898 } 04899 return cb_build_funcall_2 ("cob_set_packed_int", dst, cb_int (val)); 04900 } else { 04901 return cb_build_move_call (src, dst); 04902 } 04903 } 04904 04905 static cb_tree 04906 cb_build_move_field (cb_tree src, cb_tree dst) 04907 { 04908 struct cb_field *src_f; 04909 struct cb_field *dst_f; 04910 int src_size; 04911 int dst_size; 04912 04913 src_f = cb_field (src); 04914 src_size = cb_field_size (src); 04915 dst_f = cb_field (dst); 04916 dst_size = cb_field_size (dst); 04917 04918 if ((src_size > 0 && dst_size > 0 && src_size >= dst_size) 04919 && (!cb_field_variable_size (src_f) && !cb_field_variable_size (dst_f))) { 04920 switch (CB_TREE_CATEGORY (src)) { 04921 case CB_CATEGORY_ALPHABETIC: 04922 if (CB_TREE_CATEGORY (dst) == CB_CATEGORY_ALPHABETIC 04923 || CB_TREE_CATEGORY (dst) == CB_CATEGORY_ALPHANUMERIC) { 04924 if (dst_f->flag_justified == 0) { 04925 return cb_build_move_copy (src, dst); 04926 } 04927 } 04928 break; 04929 case CB_CATEGORY_ALPHANUMERIC: 04930 if (CB_TREE_CATEGORY (dst) == CB_CATEGORY_ALPHANUMERIC) { 04931 if (dst_f->flag_justified == 0) { 04932 return cb_build_move_copy (src, dst); 04933 } 04934 } 04935 break; 04936 case CB_CATEGORY_NUMERIC: 04937 if (CB_TREE_CATEGORY (dst) == CB_CATEGORY_NUMERIC 04938 && src_f->usage == dst_f->usage 04939 && src_f->pic->size == dst_f->pic->size 04940 && src_f->pic->digits == dst_f->pic->digits 04941 && src_f->pic->scale == dst_f->pic->scale 04942 && src_f->pic->have_sign == dst_f->pic->have_sign 04943 && src_f->flag_binary_swap == dst_f->flag_binary_swap 04944 && src_f->flag_sign_leading == dst_f->flag_sign_leading 04945 && src_f->flag_sign_separate == dst_f->flag_sign_separate) { 04946 return cb_build_move_copy (src, dst); 04947 } else if (CB_TREE_CATEGORY (dst) == CB_CATEGORY_ALPHANUMERIC 04948 && src_f->usage == CB_USAGE_DISPLAY 04949 && src_f->pic->have_sign == 0 04950 && !src_f->flag_sign_leading 04951 && !src_f->flag_sign_separate) { 04952 return cb_build_move_copy (src, dst); 04953 } 04954 break; 04955 default: 04956 break; 04957 } 04958 } 04959 04960 return cb_build_move_call (src, dst); 04961 } 04962 04963 cb_tree 04964 cb_build_move (cb_tree src, cb_tree dst) 04965 { 04966 struct cb_field *f; 04967 struct cb_field *p; 04968 04969 if (src == cb_error_node || dst == cb_error_node) { 04970 return cb_error_node; 04971 } 04972 04973 if (validate_move (src, dst, 0) < 0) { 04974 return cb_error_node; 04975 } 04976 04977 if (CB_REFERENCE_P (src)) { 04978 CB_REFERENCE (src)->type = CB_SENDING_OPERAND; 04979 } 04980 if (CB_REFERENCE_P (dst)) { 04981 CB_REFERENCE (dst)->type = CB_RECEIVING_OPERAND; 04982 } 04983 04984 if (CB_TREE_CLASS (dst) == CB_CLASS_POINTER) { 04985 return cb_build_assign (dst, src); 04986 } 04987 04988 if (CB_REFERENCE_P (src) && CB_ALPHABET_NAME_P(CB_REFERENCE(src)->value)) { 04989 return cb_build_move_call (src, dst); 04990 } 04991 if (CB_INDEX_P (dst)) { 04992 if (src == cb_null) { 04993 return cb_build_assign (dst, cb_zero); 04994 } 04995 return cb_build_assign (dst, src); 04996 } 04997 04998 if (CB_INDEX_P (src)) { 04999 return cb_build_funcall_2 ("cob_set_int", dst, cb_build_cast_integer (src)); 05000 } 05001 05002 if (CB_INTRINSIC_P (src) || CB_INTRINSIC_P (dst)) { 05003 return cb_build_move_call (src, dst); 05004 } 05005 05006 f = cb_field (dst); 05007 05008 if (CB_EXCEPTION_ENABLE (COB_EC_BOUND_SUBSCRIPT)) { 05009 for (p = f; p; p = p->parent) { 05010 if (p->flag_occurs) { 05011 return cb_build_move_call (src, dst); 05012 } 05013 } 05014 if (CB_REF_OR_FIELD_P (src)) { 05015 for (p = cb_field (src); p; p = p->parent) { 05016 if (p->flag_occurs) { 05017 return cb_build_move_call (src, dst); 05018 } 05019 } 05020 } 05021 } 05022 05023 /* output optimal code */ 05024 if (src == cb_zero) { 05025 return cb_build_move_zero (dst); 05026 } else if (src == cb_space) { 05027 return cb_build_move_space (dst); 05028 } else if (src == cb_high) { 05029 return cb_build_move_high (dst); 05030 } else if (src == cb_low) { 05031 return cb_build_move_low (dst); 05032 } else if (src == cb_quote) { 05033 return cb_build_move_quote (dst); 05034 } else if (CB_LITERAL_P (src)) { 05035 return cb_build_move_literal (src, dst); 05036 } 05037 return cb_build_move_field (src, dst); 05038 } 05039 05040 void 05041 cb_emit_move (cb_tree src, cb_tree dsts) 05042 { 05043 cb_tree l; 05044 05045 if (cb_validate_one (src)) { 05046 return; 05047 } 05048 if (cb_validate_list (dsts)) { 05049 return; 05050 } 05051 05052 for (l = dsts; l; l = CB_CHAIN (l)) { 05053 cb_emit (cb_build_move (src, CB_VALUE (l))); 05054 } 05055 } 05056 05057 /* 05058 * OPEN statement 05059 */ 05060 05061 void 05062 cb_emit_open (cb_tree file, cb_tree mode, cb_tree sharing) 05063 { 05064 if (file == cb_error_node) { 05065 return; 05066 } 05067 file = cb_ref (file); 05068 if (file == cb_error_node) { 05069 return; 05070 } 05071 current_statement->file = file; 05072 05073 if (CB_FILE (file)->organization == COB_ORG_SORT) { 05074 cb_error_x (CB_TREE (current_statement), 05075 _("Operation not allowed on SORT files")); 05076 } 05077 if (sharing == NULL) { 05078 sharing = CB_FILE (file)->sharing ? CB_FILE (file)->sharing : cb_int0; 05079 } 05080 05081 /* READ ONLY */ 05082 if (sharing == cb_int0 && CB_INTEGER (mode)->val != COB_OPEN_INPUT) { 05083 sharing = cb_int1; 05084 } 05085 05086 cb_emit (cb_build_funcall_4 ("cob_open", file, mode, 05087 sharing, CB_FILE(file)->file_status)); 05088 } 05089 05090 /* 05091 * PERFORM statement 05092 */ 05093 05094 void 05095 cb_emit_perform (cb_tree perform, cb_tree body) 05096 { 05097 if (perform == cb_error_node) { 05098 return; 05099 } 05100 CB_PERFORM (perform)->body = body; 05101 cb_emit (perform); 05102 } 05103 05104 cb_tree 05105 cb_build_perform_once (cb_tree body) 05106 { 05107 cb_tree x; 05108 05109 if (body == cb_error_node) { 05110 return cb_error_node; 05111 } 05112 x = cb_build_perform (CB_PERFORM_ONCE); 05113 CB_PERFORM (x)->body = body; 05114 return x; 05115 } 05116 05117 cb_tree 05118 cb_build_perform_times (cb_tree times) 05119 { 05120 cb_tree x; 05121 05122 if (cb_check_integer_value (times) == cb_error_node) { 05123 return cb_error_node; 05124 } 05125 05126 x = cb_build_perform (CB_PERFORM_TIMES); 05127 CB_PERFORM (x)->data = times; 05128 return x; 05129 } 05130 05131 cb_tree 05132 cb_build_perform_until (cb_tree condition, cb_tree varying) 05133 { 05134 cb_tree x; 05135 05136 x = cb_build_perform (CB_PERFORM_UNTIL); 05137 CB_PERFORM (x)->test = condition; 05138 CB_PERFORM (x)->varying = varying; 05139 return x; 05140 } 05141 05142 cb_tree 05143 cb_build_perform_forever (cb_tree body) 05144 { 05145 cb_tree x; 05146 05147 if (body == cb_error_node) { 05148 return cb_error_node; 05149 } 05150 x = cb_build_perform (CB_PERFORM_FOREVER); 05151 CB_PERFORM (x)->body = body; 05152 return x; 05153 } 05154 05155 cb_tree 05156 cb_build_perform_exit (struct cb_label *label) 05157 { 05158 cb_tree x; 05159 05160 x = cb_build_perform (CB_PERFORM_EXIT); 05161 CB_PERFORM (x)->data = CB_TREE (label); 05162 return x; 05163 } 05164 05165 /* 05166 * READ statement 05167 */ 05168 05169 void 05170 cb_emit_read (cb_tree ref, cb_tree next, cb_tree into, cb_tree key, cb_tree lock_opts) 05171 { 05172 int read_opts = 0; 05173 cb_tree file; 05174 cb_tree rec; 05175 05176 if (lock_opts == cb_int1) { 05177 read_opts = COB_READ_LOCK; 05178 } else if (lock_opts == cb_int2) { 05179 read_opts = COB_READ_NO_LOCK; 05180 } else if (lock_opts == cb_int3) { 05181 read_opts = COB_READ_IGNORE_LOCK; 05182 } else if (lock_opts == cb_int4) { 05183 read_opts = COB_READ_WAIT_LOCK; 05184 } 05185 if (ref == cb_error_node) { 05186 return; 05187 } 05188 file = cb_ref (ref); 05189 if (file == cb_error_node) { 05190 return; 05191 } 05192 rec = cb_build_field_reference (CB_FILE (file)->record, ref); 05193 if (CB_FILE (file)->organization == COB_ORG_SORT) { 05194 cb_error_x (CB_TREE (current_statement), 05195 _("Operation not allowed on SORT files")); 05196 } 05197 if (next == cb_int1 || next == cb_int2 || 05198 CB_FILE (file)->access_mode == COB_ACCESS_SEQUENTIAL) { 05199 /* READ NEXT/PREVIOUS */ 05200 if (next == cb_int2) { 05201 if (CB_FILE (file)->organization != COB_ORG_INDEXED) { 05202 cb_error_x (CB_TREE (current_statement), 05203 _("READ PREVIOUS only allowed for INDEXED SEQUENTIAL files")); 05204 } 05205 read_opts |= COB_READ_PREVIOUS; 05206 } else { 05207 read_opts |= COB_READ_NEXT; 05208 } 05209 if (key) { 05210 cb_warning (_("KEY ignored with sequential READ")); 05211 } 05212 cb_emit (cb_build_funcall_4 ("cob_read", file, cb_int0, 05213 CB_FILE(file)->file_status, 05214 cb_int (read_opts))); 05215 } else { 05216 /* READ */ 05217 cb_emit (cb_build_funcall_4 ("cob_read", 05218 file, key ? key : CB_FILE (file)->key, 05219 CB_FILE(file)->file_status, cb_int (read_opts))); 05220 } 05221 if (into) { 05222 current_statement->handler3 = cb_build_move (rec, into); 05223 } 05224 current_statement->file = file; 05225 } 05226 05227 /* 05228 * REWRITE statement 05229 */ 05230 05231 void 05232 cb_emit_rewrite (cb_tree record, cb_tree from, cb_tree lockopt) 05233 { 05234 cb_tree file; 05235 int opts = 0; 05236 05237 if (record == cb_error_node || cb_ref (record) == cb_error_node) { 05238 return; 05239 } 05240 if (!CB_REF_OR_FIELD_P (cb_ref (record))) { 05241 cb_error_x (CB_TREE (current_statement), 05242 _("REWRITE requires a record name as subject")); 05243 return; 05244 } 05245 if (cb_field (record)->storage != CB_STORAGE_FILE) { 05246 cb_error_x (CB_TREE (current_statement), 05247 _("REWRITE subject does not refer to a record name")); 05248 return; 05249 } 05250 file = CB_TREE (CB_FIELD (cb_ref (record))->file); 05251 current_statement->file = file; 05252 if (CB_FILE (file)->organization == COB_ORG_SORT) { 05253 cb_error_x (CB_TREE (current_statement), 05254 _("Operation not allowed on SORT files")); 05255 } else if (current_statement->handler_id == COB_EC_I_O_INVALID_KEY && 05256 (CB_FILE(file)->organization != COB_ORG_RELATIVE && 05257 CB_FILE(file)->organization != COB_ORG_INDEXED)) { 05258 cb_error_x (CB_TREE(current_statement), 05259 _("INVALID KEY clause invalid with this file type")); 05260 } else if ((CB_FILE (file)->lock_mode & COB_LOCK_AUTOMATIC) && lockopt) { 05261 cb_error_x (CB_TREE (current_statement), 05262 _("LOCK clause invalid with file LOCK AUTOMATIC")); 05263 } else if (lockopt == cb_int1) { 05264 opts = COB_WRITE_LOCK; 05265 } 05266 if (from) { 05267 cb_emit (cb_build_move (from, record)); 05268 } 05269 cb_emit (cb_build_funcall_4 ("cob_rewrite", file, record, 05270 cb_int (opts), CB_FILE(file)->file_status)); 05271 } 05272 05273 /* 05274 * RELEASE statement 05275 */ 05276 05277 void 05278 cb_emit_release (cb_tree record, cb_tree from) 05279 { 05280 struct cb_field *f; 05281 cb_tree file; 05282 05283 if (record == cb_error_node) { 05284 return; 05285 } 05286 if (from == cb_error_node) { 05287 return; 05288 } 05289 if (cb_ref (record) == cb_error_node) { 05290 return; 05291 } 05292 if (!CB_REF_OR_FIELD_P (cb_ref (record))) { 05293 cb_error_x (CB_TREE (current_statement), 05294 _("RELEASE requires a record name as subject")); 05295 return; 05296 } 05297 if (cb_field (record)->storage != CB_STORAGE_FILE) { 05298 cb_error_x (CB_TREE (current_statement), 05299 _("RELEASE subject does not refer to a record name")); 05300 return; 05301 } 05302 f = CB_FIELD (cb_ref (record)); 05303 file = CB_TREE (f->file); 05304 if (CB_FILE (file)->organization != COB_ORG_SORT) { 05305 cb_error_x (CB_TREE (current_statement), 05306 _("RELEASE not allowed on this record item")); 05307 return; 05308 } 05309 current_statement->file = file; 05310 if (from) { 05311 cb_emit (cb_build_move (from, record)); 05312 } 05313 cb_emit (cb_build_funcall_1 ("cob_file_release", file)); 05314 } 05315 05316 /* 05317 * RETURN statement 05318 */ 05319 05320 void 05321 cb_emit_return (cb_tree ref, cb_tree into) 05322 { 05323 cb_tree file; 05324 cb_tree rec; 05325 05326 if (ref == cb_error_node) { 05327 return; 05328 } 05329 if (into == cb_error_node) { 05330 return; 05331 } 05332 file = cb_ref (ref); 05333 if (file == cb_error_node) { 05334 return; 05335 } 05336 rec = cb_build_field_reference (CB_FILE (file)->record, ref); 05337 cb_emit (cb_build_funcall_1 ("cob_file_return", file)); 05338 if (into) { 05339 current_statement->handler3 = cb_build_move (rec, into); 05340 } 05341 current_statement->file = file; 05342 } 05343 05344 /* 05345 * ROLLBACK statement 05346 */ 05347 05348 void 05349 cb_emit_rollback (void) 05350 { 05351 cb_emit (cb_build_funcall_0 ("cob_rollback")); 05352 } 05353 05354 /* 05355 * SEARCH statement 05356 */ 05357 05358 static void 05359 search_set_keys (struct cb_field *f, cb_tree x) 05360 { 05361 struct cb_binary_op *p; 05362 int i; 05363 05364 if (CB_REFERENCE_P (x)) { 05365 x = build_cond_88 (x); 05366 } 05367 05368 p = CB_BINARY_OP (x); 05369 switch (p->op) { 05370 case '&': 05371 search_set_keys (f, p->x); 05372 search_set_keys (f, p->y); 05373 break; 05374 case '=': 05375 for (i = 0; i < f->nkeys; i++) { 05376 if (cb_field (p->x) == cb_field (f->keys[i].key)) { 05377 f->keys[i].ref = p->x; 05378 f->keys[i].val = p->y; 05379 break; 05380 } 05381 } 05382 if (i == f->nkeys) { 05383 cb_error_x (x, _("Undeclared key '%s'"), cb_field (p->x)->name); 05384 } 05385 break; 05386 default: 05387 cb_error_x (x, _("Invalid SEARCH ALL condition")); 05388 break; 05389 } 05390 } 05391 05392 static cb_tree 05393 cb_build_search_all (cb_tree table, cb_tree cond) 05394 { 05395 cb_tree c1 = NULL; 05396 cb_tree c2; 05397 struct cb_field *f; 05398 int i; 05399 05400 f = cb_field (table); 05401 /* set keys */ 05402 for (i = 0; i < f->nkeys; i++) { 05403 f->keys[i].ref = NULL; 05404 } 05405 search_set_keys (f, cond); 05406 05407 /* build condition */ 05408 for (i = 0; i < f->nkeys; i++) { 05409 if (f->keys[i].ref) { 05410 if (f->keys[i].dir == COB_ASCENDING) { 05411 c2 = cb_build_binary_op (f->keys[i].ref, '=', f->keys[i].val); 05412 } else { 05413 c2 = cb_build_binary_op (f->keys[i].val, '=', f->keys[i].ref); 05414 } 05415 if (c1 == NULL) { 05416 c1 = c2; 05417 } else { 05418 c1 = cb_build_binary_op (c1, '&', c2); 05419 } 05420 } 05421 } 05422 05423 return cb_build_cond (c1); 05424 } 05425 05426 void 05427 cb_emit_search (cb_tree table, cb_tree varying, cb_tree at_end, cb_tree whens) 05428 { 05429 if (cb_validate_one (table)) { 05430 return; 05431 } 05432 if (cb_validate_one (varying)) { 05433 return; 05434 } 05435 if (table == cb_error_node) { 05436 return; 05437 } 05438 cb_emit (cb_build_search (0, table, varying, at_end, whens)); 05439 } 05440 05441 void 05442 cb_emit_search_all (cb_tree table, cb_tree at_end, cb_tree when, cb_tree stmts) 05443 { 05444 if (cb_validate_one (table)) { 05445 return; 05446 } 05447 if (table == cb_error_node) { 05448 return; 05449 } 05450 cb_emit (cb_build_search (1, table, NULL, at_end, 05451 cb_build_if (cb_build_search_all (table, when), stmts, NULL))); 05452 } 05453 05454 /* 05455 * SET statement 05456 */ 05457 05458 void 05459 cb_emit_setenv (cb_tree x, cb_tree y) 05460 { 05461 cb_emit (cb_build_funcall_2 ("cob_set_environment", x, y)); 05462 } 05463 05464 void 05465 cb_emit_set_to (cb_tree vars, cb_tree x) 05466 { 05467 cb_tree l; 05468 cb_tree v; 05469 struct cb_cast *p; 05470 #if 0 05471 enum cb_class class = CB_CLASS_UNKNOWN; 05472 #endif 05473 05474 if (cb_validate_one (x)) { 05475 return; 05476 } 05477 if (cb_validate_list (vars)) { 05478 return; 05479 } 05480 05481 #if 0 05482 /* determine the class of targets */ 05483 for (l = vars; l; l = CB_CHAIN (l)) { 05484 if (CB_TREE_CLASS (CB_VALUE (l)) != CB_CLASS_UNKNOWN) { 05485 if (class == CB_CLASS_UNKNOWN) { 05486 class = CB_TREE_CLASS (CB_VALUE (l)); 05487 } else if (class != CB_TREE_CLASS (CB_VALUE (l))) { 05488 break; 05489 } 05490 } 05491 } 05492 if (l || (class != CB_CLASS_INDEX && class != CB_CLASS_POINTER)) { 05493 cb_error_x (CB_TREE (current_statement), 05494 _("The targets of SET must be either indexes or pointers")); 05495 return; 05496 } 05497 #endif 05498 05499 if (CB_CAST_P (x)) { 05500 p = CB_CAST (x); 05501 if (p->type == CB_CAST_PROGRAM_POINTER) { 05502 for (l = vars; l; l = CB_CHAIN (l)) { 05503 v = CB_VALUE (l); 05504 if (!CB_REFERENCE_P (v)) { 05505 cb_error_x (CB_TREE (current_statement), 05506 _("SET targets must be PROGRAM-POINTER")); 05507 CB_VALUE (l) = cb_error_node; 05508 } else if (CB_FIELD(cb_ref(v))->usage != CB_USAGE_PROGRAM_POINTER) { 05509 cb_error_x (CB_TREE (current_statement), 05510 _("SET targets must be PROGRAM-POINTER")); 05511 CB_VALUE (l) = cb_error_node; 05512 } 05513 } 05514 } 05515 } 05516 /* validate the targets */ 05517 for (l = vars; l; l = CB_CHAIN (l)) { 05518 v = CB_VALUE (l); 05519 if (CB_CAST_P (v)) { 05520 p = CB_CAST (v); 05521 if (p->type == CB_CAST_ADDRESS 05522 && !CB_FIELD (cb_ref (p->val))->flag_item_based 05523 && CB_FIELD (cb_ref (p->val))->storage != CB_STORAGE_LINKAGE) { 05524 cb_error_x (p->val, _("The address of '%s' cannot be changed"), 05525 cb_name (p->val)); 05526 CB_VALUE (l) = cb_error_node; 05527 } 05528 } 05529 } 05530 if (cb_validate_list (vars)) { 05531 return; 05532 } 05533 05534 for (l = vars; l; l = CB_CHAIN (l)) { 05535 cb_emit (cb_build_move (x, CB_VALUE (l))); 05536 } 05537 } 05538 05539 void 05540 cb_emit_set_up_down (cb_tree l, cb_tree flag, cb_tree x) 05541 { 05542 if (cb_validate_one (x)) { 05543 return; 05544 } 05545 if (cb_validate_list (l)) { 05546 return; 05547 } 05548 for (; l; l = CB_CHAIN (l)) { 05549 if (flag == cb_int0) { 05550 cb_emit (cb_build_add (CB_VALUE (l), x, cb_int0)); 05551 } else { 05552 cb_emit (cb_build_sub (CB_VALUE (l), x, cb_int0)); 05553 } 05554 } 05555 } 05556 05557 void 05558 cb_emit_set_on_off (cb_tree l, cb_tree flag) 05559 { 05560 struct cb_system_name *s; 05561 05562 if (cb_validate_list (l)) { 05563 return; 05564 } 05565 for (; l; l = CB_CHAIN (l)) { 05566 s = CB_SYSTEM_NAME (cb_ref (CB_VALUE (l))); 05567 cb_emit (cb_build_funcall_2 ("cob_set_switch", cb_int (s->token), flag)); 05568 } 05569 } 05570 05571 void 05572 cb_emit_set_true (cb_tree l) 05573 { 05574 cb_tree x; 05575 struct cb_field *f; 05576 cb_tree ref; 05577 cb_tree val; 05578 05579 for (; l; l = CB_CHAIN (l)) { 05580 x = CB_VALUE (l); 05581 if (x == cb_error_node) { 05582 return; 05583 } 05584 if (!(CB_REFERENCE_P (x) && CB_FIELD_P(CB_REFERENCE(x)->value)) 05585 && !CB_FIELD_P (x)) { 05586 cb_error_x (x, _("Invalid SET statement")); 05587 return; 05588 } 05589 f = cb_field (x); 05590 if (f->level != 88) { 05591 cb_error_x (x, _("Invalid SET statement")); 05592 return; 05593 } 05594 ref = cb_build_field_reference (f->parent, x); 05595 val = CB_VALUE (f->values); 05596 if (CB_PAIR_P (val)) { 05597 val = CB_PAIR_X (val); 05598 } 05599 cb_emit (cb_build_move (val, ref)); 05600 } 05601 } 05602 05603 void 05604 cb_emit_set_false (cb_tree l) 05605 { 05606 cb_tree x; 05607 struct cb_field *f; 05608 cb_tree ref; 05609 cb_tree val; 05610 05611 for (; l; l = CB_CHAIN (l)) { 05612 x = CB_VALUE (l); 05613 if (x == cb_error_node) { 05614 return; 05615 } 05616 if (!(CB_REFERENCE_P (x) && CB_FIELD_P(CB_REFERENCE(x)->value)) 05617 && !CB_FIELD_P (x)) { 05618 cb_error_x (x, _("Invalid SET statement")); 05619 return; 05620 } 05621 f = cb_field (x); 05622 if (f->level != 88) { 05623 cb_error_x (x, _("Invalid SET statement")); 05624 return; 05625 } 05626 if (!f->false_88) { 05627 cb_error_x (x, _("Field does not have FALSE clause")); 05628 return; 05629 } 05630 ref = cb_build_field_reference (f->parent, x); 05631 val = CB_VALUE (f->false_88); 05632 if (CB_PAIR_P (val)) { 05633 val = CB_PAIR_X (val); 05634 } 05635 cb_emit (cb_build_move (val, ref)); 05636 } 05637 } 05638 05639 /* 05640 * SORT statement 05641 */ 05642 05643 void 05644 cb_emit_sort_init (cb_tree name, cb_tree keys, cb_tree col) 05645 { 05646 cb_tree l; 05647 struct cb_field *f; 05648 05649 if (cb_validate_list (keys)) { 05650 return; 05651 } 05652 for (l = keys; l; l = CB_CHAIN (l)) { 05653 if (CB_VALUE (l) == NULL) { 05654 CB_VALUE (l) = name; 05655 } 05656 cb_ref (CB_VALUE (l)); 05657 } 05658 05659 if (CB_FILE_P (cb_ref (name))) { 05660 if (CB_FILE (cb_ref (name))->organization != COB_ORG_SORT) { 05661 cb_error_x (name, _("Invalid SORT filename")); 05662 } 05663 cb_field (current_program->cb_sort_return)->count++; 05664 cb_emit (cb_build_funcall_5 ("cob_file_sort_init", cb_ref (name), 05665 cb_int (cb_list_length (keys)), col, 05666 cb_build_cast_address (current_program->cb_sort_return), 05667 CB_FILE(cb_ref (name))->file_status)); 05668 for (l = keys; l; l = CB_CHAIN (l)) { 05669 cb_emit (cb_build_funcall_4 ("cob_file_sort_init_key", cb_ref (name), 05670 CB_PURPOSE (l), 05671 CB_VALUE (l), 05672 cb_int (cb_field (CB_VALUE(l))->offset))); 05673 } 05674 } else { 05675 f = CB_FIELD (cb_ref (name)); 05676 if (keys == NULL) { 05677 cb_error_x (name, _("Table sort without keys not implemented yet")); 05678 } 05679 cb_emit (cb_build_funcall_2 ("cob_table_sort_init", cb_int (cb_list_length (keys)), col)); 05680 for (l = keys; l; l = CB_CHAIN (l)) { 05681 cb_emit (cb_build_funcall_3 ("cob_table_sort_init_key", 05682 CB_PURPOSE (l), 05683 CB_VALUE (l), 05684 cb_int (cb_field (CB_VALUE(l))->offset))); 05685 } 05686 cb_emit (cb_build_funcall_2 ("cob_table_sort", name, 05687 (f->occurs_depending 05688 ? cb_build_cast_integer (f->occurs_depending) 05689 : cb_int (f->occurs_max)))); 05690 } 05691 } 05692 05693 void 05694 cb_emit_sort_using (cb_tree file, cb_tree l) 05695 { 05696 if (cb_validate_list (l)) { 05697 return; 05698 } 05699 for (; l; l = CB_CHAIN (l)) { 05700 if (CB_FILE (cb_ref(CB_VALUE(l)))->organization == COB_ORG_SORT) { 05701 cb_error (_("Invalid SORT USING parameter")); 05702 } 05703 cb_emit (cb_build_funcall_2 ("cob_file_sort_using", 05704 cb_ref (file), cb_ref (CB_VALUE (l)))); 05705 } 05706 } 05707 05708 void 05709 cb_emit_sort_input (cb_tree proc) 05710 { 05711 cb_emit (cb_build_perform_once (proc)); 05712 } 05713 05714 void 05715 cb_emit_sort_giving (cb_tree file, cb_tree l) 05716 { 05717 cb_tree p; 05718 int listlen; 05719 05720 if (cb_validate_list (l)) { 05721 return; 05722 } 05723 for (p = l; p; p = CB_CHAIN (p)) { 05724 if (CB_FILE (cb_ref(CB_VALUE(p)))->organization == COB_ORG_SORT) { 05725 cb_error (_("Invalid SORT GIVING parameter")); 05726 } 05727 } 05728 listlen = cb_list_length (l); 05729 p = cb_build_funcall_2 ("cob_file_sort_giving", cb_ref (file), l); 05730 CB_FUNCALL(p)->varcnt = listlen; 05731 cb_emit (p); 05732 } 05733 05734 void 05735 cb_emit_sort_output (cb_tree proc) 05736 { 05737 cb_emit (cb_build_perform_once (proc)); 05738 } 05739 05740 void 05741 cb_emit_sort_finish (cb_tree file) 05742 { 05743 if (CB_FILE_P (cb_ref (file))) { 05744 cb_emit (cb_build_funcall_1 ("cob_file_sort_close", cb_ref (file))); 05745 } 05746 } 05747 05748 /* 05749 * START statement 05750 */ 05751 05752 void 05753 cb_emit_start (cb_tree file, cb_tree op, cb_tree key) 05754 { 05755 if (cb_validate_one (key)) { 05756 return; 05757 } 05758 if (file != cb_error_node) { 05759 current_statement->file = cb_ref (file); 05760 cb_emit (cb_build_funcall_4 ("cob_start", cb_ref (file), op, 05761 key ? key : CB_FILE (cb_ref (file))->key, 05762 CB_FILE(cb_ref(file))->file_status)); 05763 } 05764 } 05765 05766 /* 05767 * STOP statement 05768 */ 05769 05770 void 05771 cb_emit_stop_run (cb_tree x) 05772 { 05773 cb_emit (cb_build_funcall_1 ("cob_stop_run", cb_build_cast_integer (x))); 05774 } 05775 05776 /* 05777 * STRING statement 05778 */ 05779 05780 void 05781 cb_emit_string (cb_tree items, cb_tree into, cb_tree pointer) 05782 { 05783 cb_tree start; 05784 cb_tree l; 05785 cb_tree end; 05786 cb_tree dlm; 05787 05788 if (cb_validate_one (into)) { 05789 return; 05790 } 05791 if (cb_validate_one (pointer)) { 05792 return; 05793 } 05794 start = items; 05795 cb_emit (cb_build_funcall_2 ("cob_string_init", into, pointer)); 05796 while (start) { 05797 05798 /* find DELIMITED item */ 05799 for (end = start; end; end = CB_CHAIN (end)) { 05800 if (CB_PAIR_P (CB_VALUE (end))) { 05801 break; 05802 } 05803 } 05804 05805 /* cob_string_delimited */ 05806 dlm = end ? CB_PAIR_X (CB_VALUE (end)) : cb_int0; 05807 cb_emit (cb_build_funcall_1 ("cob_string_delimited", dlm)); 05808 05809 /* cob_string_append */ 05810 for (l = start; l != end; l = CB_CHAIN (l)) { 05811 cb_emit (cb_build_funcall_1 ("cob_string_append", CB_VALUE (l))); 05812 } 05813 05814 start = end ? CB_CHAIN (end) : NULL; 05815 } 05816 cb_emit (cb_build_funcall_0 ("cob_string_finish")); 05817 } 05818 05819 /* 05820 * UNLOCK statement 05821 */ 05822 05823 void 05824 cb_emit_unlock (cb_tree ref) 05825 { 05826 cb_tree file; 05827 05828 if (ref != cb_error_node) { 05829 file = cb_ref (ref); 05830 cb_emit (cb_build_funcall_2 ("cob_unlock_file", 05831 file, CB_FILE(file)->file_status)); 05832 current_statement->file = file; 05833 } 05834 } 05835 05836 /* 05837 * UNSTRING statement 05838 */ 05839 05840 void 05841 cb_emit_unstring (cb_tree name, cb_tree delimited, cb_tree into, cb_tree pointer, cb_tree tallying) 05842 { 05843 if (cb_validate_one (name)) { 05844 return; 05845 } 05846 if (cb_validate_one (tallying)) { 05847 return; 05848 } 05849 if (cb_validate_list (delimited)) { 05850 return; 05851 } 05852 if (cb_validate_list (into)) { 05853 return; 05854 } 05855 cb_emit (cb_build_funcall_3 ("cob_unstring_init", name, pointer, 05856 cb_int (cb_list_length (delimited)))); 05857 cb_emit_list (delimited); 05858 cb_emit_list (into); 05859 if (tallying) { 05860 cb_emit (cb_build_funcall_1 ("cob_unstring_tallying", tallying)); 05861 } 05862 cb_emit (cb_build_funcall_0 ("cob_unstring_finish")); 05863 } 05864 05865 cb_tree 05866 cb_build_unstring_delimited (cb_tree all, cb_tree value) 05867 { 05868 if (cb_validate_one (value)) { 05869 return cb_error_node; 05870 } 05871 return cb_build_funcall_2 ("cob_unstring_delimited", value, all); 05872 } 05873 05874 cb_tree 05875 cb_build_unstring_into (cb_tree name, cb_tree delimiter, cb_tree count) 05876 { 05877 if (cb_validate_one (name)) { 05878 return cb_error_node; 05879 } 05880 if (delimiter == NULL) { 05881 delimiter = cb_int0; 05882 } 05883 if (count == NULL) { 05884 count = cb_int0; 05885 } 05886 return cb_build_funcall_3 ("cob_unstring_into", name, delimiter, count); 05887 } 05888 05889 /* 05890 * WRITE statement 05891 */ 05892 05893 void 05894 cb_emit_write (cb_tree record, cb_tree from, cb_tree opt, cb_tree lockopt) 05895 { 05896 cb_tree file; 05897 int val; 05898 05899 if (record != cb_error_node && cb_ref (record) != cb_error_node) { 05900 if (!CB_REF_OR_FIELD_P (cb_ref (record))) { 05901 cb_error_x (CB_TREE (current_statement), 05902 _("WRITE requires a record name as subject")); 05903 return; 05904 } 05905 if (cb_field (record)->storage != CB_STORAGE_FILE) { 05906 cb_error_x (CB_TREE (current_statement), 05907 _("WRITE subject does not refer to a record name")); 05908 return; 05909 } 05910 file = CB_TREE (CB_FIELD (cb_ref (record))->file); 05911 current_statement->file = file; 05912 if (CB_FILE (file)->organization == COB_ORG_SORT) { 05913 cb_error_x (CB_TREE (current_statement), 05914 _("Operation not allowed on SORT files")); 05915 } else if (current_statement->handler_id == COB_EC_I_O_INVALID_KEY && 05916 (CB_FILE(file)->organization != COB_ORG_RELATIVE && 05917 CB_FILE(file)->organization != COB_ORG_INDEXED)) { 05918 cb_error_x (CB_TREE(current_statement), 05919 _("INVALID KEY clause invalid with this file type")); 05920 } else if (lockopt) { 05921 if ((CB_FILE (file)->lock_mode & COB_LOCK_AUTOMATIC)) { 05922 cb_error_x (CB_TREE (current_statement), 05923 _("LOCK clause invalid with file LOCK AUTOMATIC")); 05924 } else if (opt != cb_int0) { 05925 cb_error_x (CB_TREE (current_statement), 05926 _("LOCK clause invalid here")); 05927 } else if (lockopt == cb_int1) { 05928 opt = cb_int (COB_WRITE_LOCK); 05929 } 05930 } 05931 if (from) { 05932 cb_emit (cb_build_move (from, record)); 05933 } 05934 if (CB_FILE (file)->organization == COB_ORG_LINE_SEQUENTIAL && 05935 opt == cb_int0) { 05936 opt = cb_int (COB_WRITE_BEFORE | COB_WRITE_LINES | 1); 05937 } 05938 /* RXW - This is horrible */ 05939 if (current_statement->handler_id == COB_EC_I_O_EOP && 05940 current_statement->handler1) { 05941 if (CB_CAST_P(opt)) { 05942 val = CB_INTEGER(CB_BINARY_OP(CB_CAST(opt)->val)->x)->val; 05943 val |= COB_WRITE_EOP; 05944 CB_BINARY_OP(CB_CAST(opt)->val)->x = cb_int (val); 05945 } else { 05946 val = CB_INTEGER(opt)->val; 05947 val |= COB_WRITE_EOP; 05948 opt = cb_int (val); 05949 } 05950 } 05951 cb_emit (cb_build_funcall_4 ("cob_write", file, record, opt, 05952 CB_FILE(file)->file_status)); 05953 } 05954 } 05955 05956 cb_tree 05957 cb_build_write_advancing_lines (cb_tree pos, cb_tree lines) 05958 { 05959 cb_tree e; 05960 int opt; 05961 05962 opt = (pos == CB_BEFORE) ? COB_WRITE_BEFORE : COB_WRITE_AFTER; 05963 e = cb_build_binary_op (cb_int (opt | COB_WRITE_LINES), '+', lines); 05964 return cb_build_cast_integer (e); 05965 } 05966 05967 cb_tree 05968 cb_build_write_advancing_mnemonic (cb_tree pos, cb_tree mnemonic) 05969 { 05970 int opt; 05971 int token; 05972 05973 token = CB_SYSTEM_NAME (cb_ref (mnemonic))->token; 05974 switch (token) { 05975 case CB_FEATURE_FORMFEED: 05976 opt = (pos == CB_BEFORE) ? COB_WRITE_BEFORE : COB_WRITE_AFTER; 05977 return cb_int (opt | COB_WRITE_PAGE); 05978 case CB_FEATURE_C01: 05979 case CB_FEATURE_C02: 05980 case CB_FEATURE_C03: 05981 case CB_FEATURE_C04: 05982 case CB_FEATURE_C05: 05983 case CB_FEATURE_C06: 05984 case CB_FEATURE_C07: 05985 case CB_FEATURE_C08: 05986 case CB_FEATURE_C09: 05987 case CB_FEATURE_C10: 05988 case CB_FEATURE_C11: 05989 case CB_FEATURE_C12: 05990 opt = (pos == CB_BEFORE) ? COB_WRITE_BEFORE : COB_WRITE_AFTER; 05991 return cb_int (opt | COB_WRITE_CHANNEL | COB_WRITE_PAGE | token); 05992 default: 05993 cb_error_x (mnemonic, _("Invalid mnemonic name")); 05994 return cb_error_node; 05995 } 05996 } 05997 05998 cb_tree 05999 cb_build_write_advancing_page (cb_tree pos) 06000 { 06001 int opt = (pos == CB_BEFORE) ? COB_WRITE_BEFORE : COB_WRITE_AFTER; 06002 06003 return cb_int (opt | COB_WRITE_PAGE); 06004 }
 1.7.4
 1.7.4