Tvůrce webu je i pro tebe! Postav třeba web. Bez grafika. Bez kodéra. Hned.
wz

Obsah / Utility / TEXTFORM / BinToTextBufN

Zdrojový kód: INCLUDE\UTIL\TEXTFORM.INC, UTIL\TEXTFORM.ASM

Související:

BinToTextBuf   Zformátování čísla BIN do bufferu
FormToTextBufN   Délka formátovaného textu

BinToTextBufN - Délka textu formátovaného čísla BIN

Funkce BinToTextBufN zjistí délku formátovaného čísla BIN v bufferu.


; -----------------------------------------------------------------------------
;             Convert BIN number into text buffer - get text length
; -----------------------------------------------------------------------------
; INPUT:	EDX:EAX = number
;		EBX = formatting parameters FORMPAR
; OUTPUT:	EAX = text length
; -----------------------------------------------------------------------------

Na vstupu funkce obsahuje registrový pár EDX:EAX číslo bez znaménka ke konverzi a registr EBX formátovací parametry FORMPAR (ne ukazatel). Na výstupu je navrácena v registru EAX délka textu.


BinToTextBufN:	call	Bin0ToTextBufN	; get text length without spaces
		push	ebx		; push EBX
		movzx	ebx,bh		; EBX <- minimal width of field
		cmp	ebx,eax		; check text length
		jb	BinToTextBufN8	; text length is OK
		xchg	eax,ebx		; EAX <- new text length
BinToTextBufN8:	pop	ebx		; pop EBX
BinToTextBufN9:	ret

Funkce BinToTextBufN volá interní funkci Bin0ToTextBufN, která nezapočítává požadovanou šířku pole. Porovnáním s požadovanou minimální šířkou se ověří, zda je délka textu čísla větší než požadovaná minimální šířka. Pokud je menší, navrátí funkce požadovanou minimální šířku namísto délky textu bez mezer.


; -----------------------------------------------------------------------------
;      Convert BIN number into text buffer - get text length without spaces
; -----------------------------------------------------------------------------
; INPUT:	EDX:EAX = number
;		EBX = formatting parameters FORMPAR
; OUTPUT:	EAX = text length
; -----------------------------------------------------------------------------

Funkce Bin0ToTextBufN zjistí délku textu bez uvažování minimální šířky textu. Na vstupu funkce obsahuje registrový pár EDX:EAX číslo bez znaménka ke konverzi a registr EBX formátovací parametry FORMPAR (ne ukazatel). Na výstupu je navrácena v registru EAX délka textu.


; ------------- Push registers

Bin0ToTextBufN:	push	edx		; push EDX
		push	edi		; push EDI

; ------------- Get number of bits in number LOW (-> AL)

		bsr	eax,eax		; AL <- highest bit in number LOW
		setz	ah		; AH <- 1 if EAX == 0, 0 if EAX != 0
		add	al,1		; AL <- number of bits in number LOW
		dec	ah		; AH <- 0 if EAX == 0, 0ffh if EAX != 0
		and	al,ah		; clear AL if number LOW is zero

; ------------- Get number of bits in number HIGH + 32 (-> DL)

		bsr	edx,edx		; DL <- highest bit in number HIGH
		setz	dh		; DH <- 1 if EDX == 0, 0 if EDX != 0
		add	dl,32+1		; DL <- add HIGH correction
		dec	dh		; DH <- 0 if EDX == 0, 0ffh if EDX != 0
		and	dl,dh		; clear DL if number HIGH is zero

; ------------- Put together HIGH and LOW bits (-> DL)

		not	dh		; DH <- 0ffh if EDX == 0, 0 if EDX != 0
		and	al,dh		; clear AL if number HIGH is not zero
		or	dl,al		; DL <- number of bits in the number

; ------------- Get number of digits in the number (-> EDX)

		movzx	edx,dl		; EDX <- number of digits in the number

Pro výpočet počtu číslic BIN čísla je nutné nejdříve zjistit pozici nejvyššího bitu v čísle. Číslo má velikost 2 dvojregistry, proto musí být informace sloučena z obou dvojregistrů.

Nejdříve bude vyhledána pozice nejvyššího bitu v registru EAX, který obsahuje nižší část čísla (LOW). Instrukce BSR uloží do registru EAX bitový offset nejvyššího nalezeného bitu 1. Pokud byl obsah registru EAX nulový, nastaví se příznak ZF. To využije následující instrukce SETZ, která uloží do registru AH hodnotu 1 v případě nulového obsahu čísla, jinak do AH uloží 0. Přičtením 1 k AL se obdrží v registru AL počet bitů původního čísla LOW. Avšak pokud byl obsah čísla LOW nulový, je obsah registru AL nedefinovaný. Proto je AL zamaskován maskou vygenerovanou z příznaku ZF v registru AH. Výsledkem je, že registr AL obsahuje buď počet bitů v původní nižší části čísla nebo nulu pokud byla nižší část čísla nulová.

Podobným způsobem je zjištěn počet bitů ve vyšší části čísla (HIGH) s tím rozdílem, že počet bitů je uložen do registru DL a je o 32 vyšší.

Další operací se údaje z nižší a vyšší části čísla sloučí do jednoho údaje. Pokud bylo číslo větší než DWORD, má maska v registru DH hodnotu 0FFh. Pokud se číslo vešlo do nižšího DWORD, má maska v DH hdnotu 0. Proto inverzí masky v DH a zamaskováním počtu bitů v AL z nižší části čísla se počet bitů vynuluje v případě, že číslo bylo větší než DWORD. Následnou instrukcí OR se údaje sloučí do registru DL, který ve výsledku obsahuje počet bitů v registrovém páru EDX:EAX.

