ugBASIC 1.18
An isomorphic BASIC language compiler for retrocomputers
Loading...
Searching...
No Matches
put_image.c
Go to the documentation of this file.
1/*****************************************************************************
2 * ugBASIC - an isomorphic BASIC language compiler for retrocomputers *
3 *****************************************************************************
4 * Copyright 2021-2026 Marco Spedaletti (asimov@mclink.it)
5 *
6 * Licensed under the Apache License, Version 2.0 (the "License");
7 * you may not use this file except in compliance with the License.
8 * You may obtain a copy of the License at
9 *
10 * http://www.apache.org/licenses/LICENSE-2.0
11 *
12 * Unless required by applicable law or agreed to in writing, software
13 * distributed under the License is distributed on an "AS IS" BASIS,
14 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15 * See the License for the specific language governing permissions and
16 * limitations under the License.
17 *----------------------------------------------------------------------------
18 * Concesso in licenza secondo i termini della Licenza Apache, versione 2.0
19 * (la "Licenza"); è proibito usare questo file se non in conformità alla
20 * Licenza. Una copia della Licenza è disponibile all'indirizzo:
21 *
22 * http://www.apache.org/licenses/LICENSE-2.0
23 *
24 * Se non richiesto dalla legislazione vigente o concordato per iscritto,
25 * il software distribuito nei termini della Licenza è distribuito
26 * "COSÌ COM'È", SENZA GARANZIE O CONDIZIONI DI ALCUN TIPO, esplicite o
27 * implicite. Consultare la Licenza per il testo specifico che regola le
28 * autorizzazioni e le limitazioni previste dalla medesima.
29 ****************************************************************************/
30
31/****************************************************************************
32 * INCLUDE SECTION
33 ****************************************************************************/
34
35#include "../../ugbc.h"
36
37/****************************************************************************
38 * CODE SECTION
39 ****************************************************************************/
40
41extern char DATATYPE_AS_STRING[][16];
42
53void put_image_vars_original( Environment * _environment, char * _image, char * _x1, char * _y1, char * _x2, char * _y2, char * _frame, char * _sequence, char * _flags ) {
54
55 if ( _environment->emptyProcedure ) {
56 return;
57 }
58
60
61 Variable * image = variable_retrieve( _environment, _image );
62
63 image->usedImage = 1;
64
65 Resource * resource = build_resource_for_sequence( _environment, _image, _frame, _sequence );
66
67 Variable * x1 = variable_retrieve_or_define( _environment, _x1, VT_POSITION, 0 );
68 Variable * y1 = variable_retrieve_or_define( _environment, _y1, VT_POSITION, 0 );
69 Variable * realFrame = NULL;
70 Variable * frame = NULL;
71 if ( _frame) {
72 frame = variable_retrieve_or_define( _environment, _frame, VT_BYTE, 0 );
73 realFrame = frame;
74 }
75 Variable * sequence = NULL;
76 if ( _sequence) {
77 sequence = variable_retrieve_or_define( _environment, _sequence, VT_BYTE, 0 );
78 realFrame = frame;
79 }
80
81 switch( resource->type ) {
82 case VT_SEQUENCE:
83 if ( image->bankAssigned != -1 ) {
84
85 char alreadyLoadedLabel[MAX_TEMPORARY_STORAGE];
86 sprintf(alreadyLoadedLabel, "%salready", label );
87
88 char bankWindowId[MAX_TEMPORARY_STORAGE];
89 sprintf( bankWindowId, "BANKWINDOWID%2.2x", image->residentAssigned );
90
91 char bankWindowName[MAX_TEMPORARY_STORAGE];
92 sprintf( bankWindowName, "BANKWINDOW%2.2x", image->residentAssigned );
93
94 // cpu_compare_and_branch_16bit_const( _environment, bankWindowId, image->variableUniqueId, alreadyLoadedLabel, 1 );
95 // if ( image->uncompressedSize ) {
96 // bank_uncompress_semi_var( _environment, image->bankAssigned, image->absoluteAddress, bankWindowName );
97 // } else {
98 // bank_read_semi_var( _environment, image->bankAssigned, image->absoluteAddress, bankWindowName, image->size );
99 // }
100 // cpu_store_16bit(_environment, bankWindowId, image->variableUniqueId );
101 // cpu_label( _environment, alreadyLoadedLabel );
102
103 Variable * frameSize = variable_temporary( _environment, VT_WORD, "(temporary)");
104 variable_store( _environment, frameSize->name, image->frameSize );
105 Variable * bank = variable_temporary( _environment, VT_BYTE, "(temporary)");
106 variable_store( _environment, bank->name, image->bankAssigned );
107 Variable * offset = variable_temporary( _environment, VT_ADDRESS, "(temporary)");
108
109 if ( !sequence ) {
110 if ( !frame ) {
111 vic2_calculate_sequence_frame_offset(_environment, offset->realName, "", "", image->frameSize, image->frameCount );
112 } else {
113 vic2_calculate_sequence_frame_offset(_environment, offset->realName, "", frame->realName, image->frameSize, image->frameCount );
114 }
115 } else {
116 if ( !frame ) {
117 vic2_calculate_sequence_frame_offset(_environment, offset->realName, sequence->realName, "", image->frameSize, image->frameCount );
118 } else {
119 vic2_calculate_sequence_frame_offset(_environment, offset->realName, sequence->realName, frame->realName, image->frameSize, image->frameCount );
120 }
121 }
122
123 Variable * address = variable_temporary( _environment, VT_ADDRESS, "(temporary)");
124 variable_store( _environment, address->name, image->absoluteAddress );
125 variable_add_inplace_vars( _environment, address->name, offset->name );
126 bank_read_vars_direct( _environment, bank->name, address->name, bankWindowName, frameSize->name );
127
128 Resource resource;
129 resource.realName = strdup( bankWindowName );
130 resource.isAddress = 0;
131
132 vic2_put_image( _environment, &resource, x1->name, y1->name, NULL, NULL, image->frameSize, 0, _flags );
133
134 } else {
135 if ( !sequence ) {
136 if ( !frame ) {
137 vic2_put_image( _environment, resource, x1->name, y1->name, "", "", image->frameSize, image->frameCount, _flags );
138 } else {
139 vic2_put_image( _environment, resource, x1->name, y1->name, frame->realName, "", image->frameSize, image->frameCount, _flags );
140 }
141 } else {
142 if ( !frame ) {
143 vic2_put_image( _environment, resource, x1->name, y1->name, "", sequence->realName, image->frameSize, image->frameCount, _flags );
144 } else {
145 vic2_put_image( _environment, resource, x1->name, y1->name, frame->realName, sequence->realName, image->frameSize, image->frameCount, _flags );
146 }
147 }
148 }
149 break;
150 case VT_IMAGES:
151 if ( image->bankAssigned != -1 ) {
152
153 char alreadyLoadedLabel[MAX_TEMPORARY_STORAGE];
154 sprintf(alreadyLoadedLabel, "%salready", label );
155
156 char bankWindowId[MAX_TEMPORARY_STORAGE];
157 sprintf( bankWindowId, "BANKWINDOWID%2.2x", image->residentAssigned );
158
159 char bankWindowName[MAX_TEMPORARY_STORAGE];
160 sprintf( bankWindowName, "BANKWINDOW%2.2x", image->residentAssigned );
161
162 // cpu_compare_and_branch_16bit_const( _environment, bankWindowId, image->variableUniqueId, alreadyLoadedLabel, 1 );
163 // if ( image->uncompressedSize ) {
164 // bank_uncompress_semi_var( _environment, image->bankAssigned, image->absoluteAddress, bankWindowName );
165 // } else {
166 // bank_read_semi_var( _environment, image->bankAssigned, image->absoluteAddress, bankWindowName, image->size );
167 // }
168 // cpu_store_16bit(_environment, bankWindowId, image->variableUniqueId );
169 // cpu_label( _environment, alreadyLoadedLabel );
170
171 Variable * frameSize = variable_temporary( _environment, VT_WORD, "(temporary)");
172 variable_store( _environment, frameSize->name, image->frameSize );
173 Variable * bank = variable_temporary( _environment, VT_BYTE, "(temporary)");
174 variable_store( _environment, bank->name, image->bankAssigned );
175 Variable * offset = variable_temporary( _environment, VT_ADDRESS, "(temporary)");
176
177 if ( sequence ) {
178 if ( image->strips ) {
179 realFrame = variable_temporary( _environment, VT_BYTE, "(real frame)" );
180 outline1("LDA %s", sequence->realName);
181 outline0("ASL");
182 outline0("TAY");
183 outline1("LDA #<%sstrip", image->realName);
184 outline0("STA TMPPTR");
185 outline1("LDA #>%sstrip", image->realName);
186 outline0("STA TMPPTR+1");
187 outline0("LDA (TMPPTR),Y" );
188 outline0("STA TMPPTR2");
189 outline0("INY" );
190 outline0("LDA (TMPPTR),Y" );
191 outline0("STA TMPPTR2+1");
192 outline1("LDA (%s)", frame->realName );
193 outline0("TAY" );
194 outline0("LDA (TMPPTR2),Y" );
195 outline1("STA %s", realFrame->realName );
196 } else {
198 }
199 }
200
201 if ( !frame ) {
202 vic2_calculate_sequence_frame_offset(_environment, offset->realName, NULL, "", image->frameSize, 0 );
203 } else {
204 vic2_calculate_sequence_frame_offset(_environment, offset->realName, NULL, realFrame->realName, image->frameSize, 0 );
205 }
206
207 Variable * address = variable_temporary( _environment, VT_ADDRESS, "(temporary)");
208 variable_store( _environment, address->name, image->absoluteAddress );
209 variable_add_inplace_vars( _environment, address->name, offset->name );
210 bank_read_vars_direct( _environment, bank->name, address->name, bankWindowName, frameSize->name );
211
212 Resource resource;
213 resource.realName = strdup( bankWindowName );
214 resource.isAddress = 0;
215
216 vic2_put_image( _environment, &resource, x1->name, y1->name, NULL, NULL, image->frameSize, 0, _flags );
217
218 } else {
219
220 if ( sequence ) {
221 if ( image->strips ) {
222 realFrame = variable_temporary( _environment, VT_BYTE, "(real frame)" );
223 outline1("LDA %s", sequence->realName);
224 outline0("ASL");
225 outline0("TAY");
226 outline1("LDA #<%sstrip", image->realName);
227 outline0("STA TMPPTR");
228 outline1("LDA #>%sstrip", image->realName);
229 outline0("STA TMPPTR+1");
230 outline0("LDA (TMPPTR),Y" );
231 outline0("STA TMPPTR2");
232 outline0("INY" );
233 outline0("LDA (TMPPTR),Y" );
234 outline0("STA TMPPTR2+1");
235 outline1("LDA (%s)", frame->realName );
236 outline0("TAY" );
237 outline0("LDA (TMPPTR2),Y" );
238 outline1("STA %s", realFrame->realName );
239 } else {
241 }
242 }
243
244 if ( !frame ) {
245 vic2_put_image( _environment, resource, x1->name, y1->name, "", NULL, image->frameSize, 0, _flags );
246 } else {
247 vic2_put_image( _environment, resource, x1->name, y1->name, realFrame->realName, NULL, image->frameSize, 0, _flags );
248 }
249 }
250 break;
251 case VT_IMAGE:
252 case VT_TARRAY:
253 if ( image->bankAssigned != -1 ) {
254
255 char alreadyLoadedLabel[MAX_TEMPORARY_STORAGE];
256 sprintf(alreadyLoadedLabel, "%salready", label );
257
258 char bankWindowId[MAX_TEMPORARY_STORAGE];
259 sprintf( bankWindowId, "BANKWINDOWID%2.2x", image->residentAssigned );
260
261 char bankWindowName[MAX_TEMPORARY_STORAGE];
262 sprintf( bankWindowName, "BANKWINDOW%2.2x", image->residentAssigned );
263
264 cpu_compare_and_branch_16bit_const( _environment, bankWindowId, image->variableUniqueId, alreadyLoadedLabel, 1 );
265 if ( image->uncompressedSize ) {
266 bank_uncompress_semi_var( _environment, image->bankAssigned, image->absoluteAddress, bankWindowName );
267 } else {
268 bank_read_semi_var( _environment, image->bankAssigned, image->absoluteAddress, bankWindowName, image->size );
269 }
270 cpu_store_16bit(_environment, bankWindowId, image->variableUniqueId );
271 cpu_label( _environment, alreadyLoadedLabel );
272
273 Resource resource;
274 resource.realName = strdup( bankWindowName );
275 resource.isAddress = 0;
276
277 vic2_put_image( _environment, &resource, x1->name, y1->name, NULL, NULL, 1, 0, _flags );
278 } else {
279 vic2_put_image( _environment, resource, x1->name, y1->name, NULL, NULL, 1, 0, _flags );
280 }
281 break;
282 default:
284 }
285
286
287}
288
289void put_image_vars_imageref( Environment * _environment, char * _image, char * _x1, char * _y1, char * _x2, char * _y2, char * _frame, char * _sequence, char * _flags ) {
290
292
293 Variable * image = variable_retrieve( _environment, _image );
294
295 Variable * x1 = variable_retrieve_or_define( _environment, _x1, VT_POSITION, 0 );
296 Variable * y1 = variable_retrieve_or_define( _environment, _y1, VT_POSITION, 0 );
297 Variable * frame = NULL;
298 if ( _frame) {
299 frame = variable_retrieve_or_define( _environment, _frame, VT_BYTE, 0 );
300 }
301 Variable * sequence = NULL;
302 if ( _sequence) {
303 sequence = variable_retrieve_or_define( _environment, _sequence, VT_BYTE, 0 );
304 }
305
306 Variable * address = variable_temporary( _environment, VT_ADDRESS, "(stub)" );
307
308 if ( !_environment->putImageRefUnsafe ) {
309 outline1("LDA %s", address_displacement( _environment, image->realName, "5") );
310 outline1("BNE %sskipx", label );
311 outline1("JMP %sskip", label );
312 outhead1("%sskipx:", label );
313 }
314
315 // Y = OFFSET
316
317 if ( _sequence ) {
318
319 outline1("LDA %s", image->realName );
320 outline0("STA TMPPTR" );
321 outline1("LDA %s", address_displacement( _environment, image->realName, "1") );
322 outline0("STA TMPPTR+1" );
323
324 outline0("CLC" );
325 outline0("LDA TMPPTR" );
326 outline0("ADC #3" );
327 outline0("STA TMPPTR" );
328 outline0("LDA TMPPTR+1" );
329 outline0("ADC #0" );
330 outline0("STA TMPPTR+1" );
331
332 if ( strlen(_sequence) == 0 ) {
333
334 } else {
335 outline1("LDA %s", sequence->realName );
336 outline0("STA MATHPTR0" );
337 cpu_call_indirect( _environment, address_displacement( _environment, image->realName, "10") );
338 }
339 if ( _frame ) {
340 if ( strlen(_frame) == 0 ) {
341
342 } else {
343 outline1("LDA %s", _frame );
344 outline0("STA MATHPTR0" );
345 cpu_call_indirect( _environment, address_displacement( _environment, image->realName, "8") );
346 }
347 }
348
349 outline0("LDA TMPPTR" );
350 outline1("STA %s", address->realName );
351 outline0("LDA TMPPTR+1" );
352 outline1("STA %s", address_displacement(_environment, address->realName, "1") );
353
354 } else {
355
356 if ( _frame ) {
357
358 outline1("LDA %s", image->realName );
359 outline0("STA TMPPTR" );
360 outline1("LDA %s", address_displacement( _environment, image->realName, "1") );
361 outline0("STA TMPPTR+1" );
362
363 outline0("CLC" );
364 outline0("LDA TMPPTR" );
365 outline0("ADC #3" );
366 outline0("STA TMPPTR" );
367 outline0("LDA TMPPTR+1" );
368 outline0("ADC #0" );
369 outline0("STA TMPPTR+1" );
370 if ( strlen(_frame) == 0 ) {
371
372 } else {
373 outline1("LDA %s", frame->realName );
374 outline0("STA MATHPTR0" );
375 cpu_call_indirect( _environment, address_displacement( _environment, image->realName, "8") );
376 }
377
378 outline0("LDA TMPPTR" );
379 outline1("STA %s", address->realName );
380 outline0("LDA TMPPTR+1" );
381 outline1("STA %s", address_displacement(_environment, address->realName, "1") );
382
383 } else {
384
385 outline1("LDA %s", image->realName );
386 outline1("STA %s", address->realName );
387 outline1("LDA %s", address_displacement( _environment, image->realName, "1") );
388 outline1("STA %s", address_displacement(_environment, address->realName, "1") );
389
390 }
391
392 }
393
394 Resource resource;
395 resource.realName = strdup( address->realName );
396 resource.isAddress = 1;
397
398 vic2_put_image( _environment, &resource, x1->name, y1->name, NULL, NULL, 1, 0, _flags );
399
400 if ( !_environment->putImageRefUnsafe ) {
401 outhead1("%sskip:", label );
402 }
403
404}
405
406void put_image_vars( Environment * _environment, char * _image, char * _x1, char * _y1, char * _x2, char * _y2, char * _frame, char * _sequence, char * _flags ) {
407
408 if ( _environment->emptyProcedure ) {
409 return;
410 }
411
412 Variable * image = variable_retrieve( _environment, _image );
413
414 switch( image->type ) {
415 case VT_IMAGE:
416 case VT_IMAGES:
417 case VT_SEQUENCE:
418 case VT_ADDRESS:
419 put_image_vars_original( _environment, _image, _x1, _y1, _x2, _y2, _frame, _sequence, _flags );
420 break;
421 case VT_IMAGEREF:
422 put_image_vars_imageref( _environment, _image, _x1, _y1, _x2, _y2, _frame, _sequence, _flags );
423 break;
424 default:
426 }
427
428}
429
430void put_image_vars_flags( Environment * _environment, char * _image, char * _x1, char * _y1, char * _x2, char * _y2, char * _frame, char * _sequence, int _flags ) {
431
432 char flagsConstantName[MAX_TEMPORARY_STORAGE]; sprintf( flagsConstantName, "PUTIMAGEFLAGS%4.4x", _flags );
433 char flagsConstantParameter[MAX_TEMPORARY_STORAGE]; sprintf( flagsConstantParameter, "#PUTIMAGEFLAGS%4.4x", _flags );
434
435 Constant * flagsConstant = constant_find( _environment, flagsConstantName );
436
437 if ( !flagsConstant ) {
438 flagsConstant = malloc( sizeof( Constant ) );
439 memset( flagsConstant, 0, sizeof( Constant ) );
440 flagsConstant->name = strdup( flagsConstantName );
441 flagsConstant->realName = strdup( flagsConstantName );
442 flagsConstant->value = _flags;
443 flagsConstant->type = CT_INTEGER;
444 flagsConstant->next = _environment->constants;
445 _environment->constants = flagsConstant;
446 }
447
448 put_image_vars( _environment, _image, _x1, _y1, _x2, _y2, _frame, _sequence, flagsConstantParameter );
449}
void cpu_store_16bit(Environment *_environment, char *_destination, int _value)
CPU 6309: emit code to store 16 bit
Definition 6309.c:1503
void cpu_compare_and_branch_16bit_const(Environment *_environment, char *_source, int _destination, char *_label, int _positive)
CPU 6309: emit code to compare two 8 bit values and jump if they are equal/different
Definition 6309.c:1578
void cpu_label(Environment *_environment, char *_label)
Definition 6309.c:356
void cpu_call_indirect(Environment *_environment, char *_value)
Definition 6309.c:3765
Variable * variable_retrieve(Environment *_environment, char *_name)
Variable * variable_retrieve_or_define(Environment *_environment, char *_name, VariableType _type, int _value)
Resource * build_resource_for_sequence(Environment *_environment, char *_image, char *_frame, char *_sequence)
void variable_add_inplace_vars(Environment *_environment, char *_source, char *_destination)
Add two variable and return the sum of them on the first.
Variable * variable_temporary(Environment *_environment, VariableType _type, char *_meaning)
Define a temporary variable.
Variable * variable_store(Environment *_environment, char *_destination, unsigned int _value)
Store a direct value to a variable.
char * address_displacement(Environment *_environment, char *_address, char *_displacement)
Constant * constant_find(Environment *_environment, char *_name)
int offset
Definition _optimizer.c:681
void put_image_vars_original(Environment *_environment, char *_image, char *_x1, char *_y1, char *_x2, char *_y2, char *_frame, char *_sequence, char *_flags)
Emit ASM code for PUT IMAGE [image] AT [int],[int].
Definition put_image.c:53
void put_image_vars_imageref(Environment *_environment, char *_image, char *_x1, char *_y1, char *_x2, char *_y2, char *_frame, char *_sequence, char *_flags)
Definition put_image.c:289
void put_image_vars_flags(Environment *_environment, char *_image, char *_x1, char *_y1, char *_x2, char *_y2, char *_frame, char *_sequence, int _flags)
Definition put_image.c:430
void put_image_vars(Environment *_environment, char *_image, char *_x1, char *_y1, char *_x2, char *_y2, char *_frame, char *_sequence, char *_flags)
Definition put_image.c:406
void bank_read_semi_var(Environment *_environment, int _bank, int _address1, char *_address2, int _size)
Emit ASM code for instruction BANK READ ....
Definition bank_read.c:50
void bank_read_vars_direct(Environment *_environment, char *_bank, char *_address1, char *_address2, char *_size)
Definition bank_read.c:138
void bank_uncompress_semi_var(Environment *_environment, int _bank, int _address1, char *_address2)
Emit ASM code for instruction BANK UNCOMPRESS ....
char * name
Definition ugbc.h:800
int value
Definition ugbc.h:815
ConstantType type
Definition ugbc.h:805
struct _Constant * next
Definition ugbc.h:832
char * realName
Definition ugbc.h:803
int putImageRefUnsafe
Definition ugbc.h:3271
Constant * constants
Definition ugbc.h:2611
int emptyProcedure
Definition ugbc.h:2932
int isAddress
Definition ugbc.h:557
VariableType type
Definition ugbc.h:559
char * realName
Definition ugbc.h:555
char * name
Definition ugbc.h:979
VariableType type
Definition ugbc.h:988
char * realName
Definition ugbc.h:982
void * malloc(YYSIZE_T)
struct _Resource Resource
#define MAX_TEMPORARY_STORAGE
Definition ugbc.h:563
@ CT_INTEGER
Definition ugbc.h:788
struct _Variable Variable
Structure of a single variable.
struct _Environment Environment
Structure of compilation environment.
@ VT_TARRAY
Definition ugbc.h:480
@ VT_WORD
Definition ugbc.h:455
@ VT_POSITION
Definition ugbc.h:468
@ VT_BYTE
Definition ugbc.h:450
@ VT_IMAGEREF
Definition ugbc.h:537
@ VT_IMAGES
Definition ugbc.h:495
@ VT_ADDRESS
Definition ugbc.h:465
@ VT_IMAGE
Definition ugbc.h:489
@ VT_SEQUENCE
Definition ugbc.h:513
struct _Constant Constant
Structure of a single constant.
#define CRITICAL_CANNOT_PUT_IMAGE_WITHOUT_STRIP(s)
Definition ugbc.h:3855
#define outline0(s)
Definition ugbc.h:4252
#define CRITICAL_PUT_IMAGE_UNSUPPORTED(v, t)
Definition ugbc.h:3532
#define outline1(s, a)
Definition ugbc.h:4253
#define MAKE_LABEL
Definition ugbc.h:3351
#define outhead1(s, a)
Definition ugbc.h:4247
char DATATYPE_AS_STRING[][16]
void vic2_calculate_sequence_frame_offset(Environment *_environment, char *_offset, char *_sequence, char *_frame, int _frame_size, int _frame_count)
Definition vic2.c:3931
void vic2_put_image(Environment *_environment, Resource *_image, char *_x, char *_y, char *_frame, char *_sequence, int _frame_size, int _frame_count, char *_flags)
Definition vic2.c:3871