flat assembler
Message board for the users of flat assembler.
Index
> Compiler Internals > db 0. A bug or not a bug? Goto page Previous 1, 2, 3, 4 Next |
Author |
|
revolution 28 Feb 2012, 00:45
Tomasz Grysztar wrote: OK, after all I managed to overcome my concerns and I started working on it (I mean adding the additional sign bit to internal calculations). |
|||
28 Feb 2012, 00:45 |
|
LocoDelAssembly 28 Feb 2012, 01:14
l_inc, If you replace all the occurrences of qword with dword you'll get 'Inconsistent store-load' message too even with the current fasm version. Perhaps it is time to introduce yet another feature like "load signed ..."?
|
|||
28 Feb 2012, 01:14 |
|
revolution 28 Feb 2012, 01:23
Code: macro signExtend.dword v.out, v.in { v.out = v.in or ((v.in and (1 shl 31)) * (((1 shl 32)-1) shl 1)) } |
|||
28 Feb 2012, 01:23 |
|
l_inc 28 Feb 2012, 01:59
LocoDelAssembly
Quote: l_inc, If you replace all the occurrences of qword with dword you'll get 'Inconsistent store-load' message too even with the current fasm version. I know. That's why I explicitly written: "There will be no way...". There is a way now: to use qword and dq. Quote: Perhaps it is time to introduce yet another feature like "load signed ..."? Not that simple. After a value is stored with store or dq, it's sign would be irrevocably lost. And "signed" load would not be able to decide, what sign should be loaded. revolution I'm not sure, who's post you're answering. But if mine: First, I suppose, this won't work with qword values because of the overflow. And second, the purpose is to get a guaranteed same value as was stored in current addressing space, not to sign extend a value (if stored 0xFFFFFFFF, I want to get the same value back, not the -1). |
|||
28 Feb 2012, 01:59 |
|
revolution 28 Feb 2012, 02:08
l_inc: You are trying to squeeze a 65-bit value into a 64-bit hole. You will always lose some information that way. If you really need to keep all the info then you have to make space for an extra bit somewhere and store the sign there. You can't get something for nothing.
Code: x = -1 store qword x at 0 ;we lost the extra sign bit, the hole was not big enough to take the full number store byte (x shr 64) at 8 ;if we need the sign bit then we gotta store it somewhere! |
|||
28 Feb 2012, 02:08 |
|
l_inc 28 Feb 2012, 02:17
revolution
Quote: You are trying to squeeze a 65-bit value into a 64-bit hole. Yes. In this way I try to show, why I don't like the improvement. Quote:
I assume, this is exactly, what is not going to be possible. I can live with that solution (even though I will need to adjust a significant part of my macros, so that they work correctly after the improvement), but my prognose is, that the old simple and working Code: dq x load y qword from $-8 will bloat into 8-16 lines with ugly if-else-end if's. |
|||
28 Feb 2012, 02:17 |
|
revolution 28 Feb 2012, 02:23
This will only affect you if you do some sort of arithmetic that requires the current particular sign handling i.e. if you use division or multiplication on the values loaded back from the memory store. I don't see how simple data movement will be affected.
|
|||
28 Feb 2012, 02:23 |
|
l_inc 28 Feb 2012, 02:29
revolution
An example of an affected macro: Code: ;Displays a decimal representation of a number macro dispDec num*, padding, leader, trailer { local digCount,tenPow,number number = num virtual dq number load number qword from $$ end virtual if number < 0 display '-' number = -number end if digCount = 0 tenPow = 1 while tenPow <= number tenPow = tenPow*10 digCount = digCount + 1 end while if digCount = 0 digCount = 1 tenPow = 10 end if if ~ leader eq display leader end if if ~ padding eq if digCount < padding times (padding-digCount) display '0' end if end if repeat digCount tenPow = tenPow/10 display number/tenPow + '0' number = number mod tenPow end repeat if ~ trailer eq display trailer end if } |
|||
28 Feb 2012, 02:29 |
|
Tomasz Grysztar 28 Feb 2012, 08:06
LOAD and STORE are just a special data directives that load or store some amount of bytes, just like DQ. The way you use them here is equal to displaying the content of bytes generated by DQ directive, and you already loose some information, like relocation (if the value is relocatable then such display is not very useful, since you no longer know what is it relative to). If you hoped that this would be a kind of complete storage of numerical constant, then you misinterpreted the purpose of those directives. You just display what bytes fasm generates when asked to write the value as 8-byte one.
Adding the 65-th bit is going to clean up some quirks in which DQ behaved differently from DD/DW/DB - thus for any calculations that fit in 64 bits it is going to get transparent what format fasm uses internally (that is: DQ will too become compatible with what it would be like if fasm used 129-bit, or even arbitrary 2-adic precision internally). It may break some things if someone misused the old QWORD bugs. However the above macro will still do exactly what it does now, it will display the value of 8 bytes generated with DQ directive. |
|||
28 Feb 2012, 08:06 |
|
l_inc 28 Feb 2012, 10:11
Tomasz Grysztar
Quote: Adding the 65-th bit is going to clean up some quirks in which DQ behaved differently from DD/DW/DB I know what the improvement is intended to do. IMHO it breaks too many things. Quote: It may break some things if someone misused the old QWORD bugs. There has been never provided a definition for use and misuse. Especially because (as you also said) there were found (and became standard) interesting usage examples for directives never intended for those purposes. Quote: However the above macro will still do exactly what it does now, it will display the value of 8 bytes generated with DQ directive. Doing the same means to return same result for any usage. It won't return the same result for dispDec -1, will it? Quote: and you already loose some information, like relocation (if the value is relocatable then such display is not very useful, since you no longer know what is it relative to) We already discussed, that this behaviour is exactly the purpose, but forget this macro: I wouldn't like to repeat the same discussion again. Consider the following macro: Code: ;allows to define a sorted array of signed dwords macro ddSorted [vals*] { common local count,qw1,qw2 count = 0 virtual forward dq vals count = count + 1 common repeat count-1 repeat count-% load qw1 qword from $$+(%-1)*8 load qw2 qword from $$+%*8 if qw1 > qw2 store qword qw2 at $$+(%-1)*8 store qword qw1 at $$+%*8 end if end repeat end repeat count = 0 forward local newvals load newvals qword from $$+count*8 count = count + 1 common end virtual forward dd newvals } Will it give the same result for the code ddSorted -2,2,0,+1,-1 as now? Or is the compilation going to fail? Any ideas of how to achieve the same with the new fasm behaviour? |
|||
28 Feb 2012, 10:11 |
|
revolution 28 Feb 2012, 10:18
There is a standard technique that can be used to sort signed values with an unsigned comparison.
Code: v1 ;value1 v2 ;value2 ;signed compare if v1 > v2 ... ;unsigned compare if (v1 xor (1 shl 63)) > (v2 xor (1 shl 63)) ... |
|||
28 Feb 2012, 10:18 |
|
l_inc 28 Feb 2012, 10:24
revolution
I assume, you didn't understand the operation. The sign is not defined by the MSB. It's lost by storing values. And there is no efficient way to sort an array under fasm without storing values. Btw. the provided macro need to perform the same way with qwords after changing the last dd with dq. |
|||
28 Feb 2012, 10:24 |
|
Tomasz Grysztar 28 Feb 2012, 10:27
l_inc wrote: There has been never provided a definition for use and misuse. Especially because (as you also said) there were found (and became standard) interesting usage examples for directives never intended for those purposes. l_inc wrote: Doing the same means to return same result for any usage. It won't return the same result for dispDec -1, will it? l_inc wrote: Will it give the same result for the code ddSorted -2,2,0,+1,-1 as now? Or is the compilation going to fail? Any ideas of how to achieve the same with the new fasm behaviour? |
|||
28 Feb 2012, 10:27 |
|
revolution 28 Feb 2012, 10:33
l_inc wrote: revolution Code: ;65-bit -2 = 0x1fff...ffe -1 = 0x1fff...fff 0 = 0x0000...000 +1 = 0x0000...001 +2 = 0x0000...002 ;64-bit (like now) -2 = 0x0fff...ffe -1 = 0x0fff...fff 0 = 0x0000...000 +1 = 0x0000...001 +2 = 0x0000...002 It is only different when you try to overflow the storage. Code: dq -(1 shl 63+1) ;What will happen? It should give an error |
|||
28 Feb 2012, 10:33 |
|
Tomasz Grysztar 28 Feb 2012, 10:38
These problems arise from the fact that undocumented behaviors were used. That's another reason why I want to do this change - I want to have a consistent behavior independent of internal precision, so the values that fasm generates will be the same that infinite precision fasm would generate and the only difference from infinite precision fasm will be that it will fail with "out of range" error due to limitations of the internal representation.
After the corrections fasm will detect overflow not only on multiplication, but also on sum and substraction, and even in case of "shl", as fasm's "shl" operator was designed to not truncate values ("db 7 shl 8" does not generate truncated value, it overflows - and so should "dq 7 shl 62"). revolution, I'm sorry, your macro will probably not work because of this. |
|||
28 Feb 2012, 10:38 |
|
Tomasz Grysztar 28 Feb 2012, 10:59
revolution wrote:
|
|||
28 Feb 2012, 10:59 |
|
revolution 28 Feb 2012, 11:16
Tomasz Grysztar wrote: revolution, I'm sorry, your macro will probably not work because of this. |
|||
28 Feb 2012, 11:16 |
|
Tomasz Grysztar 28 Feb 2012, 11:32
Oh, sorry, I did not look at it in detail, I thought that you are trying to generate a sign bit with shifting. Something like:
Code: macro signExtend.dword v.out, v.in { if ~ v.in and 80000000h v.out = v.in else v.out = v.in and 7FFFFFFFh - 80000000h end if } |
|||
28 Feb 2012, 11:32 |
|
LocoDelAssembly 28 Feb 2012, 15:42
Would this if-less version work and be compatible with any future extra precision?
Code: macro def [name, size] { macro signExtend.#name v.out, v.in \{ v.out = v.in or (0 - v.in shr (size - 1)) shl (size - 1) \} } def byte, 8,\ word, 16,\ dword, 32,\ qword, 64 purge def But still, I think something like "load signed" should be native, and also store and data definition directives should support both signed and unsigned for easy range checking (if nothing is specified then the default is used which is the union of both if I understand the current behavior correctly) |
|||
28 Feb 2012, 15:42 |
|
Goto page Previous 1, 2, 3, 4 Next < Last Thread | Next Thread > |
Forum Rules:
|
Copyright © 1999-2025, Tomasz Grysztar. Also on GitHub, YouTube.
Website powered by rwasa.