{{pfw:banner.png}} ====== Print Hex ====== In this programming pearl Albert Nijhof shows how to print hexadecimal numbers with a given number of digits even if your Forth system does not provide pictured numeric output by means of ''%%<#%%'' ''%%#%%'' ''%%#>%%'' etc. 1 \ Tools -- Formatted unsigned single hex number output, not using BASE 2 \ an-17jan2022 3 4 decimal 5 : .1HX ( x -- ) \ Print last digit of x in hex 6 15 and \ lowest nibble 7 9 over < 7 and + \ for A..F 8 [char] 0 + emit ; 9 10 : .NHX ( x n -- ) \ Print last n digits of x in hex 11 1 max 16 min >r \ x r: n 12 r@ 1 ?do dup 4 rshift loop \ collect on data stack 13 r> 0 do .1hx loop space ; 14 \ ----- end of code ----- 15 16 (* 17 Examples 18 decimal 19 19150 2 .nhx \ CE 20 19150 3 .nhx \ ACE 21 19150 4 .nhx \ 4ACE 22 19150 8 .nhx \ 00004ACE 23 24 \ .NHX version without DO-LOOP 25 : .NHX ( x n -- ) \ Print last n digits of x in hex 26 swap >r 1 max 16 min dup \ n n r: x 27 begin 1- dup while r@ 4 rshift >r \ collect on return stack 28 repeat drop 29 begin r> .1hx 1- dup 0= until drop space ; 30 *) 31 \ <><> ===== Idea ===== In order to print numbers you have to extract the digits one by one and print each digit. As typical written number representation starts with the most signficant digit but extracting is easier starting with the least significant digit, the order of digits needs to be reversed. This could be done by storing the digits either on the data stack or on the return stack and make use of their last in first out property. ===== Display a single digit: .1HX ===== The word ''%%.1HX%%'' (//print 1 digit in hex//, line 5-8) displays a single digit, the last digit of the unsigned number ''%%x%%''. Line 6 extracts the least significant nibble of ''%%x%%'' and ignores its more significant part. To display that nibble as a character the word uses arithmetic with comparison results: The phrase ''%%9 over <%%'' in line 7 compares the nibble with ''%%9%%''. If it is greater than ''%%9%%'' then the phrase results in -1 (all bits set).\\ If the nibble is not greater than ''%%9%%'' (i.e. 0 to 9) the phrase results in 0 (all bits 0). ''%%7 and%%'' extracts the least 3 bits. Now we have either ''%%7%%'' or ''%%0%%'', that we add to the nibble itself. This leads to the nibble+7, if nibble is greater ''%%9%%'' or to the original nibble value, if it is not greater that ''%%9%%''. This is to bridge the gap in the ASCII character sequence between '9' and 'A' as we see now. If we look at the ASCII character sequence we see: 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 '0' '1' '2' '3' '4' '5' '6' '7' '8' '9' ':' ';' '<' '=' '>' '?' '@' 'A' 'B' ... There is a gap between the character '9' and the character 'A' which is 7 characters wide. Line 8 adds the character value of ''%%'0'%%'' to the intermediate value. Nibbles 0 to 9 will be mapped to character '0' to '9'. Nibbles 10 to 15 will be mapped (bridging the gap) to character 'A' to 'F' as is required for hexadecimal display. This character is eventually displayed by the ''%%emit%%'' at the end of ''%%.1HX%%'' (line 8). ===== Display n digits: .NHX ===== In order to display complete numbers the word ''%%.NHX%%'' (//print n digits in hex//) is defined (line 10-13). ''%%x%%'' is the number to display and ''%%n%%'' ist the number to digits to show. Line 10 does some clipping so that ''%%n%%'' is always in the range 1 to 16. This avoids unpleasant suprises of seemingly endlesss output if the passed ''%%n%%'' happens to be outside that range. To display complete numbers ''%%.NHX%%'' first iterates n-1 times (''%%DO LOOP%%'' on line 12) and puts shifted numbers on the stack so that each numbers least significant nibble is on of te nibbles of ''%%x%%''. The number with the least significant nibble of ''%%x%%'' first, the number with the most significant nibble of ''%%x%%'' on top of stack. After that the nibbles on the stack are processed in reversed order. They are displayed via ''%%.1HX%%'' (''%%DO LOOP%%'', line 13) that runs ''%%n%%'' times.\\ ''%%.NHX%%'' ends by printing a space so that consecutive calls to ''%%.NHX%%'' will print number space separated. ===== No DO LOOP ===== If your system does not provide ''%%DO LOOP%%'' then you can rewrite the loops using ''%%BEGIN WHILE REPEAT%%'' or ''%%BEGIN UNTIL%%'' as you can see in line 27-29. Line 26 does parameter clipping of ''%%n%%'' as before. To avoid stack juggling the first loop (line 28) collects nibbles on the return stack. (That wouldn't have been possible with ''%%DO LOOP%%'' above as the loop parameters block the return stack.) The second loop (line 29) retrieves the nibbles from the return stack and prints them with ''%%.1HX%%'' as above. When finished it drops the loop parameter. The obligatory space ends the number output (line 29). ===== Example output ===== Lines 18-22 show hot to use ''%%.NHX%%'' and what output you can expect. If the number of digits ''%%n%%'' given is less then the actual numbers of digits to completely display ''%%x%%'' then ''%%.NHX%%'' truncates the display showing only the least signiicant digits. If ''%%n%%'' is larger, then ''%%.NHX%%'' adds leading '0's. ---- uho 2022-01-20