V binárním formátu čísla odpovídá počet bitů přímo počtu číslic.


; ------------- Limit minimal number of digits (-> EDX)

		cmp	dl,bl		; check minimal number of digits
		ja	Bin0ToTextBufN2	; number of digits is OK
		mov	dl,bl		; EDX <- minimal numer of digits

Vypočtená délka čísla se zvýší podle požadovaného minimálního počtu číslic (zadaného položkou FORMPAR_Prec formátovacích parametrů).


; ------------- Check if use zeros instead of spaces

Bin0ToTextBufN2:bt	ebx,FORMFLAG_Zero_b ; add zeros instead of spaces?
		jnc	Bin0ToTextBufN6	; no zeros

; ------------- Prepare width of field (-> EAX)

		movzx	eax,bh		; EAX <- minimal width of field
		bt	ebx,FORMFLAG_Alt_b ; use prefix?
		jnc	Bin0ToTextBufN3	; not using prefix
		sub	al,2		; without prefix
		jbe	Bin0ToTextBufN6	; width is low
Bin0ToTextBufN3:bt	ebx,FORMFLAG_Alt2_b ; use suffix?
		jnc	Bin0ToTextBufN4	; not using suffix
		sub	al,1		; without suffix
		jbe	Bin0ToTextBufN6	; width is low

; ------------- Check if use thousand separator

Bin0ToTextBufN4:bt	ebx,FORMFLAG_Thsn_b ; use thousand separator?
		jnc	Bin0ToTextBufN5	; not using thousand separator

; ------------- Number of digits without thousand separators (-> EAX)

		inc	eax		; EAX <- width + 1
		imul	eax,eax,58255	; multiply with 10000h*8/9, round up
		shr	eax,16		; EAX <- max. number of digits

; ------------- New minimal number of digits

Bin0ToTextBufN5:cmp	eax,edx		; check number of digits
		jb	Bin0ToTextBufN6	; number of digits is not greater
		xchg	eax,edx		; EDX <- new text length

Další výpočet musí zajistit doplnění nul k číslu, je-li aktivní příznakový bit FORMFLAG_Zero_b.

Do registru EAX se připraví požadovaná minimální šířka pole (z registru BH, tj. parametr FORMPAR_Width formátovacích parametrů) snížená o případný prefix a sufix. Při malé šířce pole se nuly přidávat nebudou.

Je-li požadován oddělovač řádů (přepínač FORMFLAG_Thsn_b), musí se od počtu znaků odečíst počet oddělovačů. Na 8 číslic připadá 1 znak oddělovače, přičemž před oddělovačem musí být vždy minimálně 1 číslice, proto počet číslic pro danou šířku pole EAX se vypočte inkrementací šířky pole o 1 a vynásobením číslem 8/9.

Po výpočtu počtu číslic potřebných k doplnění čísla nulami se výsledek porovná se skutečnou šířkou čísla a případně se počet číslic zvýší.


; ------------- Add thousand separator

Bin0ToTextBufN6:bt	ebx,FORMFLAG_Thsn_b ; use thousand separator?
		jnc	Bin0ToTextBufN7	; not using thousand separator
		mov	eax,edx		; EAX <- number of digits
		dec	eax		; without 1 digit
		jle	Bin0ToTextBufN7	; no digit
		shr	eax,3		; EAX <- number of separators
		add	edx,eax		; EDX <- add thousand separators

Je-li požadován oddělovač řádů (je nastaven přepínač FORMFLAG_Thsn_b), je potřeba k délce čísla přičíst oddělovače řádů. Na 8 číslic čísla připadá 1 oddělovač řádů, přičemž před oddělovačem musí být minimálně jedna číslice. Proto se počet oddělovačů získá snížením počtu číslic o 1 a vydělením údaje osmi. Vypočtený počet oddělovačů se přičte ke střadači délky textu v registru EDX.


; ------------- Add prefix "0b"

Bin0ToTextBufN7:bt	ebx,FORMFLAG_Alt_b ; use prefix?
		jnc	Bin0ToTextBufN8	; not using prefix
		cmp	edx,byte 1	; check minimal number of digits
		adc	edx,byte 2	; EDX <- add 2 or 3 characters

Je-li požadován prefix "0b" před číslem (je nastaven přepínač FORMFLAG_Alt_b), zvýší se délka textu o 2. Současně je zde zajištěno kontrolou minimálního počtu číslic, aby v případě prefixu mělo číslo alespoň 1 číslici (tj. aby při nule byl text "0b0").


; ------------- Add suffix "b"

Bin0ToTextBufN8:bt	ebx,FORMFLAG_Alt2_b ; use suffix?
		jnc	Bin0ToTextBufN9	; not using suffix
		cmp	edx,byte 1	; check minimal number of digits
		adc	edx,byte 1	; EDX <- add 1 or 2 characters

; ------------- Pop registers

Bin0ToTextBufN9:xchg	eax,edx		; EAX <- text length
		pop	edi		; pop EDI
		pop	edx		; pop EDX
		ret

Je-li požadován sufix "b" za číslem (je nastaven přepínač FORMFLAG_Alt2_b), zvýší se délka textu o 1. Současně je zde zajištěno kontrolou minimálního počtu číslic, aby v případě sufixu mělo číslo alespoň 1 číslici (tj. aby při nule byl text "0b").


Obsah / Utility / TEXTFORM / BinToTextBufN