Oxygen Basic

Information => Open Forum => Topic started by: Charles Pegge on October 31, 2014, 11:37:29 PM

Title: DLLC
Post by: Charles Pegge on October 31, 2014, 11:37:29 PM
Hi John, Mike

There are a number of DLLCO2 examples in ProjectsB/ScriptBasic/DLLC/

They require Oxygen for JIT compiling, as well as DLLC.

I'll zip the whole folder below, since I've been editing a few of the examples.

Here is a piece of Assembly code working in SB. It is treated just like any other o2 function source code.

dllco2_EAsm
Code: [Select]

  include "dllcinc.sb"

  oxy=dllfile("/scriptbasic/modules/oxygen.dll")

  o2_basic = dllproc( oxy, "o2_basic i =(c*source) " )
  o2_exec  = dllproc( oxy, "o2_exec  i =(i call)   " )
  o2_error = dllproc( oxy, "o2_error c*=()         " )
  o2_errno = dllproc( oxy, "o2_errno i =()         " )
  o2_len   = dllproc( oxy, "o2_len   i =()         " )
  o2_mode  = dllproc( oxy, "o2_mode     (i mode)   " )

  dllcall o2_mode,1

' ==============================
  src="""
  extern

  function reverse(char*s)
  ========================
  addr ecx,s
  mov edx,0
 .rlen
  mov al,[ecx]
  cmp al,0
  jz xlen
  inc edx
  inc ecx
  jmp rlen
 .xlen
  ;
  addr ecx,s
  add  edx,ecx
  dec ecx
  ;
 .rswap
  inc ecx
  dec edx
  cmp edx,ecx
  jle xswap
  mov al,[ecx]
  mov ah,[edx]
  mov [ecx],ah
  mov [edx],al
  jmp rswap
 .xswap
  end function

  sub finish()
  terminate
  end sub

  function link(sys n) as sys
  select n
  case 0 : return @finish
  case 1 : return @reverse
  end select
  end function

  end extern
 

  addr link
  """
' ==============================

  function oxygen(src)
  dllcall o2_basic,src
  if (dllcall(o2_errno)<>0) then
    dllprnt dllcall(o2_error)
    a=0
  else
    a=dllcall(o2_exec,0)
  end if
  oxygen=a
  end function
  '
  a=oxygen(src)
  '
  if (a<>0) then
  '
' ==============================
  '
  Finish  = dllproc(a,"Finish     ()        ", dllcald(a,0) )
  Reverse = dllproc(a,"Reverse    (c*value) ", dllcald(a,1) )
  '
' ==============================
  '
  s="abcdef1234567"
  print "Reversed " & s & " = "
  dllcall(Reverse,s)
  print s & "\n"
  dllcall(Finish)
  '
  end if
  dllfile
  line input q





.
Title: Re: DLLC
Post by: JRS on October 31, 2014, 11:41:27 PM
Too cool Charles, thanks!

Off to the playground.  :)

It works!


C:\OxygenBasic\projectsB\ScriptBasic\DLLC>scriba dllco2_EAsm.sb
Reversed abcdef1234567 = 7654321fedcba

C:\OxygenBasic\projectsB\ScriptBasic\DLLC>
Title: Re: DLLC
Post by: Charles Pegge on November 01, 2014, 12:06:03 AM
To simplify the example, there is an include file for Oxygen: DLLCO2.sb. It takes care of Oxygen setup.

DLLCO2.sb
Code: [Select]
  include "dllcinc.sb"

  oxy=dllfile("/scriptbasic/modules/oxygen.dll")

  o2_basic = dllproc( oxy, "o2_basic i =(c*source) " )
  o2_exec  = dllproc( oxy, "o2_exec  i =(i call)   " )
  o2_error = dllproc( oxy, "o2_error c*=()         " )
  o2_errno = dllproc( oxy, "o2_errno i =()         " )
  o2_len   = dllproc( oxy, "o2_len   i =()         " )
  o2_mode  = dllproc( oxy, "o2_mode     (i mode)   " )

  dllcall o2_mode,1

  function oxygen(src)
  dllcall o2_basic,src
  if (dllcall(o2_errno)<>0) then
    dllprnt dllcall(o2_error)
    a=0
    line input q
  else
    a=dllcall(o2_exec,0)
  end if
  oxygen=a
  end function

Example now simplified:

dllco22_EAsm.sb
Code: [Select]

  include "DLLCO2.sb"

' ==============================
  src="""
  extern

  function reverse(char*s)
  ========================
  addr ecx,s
  mov edx,0
 .rlen
  mov al,[ecx]
  cmp al,0
  jz xlen
  inc edx
  inc ecx
  jmp rlen
 .xlen
  ;
  addr ecx,s
  add  edx,ecx
  dec ecx
  ;
 .rswap
  inc ecx
  dec edx
  cmp edx,ecx
  jle xswap
  mov al,[ecx]
  mov ah,[edx]
  mov [ecx],ah
  mov [edx],al
  jmp rswap
 .xswap
  end function

  sub finish()
  terminate
  end sub

  function link(sys n) as sys
  select n
  case 0 : return @finish
  case 1 : return @reverse
  end select
  end function

  end extern
 

  addr link
  """
' ==============================

  '
  a=oxygen(src)
  '
  if (a<>0) then
  '
' ==============================
  '
  Finish  = dllproc(a,"Finish     ()        ", dllcald(a,0) )
  Reverse = dllproc(a,"Reverse    (c*value) ", dllcald(a,1) )
  '
' ==============================
  '
  s="abcdef1234567"
  print "Reversed " & s & " = "
  dllcall(Reverse,s)
  print s & "\n"
  dllcall(Finish)
  '
  end if
  dllfile
  line input q
Title: Re: DLLC
Post by: JRS on November 01, 2014, 12:34:59 AM
Sweet!

Who needs an Olly mess on your hands.  ;D
Title: Re: DLLC
Post by: JRS on November 01, 2014, 12:46:04 AM
Charles,

The new sweet version went sour. :-(

1. You need to REM all your =================== lines.

2.
C:\OxygenBasic\projectsB\ScriptBasic\DLLC>scriba dllco2_EAsm.sb
 ; ASM ERR:     rlen!!  Unidentified instruction: rlen
 ; AFTER:       .reverse#char
 ; LINE:        8


Title: Re: DLLC
Post by: Charles Pegge on November 01, 2014, 12:55:01 AM
John, could you ensure that you have a recent Oxygen.dll in ScriptBasic/modules

http://www.oxygenbasic.org/o2zips/Oxygen.zip
Title: Re: DLLC
Post by: JRS on November 01, 2014, 01:02:09 AM
Only thing I did was update the two programs you posted. It worked before.

Returning back to the (saved) original SB include and script works fine.

I updated the oxygen.dll (just downloaded the WIP ZIP a couple hours before) and no change. Original works, updated doesn't.
Title: Re: DLLC
Post by: JRS on November 01, 2014, 01:51:06 AM
Stop Debugging!

My FU!!!

I trimmed off the leading indent and I must of chopped something off. The missing REM should have been a clue.

Sorry!
Title: Re: DLLC
Post by: Charles Pegge on November 01, 2014, 02:25:50 AM
John, Did you trim off the .rlen dot ? :)
Title: Re: DLLC
Post by: Mike Lobanovsky on November 01, 2014, 08:20:18 AM
John, Did you trim off the .rlen dot ? :)

;D

I suggest we move all our ASM exercises from now on to the ScriptBASIC forum as a retribution for John's recent attempts to prosecute assembly on BASIC sites. He's been found guilty of illegal possession of this drug in his SB in massive quantities.
Title: Re: DLLC
Post by: JRS on November 01, 2014, 09:18:30 AM
Thanks again Charles for the refresh on just how powerful the DLLC extension module is. I have only tapped a small portion of it's ability.

1 + 1 = 3 (If you don't use a condom)   :o

Just noticed the t.txt hooker tease. Helpful!

.
Title: Re: DLLC
Post by: Aurel on November 01, 2014, 12:00:23 PM
Quote
Original works, updated doesn't.
and what is new ?....
I constantly have a feeling that oxygen is in wrong direction....
Title: Re: DLLC
Post by: JRS on November 01, 2014, 12:04:04 PM
Quote
I constantly have a feeling that oxygen is in wrong direction....

Thanks Aurel for your conformation Charles is on the right path. (white is black, good is evil, ...)

@Mike - prosecute assembly? Do you mean prostitute assembly?
Title: Re: DLLC
Post by: Mike Lobanovsky on November 01, 2014, 12:19:32 PM
@Mike - prosecute assembly? Do you mean prostitute assembly?

No John, I mean the former. I've just recalled some recent message of yours where you would wonder what the purpose of discussing assembly at a BASIC forum was. Noone realized at that time that SB itself was in a state of secret liaison with the subject... ;)

Quote
Just noticed the t.txt hooker tease.

I remember perfectly well that my message #666 was addressed to the late Mr.Vidlanovic. ;D
Title: Re: DLLC
Post by: JRS on November 01, 2014, 12:28:08 PM
Quote from: Mike
Noone realized at that time that SB itself was in a state of secret liaison with the subject...

Quote from: Mike
you would wonder what the purpose of discussing assembly at a BASIC forum was.

That topic got started when Rob and I were discussing how some people make simple tasks look difficult to presume self worth. Probably should of just let it go.

DLLC has been around for some time now as SB's O2 super hero. I just wasn't clear where Charles stood with in-line ASM. I don't have official approval from Charles but I coined the JIT function in SB as a Virtual DLL. To me that is the easiest way to explain its use. As icing, DLLC can run them in a thread.  8)
Title: Re: DLLC
Post by: JRS on November 01, 2014, 04:09:52 PM
A hello world for the SB O2 connection via DLLC.

(http://files.allbasic.info/ScriptBasic/O2/vdll_hello.png)

Code: Script BASIC
  1. ' SBO2 Hello World
  2.  
  3. INCLUDE "DLLCO2.sb"
  4.  
  5. oxygen "PRINT \"Hello World!\""
  6. dllfile
  7.  
Title: Re: DLLC
Post by: JRS on November 02, 2014, 11:27:46 AM
I was thinking that encapsulating O2 JIT functions in SB MODULE/END MODULE name spaces as a need for speed library could be cool.

Title: Re: DLLC
Post by: JRS on November 02, 2014, 07:28:15 PM
I started the new O2 extension module encapsulating Charles's string reverse function in assembly. (JIT function)

Code: Script BASIC
  1. IMPORT O2.inc
  2.  
  3. FOR x = 65 TO 90
  4.   alpha &= CHR(x)
  5. NEXT
  6. PRINT alpha,"\n"
  7. PRINT O2::RevStr(alpha),"\n"
  8.  
  9. O2::Done()
  10.  

Output

C:\scriptbasic\o2dev>scriba testrev.sb
ABCDEFGHIJKLMNOPQRSTUVWXYZ
ZYXWVUTSRQPONMLKJIHGFEDCBA

C:\scriptbasic\o2dev>


This is the O2.inc file. I still need to tidy it up but it's a foundation to add more OxygenBasic functions to the library.

Code: Script BASIC
  1. MODULE O2
  2.  
  3. include "dllcinc.sb"
  4.  
  5. oxy=dllfile("/scriptbasic/Debugger/modules/oxygen.dll")
  6.  
  7. o2_basic = dllproc( oxy, "o2_basic i =(c*source) " )
  8. o2_exec  = dllproc( oxy, "o2_exec  i =(i call)   " )
  9. o2_error = dllproc( oxy, "o2_error c*=()         " )
  10. o2_errno = dllproc( oxy, "o2_errno i =()         " )
  11. o2_len   = dllproc( oxy, "o2_len   i =()         " )
  12. o2_mode  = dllproc( oxy, "o2_mode     (i mode)   " )
  13.  
  14. dllcall o2_mode,1
  15.  
  16. function oxygen(src)
  17. dllcall o2_basic,src
  18. if (dllcall(o2_errno)<>0) then
  19.   dllprnt dllcall(o2_error)
  20.   a=0
  21.   line input q
  22. else
  23.   a=dllcall(o2_exec,0)
  24. end if
  25. oxygen=a
  26. end function
  27.  
  28. src = """
  29.  extern
  30.  
  31.  function reverse(char*s)
  32.  ========================
  33.  addr ecx,s
  34.  mov edx,0
  35. .rlen
  36.  mov al,[ecx]
  37.  cmp al,0
  38.  jz xlen
  39.  inc edx
  40.  inc ecx
  41.  jmp rlen
  42. .xlen
  43.  ;
  44.  addr ecx,s
  45.  add  edx,ecx
  46.  dec ecx
  47.  ;
  48. .rswap
  49.  inc ecx
  50.  dec edx
  51.  cmp edx,ecx
  52.  jle xswap
  53.  mov al,[ecx]
  54.  mov ah,[edx]
  55.  mov [ecx],ah
  56.  mov [edx],al
  57.  jmp rswap
  58. .xswap
  59.  end function
  60.  
  61.  sub finish()
  62.  terminate
  63.  end sub
  64.  
  65.  function link(sys n) as sys
  66.  select n
  67.  case 0 : return @finish
  68.  case 1 : return @reverse
  69.  end select
  70.  end function
  71.  
  72.  end extern
  73.  
  74.  
  75.  addr link
  76. """
  77.  
  78.   a = oxygen(src)
  79.   Finish  = dllproc(a,"Finish     ()        ", dllcald(a,0) )
  80.   Reverse = dllproc(a,"Reverse    (c*value) ", dllcald(a,1) )
  81.  
  82. FUNCTION RevStr(strarg)
  83.   dllcall(Reverse, strarg)
  84.   RevStr = strarg
  85. END FUNCTION    
  86.  
  87. FUNCTION Done
  88.   rtnval = dllcall(Finish)
  89.   dllfile
  90.   Done = rtnval
  91. END FUNCTION    
  92.  
  93. END MODULE
  94.  
Title: Re: DLLC
Post by: JRS on November 02, 2014, 11:12:11 PM
How the OxygenBasic project is doing.

.
Title: Re: DLLC
Post by: JRS on November 03, 2014, 01:06:06 PM
Charles,

I thought I would add your Word parser for Basic - (getword) (http://www.oxygenbasic.org/forum/index.php?topic=611.0) function for the SB O2.inc library. I'm getting an exception error calling the O2 getword function and don't know why.

Code: Script BASIC
  1. IMPORT O2.inc
  2.  
  3. FOR x = 65 TO 90
  4.   alpha &= CHR(x)
  5. NEXT
  6. PRINT alpha,"\n"
  7. PRINT O2::RevStr(alpha),"\n"
  8.  
  9. Next_Word:
  10.   wd = O2::GetWords("abc(d[xx]+7/6,`qwerty`)", 1)
  11. PRINT "Got Here\n"
  12.  
  13.   IF NOT wd=undef THEN
  14.     PRINT wd,"\n"
  15.     GOTO Next_Word
  16.   END IF
  17.  
  18. O2::Done()
  19.  

O2.inc
Code: Script BASIC
  1. MODULE O2
  2.  
  3. include "dllcinc.sb"
  4.  
  5. oxy=dllfile("/scriptbasic/Debugger/modules/oxygen.dll")
  6.  
  7. o2_basic = dllproc( oxy, "o2_basic i =(c*source) " )
  8. o2_exec  = dllproc( oxy, "o2_exec  i =(i call)   " )
  9. o2_error = dllproc( oxy, "o2_error c*=()         " )
  10. o2_errno = dllproc( oxy, "o2_errno i =()         " )
  11. o2_len   = dllproc( oxy, "o2_len   i =()         " )
  12. o2_mode  = dllproc( oxy, "o2_mode     (i mode)   " )
  13.  
  14. dllcall o2_mode,1
  15.  
  16. src = """
  17.  extern
  18.  
  19. function reverse(char*s)
  20. '=======================
  21.  addr ecx,s
  22.  mov edx,0
  23. .rlen
  24.  mov al,[ecx]
  25.  cmp al,0
  26.  jz xlen
  27.  inc edx
  28.  inc ecx
  29.  jmp rlen
  30. .xlen
  31.  ;
  32.  addr ecx,s
  33.  add  edx,ecx
  34.  dec ecx
  35.  ;
  36. .rswap
  37.  inc ecx
  38.  dec edx
  39.  cmp edx,ecx
  40.  jle xswap
  41.  mov al,[ecx]
  42.  mov ah,[edx]
  43.  mov [ecx],ah
  44.  mov [edx],al
  45.  jmp rswap
  46. .xswap
  47.  end function
  48.  
  49. function getword(char*ss,sys*b) as char*
  50. '=======================================
  51. if b=0 then b=1
  52. byte s at @ss
  53. byte c,d
  54. sys bb,bc
  55. static char z[128]
  56. a=0
  57. bb=b
  58.  
  59. 'SKIP LEADING SPACES
  60. do
  61.  c=s[b]
  62.  select c
  63.   case 33 to 255,0 : exit do 'SKIP SPACE
  64.  end select
  65.  b++
  66. end do
  67. bc=b
  68. '
  69. 'QUOTES
  70. select c
  71. case 34,39
  72.   do
  73.     b+=1
  74.     d=s[b]
  75.     if d=0 or d=c then b+=1 : jmp fwd done
  76.   end do
  77. end select
  78. 'WORDS AND SYMBOLS
  79. do
  80.  c=s[b]
  81.  select c
  82.  case 0 to 32    : exit do
  83.  case 35         : jmp fwd more
  84.  case 33 to 47   : 'symbols
  85.  case 48 to 57   : jmp fwd more 'numbers
  86.  case 58 to 64   : 'symbols
  87.  case 65 to 90   : jmp fwd more 'capitals
  88.  case 95         : jmp fwd more 'underscore
  89.  case 91 to 96   : 'symbols
  90.  case 97 to 122  : jmp fwd more 'lower case
  91.  case 123 to 127 : 'symbols
  92.  case 128 to 255 : jmp fwd more 'higher ascii
  93. end select
  94.  
  95. if b=bc then b++
  96.  exit do
  97.  
  98.  more:
  99.  b++
  100. end do
  101.  
  102. done:
  103.  
  104. if b > bb then
  105.  z=mid ss,bc,b-bc
  106. else
  107.  z = ""
  108. end if
  109. return z
  110.  
  111. end function
  112.  
  113. sub finish()
  114. '===========
  115.  terminate
  116. end sub
  117.  
  118. function link(sys n) as sys
  119. '==========================
  120.  select n
  121.  case 0 : return @finish
  122.  case 1 : return @reverse
  123.  case 2 : return @getword
  124.  end select
  125. end function
  126.  
  127. end extern
  128.  
  129. addr link
  130. """
  131.  
  132. a = oxygen(src)
  133. Finish  = dllproc(a,"Finish ()", dllcald(a,0))
  134. Reverse = dllproc(a,"Reverse (c*value)", dllcald(a,1))
  135. Words   = dllproc(a,"getword c* = (c* strraw, i* start)", dllcald(a,2))
  136.  
  137. FUNCTION oxygen(src)
  138.   dllcall o2_basic,src
  139.   IF (dllcall(o2_errno)<> 0) THEN
  140.     dllprnt dllcall(o2_error)
  141.     a = 0
  142.   ELSE
  143.     a = dllcall(o2_exec,0)
  144.   END IF
  145.   oxygen = a
  146. END FUNCTION
  147.  
  148. FUNCTION RevStr(strarg)
  149.   dllcall(Reverse, strarg)
  150.   RevStr = strarg
  151. END FUNCTION
  152.  
  153. FUNCTION GetWords(strarg, longarg)
  154.   GetWords = dllcall(Words, strarg, longarg)
  155. END FUNCTION
  156.  
  157. FUNCTION Done
  158.   rtnval = dllcall(Finish)
  159.   dllfile
  160.   Done = rtnval
  161. END FUNCTION
  162.  
  163. END MODULE
  164.  
Title: Re: DLLC
Post by: Charles Pegge on November 03, 2014, 01:25:12 PM
Hi John,

String is a bstring, not a char*. Best to create Getword returning static char* words.

Code: OxygenBasic
  1.   function getword(char*s,sys*b) as char*
  2.   '======================================
  3.  'b=1
  4.  sys a,c,d,bb,bc
  5.   static char z[128]
  6.   a=0
  7.   bb=b
  8.   do
  9.     c=asc s,b
  10.     if c=0 then exit do
  11.     if c>32 then exit do
  12.     b+=1
  13.   end do
  14.   bc=b
  15.   if c=34 or c=96 then 'quotes
  16.   do
  17.       b+=1
  18.       d=asc s,b
  19.       if d=0 or d=c then b+=1 : jmp fwd done
  20.    end do
  21.   end if
  22.   do
  23.     c=asc s,b
  24.     select c
  25.     case 0 to 32 : exit do
  26.     case 33 to 47
  27.       if c=35 then jmp fwd more '#
  28.     if b=bc then b+=1
  29.       exit do
  30.     case 48 to 57 : jmp fwd more 'numbers
  31.   case 58 to 64
  32.       if b=bc then b+=1
  33.       exit do
  34.     case 65 to 90 : jmp fwd more 'capitals
  35.   case 91 to 96
  36.       if c=95 then jmp fwd more 'underscore
  37.     if b=bc then b+=1
  38.       exit do    
  39.     case 97 to 122 : jmp fwd more 'lower case
  40.   case 123 to 127
  41.       if b=bc then b+=1
  42.       exit do
  43.     end select
  44.     '
  45.   'higher ascii chars treated as part of word
  46.   '
  47.   more:
  48.     '
  49.   b+=1
  50.   end do
  51.   '
  52.  done:
  53.   '
  54.  if b>bb then
  55.     z=mid s,bb,b-bb
  56.   else
  57.     z=""
  58.   end if
  59.   return z
  60.  
  61.   end function
  62.  
Title: Re: DLLC
Post by: JRS on November 03, 2014, 01:49:28 PM
Hi Charles and thanks for your help.

I replaced the getword function with your latest and it still exception errors.

Code: Script BASIC
  1. Words   = dllproc(a,"getword static c*=(c*strraw, i start)", dllcald(a,2) )
  2.  

I changed the DLLC function define and it doesn't exception error anymore but wd is returning undef. Should I define the second argument as a variable? Is this being updated with the start of the next find?
Title: Re: DLLC
Post by: Charles Pegge on November 03, 2014, 02:13:24 PM
Yes, should be i*

starting value 1,  then updated by getword.
Title: Re: DLLC
Post by: JRS on November 03, 2014, 02:31:31 PM
Got it working. Thanks for the help & code Charles! A very handy function to have in the extended SB toolbox.

Note: The O2.inc code includes Charles's improved getword() function.

Code: Script BASIC
  1. IMPORT O2.inc
  2.  
  3. FOR x = 65 TO 90
  4.   alpha &= CHR(x)
  5. NEXT
  6. PRINT alpha,"\n"
  7. PRINT O2::RevStr(alpha),"\n"
  8. p = 1
  9. Next_Word:
  10. wd = O2::GetWords("abc(d[xx]+7/6,`qwerty`)", p)
  11. IF wd <> "" THEN
  12.   PRINT wd,"\n"
  13.   GOTO Next_Word
  14. END IF
  15.  
  16. O2::Done()
  17.  

O2.inc
Code: Script BASIC
  1. MODULE O2
  2.  
  3. include "dllcinc.sb"
  4.  
  5. oxy=dllfile("/scriptbasic/Debugger/modules/oxygen.dll")
  6.  
  7. o2_basic = dllproc( oxy, "o2_basic i =(c*source) " )
  8. o2_exec  = dllproc( oxy, "o2_exec  i =(i call)   " )
  9. o2_error = dllproc( oxy, "o2_error c*=()         " )
  10. o2_errno = dllproc( oxy, "o2_errno i =()         " )
  11. o2_len   = dllproc( oxy, "o2_len   i =()         " )
  12. o2_mode  = dllproc( oxy, "o2_mode     (i mode)   " )
  13.  
  14. dllcall o2_mode,1
  15.  
  16. src = """
  17.  extern
  18.  
  19. function reverse(char*s)
  20. '=======================
  21.  addr ecx,s
  22.  mov edx,0
  23. .rlen
  24.  mov al,[ecx]
  25.  cmp al,0
  26.  jz xlen
  27.  inc edx
  28.  inc ecx
  29.  jmp rlen
  30. .xlen
  31.  ;
  32.  addr ecx,s
  33.  add  edx,ecx
  34.  dec ecx
  35.  ;
  36. .rswap
  37.  inc ecx
  38.  dec edx
  39.  cmp edx,ecx
  40.  jle xswap
  41.  mov al,[ecx]
  42.  mov ah,[edx]
  43.  mov [ecx],ah
  44.  mov [edx],al
  45.  jmp rswap
  46. .xswap
  47.  end function
  48.  
  49. function getword(char*ss,sys*b) as char*
  50. '=======================================
  51. if b=0 then b=1
  52. byte s at @ss
  53. byte c,d
  54. sys bb,bc
  55. static char z[128]
  56. a=0
  57. bb=b
  58.  
  59. 'SKIP LEADING SPACES
  60. do
  61.  c=s[b]
  62.  select c
  63.   case 33 to 255,0 : exit do 'SKIP SPACE
  64.  end select
  65.  b++
  66. end do
  67. bc=b
  68. '
  69. 'QUOTES
  70. select c
  71. case 34,39
  72.   do
  73.     b+=1
  74.     d=s[b]
  75.     if d=0 or d=c then b+=1 : jmp fwd done
  76.   end do
  77. end select
  78. 'WORDS AND SYMBOLS
  79. do
  80.  c=s[b]
  81.  select c
  82.  case 0 to 32    : exit do
  83.  case 35         : jmp fwd more
  84.  case 33 to 47   : 'symbols
  85.  case 48 to 57   : jmp fwd more 'numbers
  86.  case 58 to 64   : 'symbols
  87.  case 65 to 90   : jmp fwd more 'capitals
  88.  case 95         : jmp fwd more 'underscore
  89.  case 91 to 96   : 'symbols
  90.  case 97 to 122  : jmp fwd more 'lower case
  91.  case 123 to 127 : 'symbols
  92.  case 128 to 255 : jmp fwd more 'higher ascii
  93. end select
  94.  
  95. if b=bc then b++
  96.  exit do
  97.  
  98.  more:
  99.  b++
  100. end do
  101.  
  102. done:
  103.  
  104. if b > bb then
  105.  z=mid ss,bc,b-bc
  106. else
  107.  z = ""
  108. end if
  109. return z
  110.  
  111. end function
  112.  
  113. sub finish()
  114. '===========
  115.  terminate
  116. end sub
  117.  
  118. function link(sys n) as sys
  119. '==========================
  120.  select n
  121.  case 0 : return @finish
  122.  case 1 : return @reverse
  123.  case 2 : return @getword
  124.  end select
  125. end function
  126.  
  127. end extern
  128.  
  129. addr link
  130. """
  131.  
  132. a = oxygen(src)
  133. Finish  = dllproc(a,"Finish ()", dllcald(a,0))
  134. Reverse = dllproc(a,"Reverse (c*value)", dllcald(a,1))
  135. Words   = dllproc(a,"getword c* = (c* strraw, i* start)", dllcald(a,2))
  136.  
  137. FUNCTION oxygen(src)
  138.   dllcall o2_basic,src
  139.   IF (dllcall(o2_errno)<> 0) THEN
  140.     dllprnt dllcall(o2_error)
  141.     a = 0
  142.   ELSE
  143.     a = dllcall(o2_exec,0)
  144.   END IF
  145.   oxygen = a
  146. END FUNCTION
  147.  
  148. FUNCTION RevStr(strarg)
  149.   dllcall(Reverse, strarg)
  150.   RevStr = strarg
  151. END FUNCTION
  152.  
  153. FUNCTION GetWords(strarg, longarg)
  154.   GetWords = dllcall(Words, strarg, longarg)
  155. END FUNCTION
  156.  
  157. FUNCTION Done
  158.   rtnval = dllcall(Finish)
  159.   dllfile
  160.   Done = rtnval
  161. END FUNCTION
  162.  
  163. END MODULE
  164.  

Output

C:\scriptbasic\o2dev>scriba testrev.sb
ABCDEFGHIJKLMNOPQRSTUVWXYZ
ZYXWVUTSRQPONMLKJIHGFEDCBA
abc
(
d
[
xx
]
+
7
/
6
,
`qwerty`
)

C:\scriptbasic\o2dev>

Title: Re: DLLC
Post by: Charles Pegge on November 03, 2014, 09:47:22 PM
Very good, John.

I checked out c* return to ensure DLLC is casting this value correctly.

Improved version of GetWord:
Code: OxygenBasic
  1.   function getword(char*ss,sys*b) as char*
  2.   '=======================================
  3.  if b=0 then b=1
  4.   byte s at @ss
  5.   byte c,d
  6.   sys bb,bc
  7.   static char z[128]
  8.   a=0
  9.   bb=b
  10.   '
  11.  'SKIP LEADING SPACES
  12.  do
  13.     c=s[b]
  14.     select c
  15.     case 33 to 255,0 : exit do 'SKIP SPACE
  16.    end select
  17.     b++
  18.   end do
  19.   bc=b
  20.   '
  21.  'QUOTES
  22.  select c
  23.   case 34,39
  24.    do
  25.       b+=1
  26.       d=s[b]
  27.       if d=0 or d=c then b+=1 : jmp fwd done
  28.    end do
  29.   end select
  30.   'WORDS AND SYMBOLS
  31.  do
  32.     c=s[b]
  33.     select c
  34.     case 0 to 32    : exit do
  35.     case 35         : jmp fwd more
  36.     case 33 to 47   : 'symbols
  37.    case 48 to 57   : jmp fwd more 'numbers
  38.    case 58 to 64   : 'symbols
  39.    case 65 to 90   : jmp fwd more 'capitals
  40.    case 95         : jmp fwd more 'underscore
  41.    case 91 to 96   : 'symbols
  42.    case 97 to 122  : jmp fwd more 'lower case
  43.    case 123 to 127 : 'symbols
  44.    case 128 to 255 : jmp fwd more 'higher ascii
  45.    end select
  46.     '
  47.    '
  48.    if b=bc then b++
  49.     exit do
  50.     '
  51.    more:
  52.     b++
  53.   end do
  54.   '
  55.  done:
  56.   '
  57.  if b>bb then
  58.     z=mid ss,bc,b-bc
  59.   else
  60.     z=""
  61.   end if
  62.   return z
  63.  
  64.   end function
  65.  
Title: Re: DLLC
Post by: JRS on November 03, 2014, 10:09:44 PM
Thanks Charles!

What are the improvements and can you post an example showing the improvements?

FYI The new getword() function works fine with the old example string.
Title: Re: DLLC
Post by: Charles Pegge on November 03, 2014, 10:32:38 PM
Improvements: :)

Strip leading spaces,
tolerant of 0 index,
faster parsing with bytes,
cleaner layout of cases,
elimination of duplicated code.
Title: Re: DLLC
Post by: JRS on November 03, 2014, 10:35:29 PM
Sweet!

On to my next must have O2 function.  :D
Title: Re: DLLC
Post by: JRS on November 03, 2014, 11:56:10 PM
I think I found my next O2 function to steal.  ;)

Aligned Text (http://www.oxygenbasic.org/forum/index.php?topic=473.msg3685#msg3685)

For this function I would like to show the bstr feature of DLLC so I can use additional O2 functions out of the box.

One of my goals with the O2 interface is fast array functions. Dave did a nice job of flushing out array handling in the SB IDE/Debugger. Building dynamic matrix structures quickly would lead to a more robust SDL interface in SB.

Title: Re: DLLC
Post by: JRS on November 04, 2014, 02:28:47 AM
Charles,

I'm getting a missing END IF with the aligned text example in O2.

If you have time to get this working in SB with bstr support, I should be able to carry on from there.

It would also be nice if the user could select the delimiter to use. $ looks like a bunch of string variable references.

.
Title: Re: DLLC
Post by: Charles Pegge on November 04, 2014, 05:34:33 AM
Caught a nasty little bug there!

Fix:
http://www.oxygenbasic.org/o2zips/Oxygen.zip

I have also updated the example to support user definable delimiters of any length (default comma)

http://www.oxygenbasic.org/forum/index.php?topic=473.msg3685#msg3685
Title: Re: DLLC
Post by: Aurel on November 04, 2014, 08:28:56 AM

Quote
Improvements:

Strip leading spaces,
tolerant of 0 index,
faster parsing with bytes,
cleaner layout of cases,
elimination of duplicated code.

Charles
Where is example for this  improvements.
i wish to compare with old code
Title: Re: DLLC
Post by: Charles Pegge on November 04, 2014, 08:44:07 AM
Hi Aurel,

It's back on page 2 of this thread.
Title: Re: DLLC
Post by: JRS on November 04, 2014, 10:41:39 AM
Quote
Caught a nasty little bug there!

It left me scratching my head. Luckily it was 3 AM, I was tired and it never got to the bleeding stage.  ;)

I'm thinking of making the O2 source for the DLLC interface a separate file. This way the O2 code can be run as a SB ext. or a O2 compiled DLL. It can work sort of like SB's program cache feature. If a tokenized version of the script doesn't exist, it generates it from source.

Will you have time to do the aligned text function for SB using the bstr feature of DLLC?



 
Title: Re: DLLC
Post by: JRS on November 04, 2014, 02:08:09 PM
OT - Do I know how to pick them or what?

40 Hottest Female Celebrity Bodies of All Time (http://www.rantlifestyle.com/2013/12/12/30-hottest-female-celebrity-bodies-time/?utm_medium=Referral&utm_source=Adblade#slide_81)

.
Title: Re: DLLC
Post by: Charles Pegge on November 04, 2014, 02:34:52 PM
Hi John,

I think char* will provide the best interface for SB

Proposing a function to make this interface:

Code: OxygenBasic
  1. function AlignText(char *in, *ju,*dl,*cr) as char*
  2. ==================================================
  3. static AlignedText tt
  4. tt.justify         ju
  5. tt.delimiter       dl
  6. tt.endofline       cr
  7. tt.SetText         in
  8. return             tt.layout
  9. end function
  10.  
Title: Re: DLLC
Post by: JRS on November 04, 2014, 02:43:42 PM
Quote
I think char* will provide the best interface for SB

Makes sense. Dave's COM ext. module takes care of the bstr & variant side of Windows as a seamless interface. If we need DLLC bstr, stuctures, GUIDs, ..., it's in there.  8)

Yes! I like the proposed Aligned Text direction.

Title: Re: DLLC
Post by: JRS on November 04, 2014, 04:40:21 PM
Here is the new O2.inc with the O2 functions in a standard O2 source file.

Code: Script BASIC
  1. MODULE O2
  2.  
  3. include "dllcinc.sb"
  4.  
  5. oxy=dllfile("/scriptbasic/Debugger/modules/oxygen.dll")
  6.  
  7. o2_basic = dllproc( oxy, "o2_basic i =(c*source) " )
  8. o2_exec  = dllproc( oxy, "o2_exec  i =(i call)   " )
  9. o2_error = dllproc( oxy, "o2_error c*=()         " )
  10. o2_errno = dllproc( oxy, "o2_errno i =()         " )
  11. o2_len   = dllproc( oxy, "o2_len   i =()         " )
  12. o2_mode  = dllproc( oxy, "o2_mode     (i mode)   " )
  13.  
  14. dllcall o2_mode,1
  15.  
  16. OPEN "/scriptbasic/Debugger/include/O2.src" FOR INPUT AS #1
  17. src = INPUT(LOF(1), 1)
  18. CLOSE(1)
  19.  
  20. a = oxygen(src)
  21. Finish  = dllproc(a,"Finish ()", dllcald(a,0))
  22. Reverse = dllproc(a,"Reverse (c*value)", dllcald(a,1))
  23. Words   = dllproc(a,"getword c* = (c* strraw, i* start)", dllcald(a,2))
  24.  
  25. FUNCTION oxygen(src)
  26.   dllcall o2_basic,src
  27.   IF (dllcall(o2_errno)<> 0) THEN
  28.     dllprnt dllcall(o2_error)
  29.     a = 0
  30.   ELSE
  31.     a = dllcall(o2_exec,0)
  32.   END IF
  33.   oxygen = a
  34. END FUNCTION
  35.  
  36. FUNCTION RevStr(strarg)
  37.   dllcall(Reverse, strarg)
  38.   RevStr = strarg
  39. END FUNCTION
  40.  
  41. FUNCTION GetWords(strarg, longarg)
  42.   GetWords = dllcall(Words, strarg, longarg)
  43. END FUNCTION
  44.  
  45. FUNCTION Done
  46.   rtnval = dllcall(Finish)
  47.   dllfile
  48.   Done = rtnval
  49. END FUNCTION
  50.  
  51. END MODULE
  52.  

O2.src
Code: OxygenBasic
  1. ' O2 source
  2.  
  3. extern
  4.  
  5. function reverse(char*s)
  6. '=======================
  7.  addr ecx,s
  8.   mov edx,0
  9.  .rlen
  10.   mov al,[ecx]
  11.   cmp al,0
  12.   jz xlen
  13.   inc edx
  14.   inc ecx
  15.   jmp rlen
  16.  .xlen
  17.   ;
  18.   addr ecx,s
  19.   add  edx,ecx
  20.   dec ecx
  21.   ;
  22.  .rswap
  23.   inc ecx
  24.   dec edx
  25.   cmp edx,ecx
  26.   jle xswap
  27.   mov al,[ecx]
  28.   mov ah,[edx]
  29.   mov [ecx],ah
  30.   mov [edx],al
  31.   jmp rswap
  32.  .xswap
  33.   end function
  34.  
  35. function getword(char*ss,sys*b) as char*
  36. '=======================================
  37. if b=0 then b=1
  38. byte s at @ss
  39. byte c,d
  40. sys bb,bc
  41. static char z[128]
  42. a=0
  43. bb=b
  44.  
  45. 'SKIP LEADING SPACES
  46. do
  47.   c=s[b]
  48.   select c
  49.    case 33 to 255,0 : exit do 'SKIP SPACE
  50.  end select
  51.   b++
  52. end do
  53. bc=b
  54.  '
  55. 'QUOTES
  56. select c
  57.  case 34,39
  58.    do
  59.      b+=1
  60.      d=s[b]
  61.      if d=0 or d=c then b+=1 : jmp fwd done
  62.    end do
  63. end select
  64. 'WORDS AND SYMBOLS
  65. do
  66.   c=s[b]
  67.   select c
  68.   case 0 to 32    : exit do
  69.   case 35         : jmp fwd more
  70.   case 33 to 47   : 'symbols
  71.  case 48 to 57   : jmp fwd more 'numbers
  72.  case 58 to 64   : 'symbols
  73.  case 65 to 90   : jmp fwd more 'capitals
  74.  case 95         : jmp fwd more 'underscore
  75.  case 91 to 96   : 'symbols
  76.  case 97 to 122  : jmp fwd more 'lower case
  77.  case 123 to 127 : 'symbols
  78.  case 128 to 255 : jmp fwd more 'higher ascii
  79. end select
  80.  
  81. if b=bc then b++
  82.   exit do
  83.  
  84.   more:
  85.   b++
  86. end do
  87.  
  88. done:
  89.  
  90. if b > bb then
  91.   z=mid ss,bc,b-bc
  92. else
  93.   z = ""
  94. end if
  95. return z
  96.  
  97. end function
  98.  
  99. sub finish()
  100. '===========
  101.  terminate
  102. end sub
  103.  
  104. function link(sys n) as sys
  105. '==========================
  106.  select n
  107.   case 0 : return @finish
  108.   case 1 : return @reverse
  109.   case 2 : return @getword
  110.   end select
  111. end function
  112.  
  113. end extern
  114.  
  115. addr link
  116.  
Title: Re: DLLC
Post by: JRS on November 04, 2014, 05:35:18 PM
O2.inc could be a front end to dynamically configure an O2 virtual DLL that is generated under program control.

Code: Script BASIC
  1. OPEN "/scriptbasic/Debugger/include/O2.src" FOR INPUT AS #1
  2. src = INPUT(LOF(1), 1)
  3. CLOSE(1)
  4.  
  5. Use_Reverse = TRUE
  6.  
  7. IF Use_Reverse THEN
  8.   IF src LIKE "*function reverse*end function*" THEN
  9.     func = "\nfunction reverse" & JOKER(2) & "end function\n"
  10.     PRINT func
  11.   END IF
  12. END IF
  13.  

Output

C:\scriptbasic\o2dev>scriba getfunc.sb

function reverse(char*s)
'=======================
  addr ecx,s
  mov edx,0
 .rlen
  mov al,[ecx]
  cmp al,0
  jz xlen
  inc edx
  inc ecx
  jmp rlen
 .xlen
  ;
  addr ecx,s
  add  edx,ecx
  dec ecx
  ;
 .rswap
  inc ecx
  dec edx
  cmp edx,ecx
  jle xswap
  mov al,[ecx]
  mov ah,[edx]
  mov [ecx],ah
  mov [edx],al
  jmp rswap
 .xswap
  end function

C:\scriptbasic\o2dev>


Samurai Training 1860

.
Title: Re: DLLC
Post by: Charles Pegge on November 05, 2014, 12:47:01 PM

AlignedText withe wrapper function for SB use
Code: OxygenBasic
  1.  
  2. =================
  3. Class AlignedText
  4. =================
  5.  
  6. indexbase 1
  7.  
  8. string  buf, bufo, pr, cr, tab, jus, dlm
  9. sys     Cols, Rows, ColWidth[0x100], TotWidth, ColPad, ld
  10.  
  11. method SetText(char*s)
  12. ======================
  13. if not len cr then cr=chr(13,10)
  14. tab=chr(9)
  15. if not len jus then jus=string 200,"L"
  16. buf=s
  17. measure
  18. end method
  19.  
  20.  
  21. method measure()
  22. ================
  23. sys a, b, wa, wb, cm, c, cw
  24. a=1 : b=1
  25. Cols=0 : Rows=0 : ColPad=3
  26. ld=len dlm
  27. if not ld then dlm="," : ld=1 'default to comma
  28. do
  29.   wb=b
  30.   a=instr b,buf,cr
  31.   if a=0 then exit do
  32.   cm=0
  33.   c++
  34.   do
  35.     wa=instr wb,buf,dlm
  36.     if wa=0 or wa>a then exit do
  37.     cm++
  38.     if cm>cols then cols=cm
  39.     cw=wa-wb
  40.     if cw > ColWidth[cm] then ColWidth[cm]=cw
  41.     wb=wa+ld
  42.   end do
  43.   b=a+len cr
  44. end do
  45. rows=c
  46. '
  47. c=0
  48. for i=1 to cols
  49.   ColWidth[ i ]+=ColPad
  50.   c+=ColWidth[ i ]
  51. next
  52. TotWidth=c+len cr
  53. 'print ShowMetrics
  54. end method
  55.  
  56.  
  57. method ShowMetrics() as char*
  58. =============================
  59. pr="METRICS:" cr cr
  60. pr+=rows tab cols tab totwidth cr cr
  61. pr+="column" tab "spacing" cr
  62. for i=1 to cols
  63.   pr+=i tab ColWidth[ i ] cr
  64. next
  65. return pr
  66. end method
  67.  
  68.  
  69. method justify(char*j)
  70. ======================
  71. jus=j
  72. end method
  73.  
  74. method delimiter(char*j)
  75. ========================
  76. dlm=j
  77. end method
  78.  
  79. method endofline(char*j)
  80. ========================
  81. cr=j
  82. end method
  83.  
  84.  
  85. method layout() as char*
  86. ========================
  87. sys a, b, wa, wb, wl, cm, lpos, cpos
  88. bufo=space Rows*TotWidth
  89. a=1 : b=1
  90. do
  91.   wb=b
  92.   a=instr(b,buf,cr)
  93.   if a=0 then exit do
  94.   cm=0
  95.   cpos=1
  96.   do
  97.     wa=instr(wb,buf,dlm)
  98.     if wa=0 or wa>a then exit do
  99.     '
  100.    cm++
  101.     '
  102.    'JUSTIFICATION
  103.    '
  104.    wl=wa-wb
  105.     p=lpos+cpos 'default "L" LEFT ALIGN
  106.    '
  107.    select case asc(jus,cm)
  108.       case "R" : p=lpos+cpos+ColWidth[cm]-wl-Colpad
  109.       case "C" : p=lpos+cpos+( ColWidth[cm]-wl-Colpad )*.5
  110.     end select
  111.     '
  112.    mid bufo,p, mid buf,wb,wl
  113.     cpos+=colwidth[cm]
  114.     wb=wa+ld
  115.   end do
  116.   b=a+len cr
  117.   lpos+=TotWidth
  118.   if lpos<len(bufo) then mid bufo,lpos-1,cr
  119. end do
  120. return bufo
  121. end method
  122.  
  123. end class
  124.  
  125. '#recordof AlignedText
  126.  
  127. AlignedText atxt
  128.  
  129. function AlignText(char *in,*ju,*dl,*cr) as char*
  130. =================================================
  131. atxt.justify         ju
  132. atxt.delimiter       dl
  133. atxt.endofline       cr
  134. atxt.SetText         in
  135. return               atxt.layout
  136. end function
  137.  
  138. '====
  139. 'TEST
  140. '====
  141.  
  142. putfile "t.txt", AlignText (
  143. quote """
  144. Given;;a;;text;;file;;of;;many;;lines,;;where;;fields;;within;;a;;line;;
  145. are;;delineated;;by;;a;;single;;'dollar';;character,;;write;;a;;program
  146. that;;aligns;;each;;column;;of;;fields;;by;;ensuring;;that;;words;;in;;each;;
  147. column;;are;;separated;;by;;at;;least;;one;;space.
  148. Further,;;allow;;for;;each;;word;;in;;a;;column;;to;;be;;either;;left;;
  149. justified,;;right;;justified,;;or;;center;;justified;;within;;its;;column.
  150. """ ,
  151. "LLLLCCCRRRRR",
  152. ";;",
  153. chr(13,10)
  154. )
  155.  
  156.  
Title: Re: DLLC
Post by: JRS on November 05, 2014, 01:36:33 PM
I'm getting an exception error with the following code.

testo2.sb
Code: Script BASIC
  1. IMPORT O2.inc
  2.  
  3. FOR x = 65 TO 90
  4.   alpha &= CHR(x)
  5. NEXT
  6. PRINT alpha,"\n"
  7. PRINT O2::RevStr(alpha),"\n"
  8. p = 1
  9. Next_Word:
  10.   wd = O2::GetWords("abc(d[xx]+7/6,\"qwerty\")", p)
  11.   IF wd <> "" THEN
  12.     PRINT wd,"\n"
  13.     GOTO Next_Word
  14.   END IF
  15. quo = """
  16. Given;;a;;text;;file;;of;;many;;lines,;;where;;fields;;within;;a;;line;;
  17. are;;delineated;;by;;a;;single;;'dollar';;character,;;write;;a;;program
  18. that;;aligns;;each;;column;;of;;fields;;by;;ensuring;;that;;words;;in;;each;;
  19. column;;are;;separated;;by;;at;;least;;one;;space.
  20. Further,;;allow;;for;;each;;word;;in;;a;;column;;to;;be;;either;;left;;
  21. justified,;;right;;justified,;;or;;center;;justified;;within;;its;;column.
  22. """
  23. PRINT O2::ColumnAlign(quo, "LLLLCCCRRRRR", ";;", "\n")
  24.  
  25. O2::Done()
  26.  

O2.inc
Code: Script BASIC
  1. MODULE O2
  2.  
  3. include "dllcinc.sb"
  4.  
  5. oxy=dllfile("/scriptbasic/Debugger/modules/oxygen.dll")
  6.  
  7. o2_basic = dllproc( oxy, "o2_basic i =(c*source) " )
  8. o2_exec  = dllproc( oxy, "o2_exec  i =(i call)   " )
  9. o2_error = dllproc( oxy, "o2_error c*=()         " )
  10. o2_errno = dllproc( oxy, "o2_errno i =()         " )
  11. o2_len   = dllproc( oxy, "o2_len   i =()         " )
  12. o2_mode  = dllproc( oxy, "o2_mode     (i mode)   " )
  13.  
  14. dllcall o2_mode,1
  15.  
  16. OPEN "/scriptbasic/Debugger/include/O2.src" FOR INPUT AS #1
  17. src = INPUT(LOF(1), 1)
  18. CLOSE(1)
  19.  
  20. a = oxygen(src)
  21. Finish  = dllproc(a,"Finish ()", dllcald(a,0))
  22. Reverse = dllproc(a,"Reverse (c*value)", dllcald(a,1))
  23. Words   = dllproc(a,"getword c* = (c* strraw, i* start)", dllcald(a,2))
  24. Align   = dllproc(a,"AlignText c* = (c* in, c* ju, C* dl, c* cr)", dllcald(a,3))
  25.  
  26. FUNCTION oxygen(src)
  27.   dllcall o2_basic,src
  28.   IF (dllcall(o2_errno)<> 0) THEN
  29.     dllprnt dllcall(o2_error)
  30.     a = 0
  31.   ELSE
  32.     a = dllcall(o2_exec,0)
  33.   END IF
  34.   oxygen = a
  35. END FUNCTION
  36.  
  37. FUNCTION Done
  38.   rtnval = dllcall(Finish)
  39.   dllfile
  40.   Done = rtnval
  41. END FUNCTION
  42.  
  43. ' SB wrapper functions
  44.  
  45. FUNCTION RevStr(strarg)
  46.   dllcall(Reverse, strarg)
  47.   RevStr = strarg
  48. END FUNCTION
  49.  
  50. FUNCTION GetWords(strarg, longarg)
  51.   GetWords = dllcall(Words, strarg, longarg)
  52. END FUNCTION
  53.  
  54. FUNCTION ColumnAlign(in_str, just_str, dlm_str, eol_str)
  55.   ColumnAlign = dllcall(Align, in_str, just_str, dlm_str, eol_str)
  56. END FUNCTION
  57.  
  58. END MODULE
  59.  

O2.src
Code: OxygenBasic
  1. ' O2 source
  2.  
  3. extern
  4.  
  5. function reverse(char*s)
  6. '=======================
  7.  addr ecx,s
  8.   mov edx,0
  9.  .rlen
  10.   mov al,[ecx]
  11.   cmp al,0
  12.   jz xlen
  13.   inc edx
  14.   inc ecx
  15.   jmp rlen
  16.  .xlen
  17.   ;
  18.   addr ecx,s
  19.   add  edx,ecx
  20.   dec ecx
  21.   ;
  22.  .rswap
  23.   inc ecx
  24.   dec edx
  25.   cmp edx,ecx
  26.   jle xswap
  27.   mov al,[ecx]
  28.   mov ah,[edx]
  29.   mov [ecx],ah
  30.   mov [edx],al
  31.   jmp rswap
  32.  .xswap
  33.   end function
  34.  
  35. function getword(char*ss,sys*b) as char*
  36. '=======================================
  37. if b=0 then b=1
  38. byte s at @ss
  39. byte c,d
  40. sys bb,bc
  41. static char z[128]
  42. a=0
  43. bb=b
  44.  
  45. 'SKIP LEADING SPACES
  46. do
  47.   c=s[b]
  48.   select c
  49.    case 33 to 255,0 : exit do 'SKIP SPACE
  50.  end select
  51.   b++
  52. end do
  53. bc=b
  54.  '
  55. 'QUOTES
  56. select c
  57.  case 34,39
  58.    do
  59.      b+=1
  60.      d=s[b]
  61.      if d=0 or d=c then b+=1 : jmp fwd done
  62.    end do
  63. end select
  64. 'WORDS AND SYMBOLS
  65. do
  66.   c=s[b]
  67.   select c
  68.   case 0 to 32    : exit do
  69.   case 35         : jmp fwd more
  70.   case 33 to 47   : 'symbols
  71.  case 48 to 57   : jmp fwd more 'numbers
  72.  case 58 to 64   : 'symbols
  73.  case 65 to 90   : jmp fwd more 'capitals
  74.  case 95         : jmp fwd more 'underscore
  75.  case 91 to 96   : 'symbols
  76.  case 97 to 122  : jmp fwd more 'lower case
  77.  case 123 to 127 : 'symbols
  78.  case 128 to 255 : jmp fwd more 'higher ascii
  79. end select
  80.  
  81. if b=bc then b++
  82.   exit do
  83.  
  84.   more:
  85.   b++
  86. end do
  87.  
  88. done:
  89.  
  90. if b > bb then
  91.   z=mid ss,bc,b-bc
  92. else
  93.   z = ""
  94. end if
  95. return z
  96.  
  97. end function
  98.  
  99. =================
  100. Class AlignedText
  101. =================
  102.  
  103. indexbase 1
  104.  
  105. string  buf, bufo, pr, cr, tab, jus, dlm
  106. sys     Cols, Rows, ColWidth[0x100], TotWidth, ColPad, ld
  107.  
  108. method SetText(char*s)
  109. ======================
  110. if not len cr then cr=chr(13,10)
  111. tab=chr(9)
  112. if not len jus then jus=string 200,"L"
  113. buf=s
  114. measure
  115. end method
  116.  
  117.  
  118. method measure()
  119. ================
  120. sys a, b, wa, wb, cm, c, cw
  121. a=1 : b=1
  122. Cols=0 : Rows=0 : ColPad=3
  123. ld=len dlm
  124. if not ld then dlm="," : ld=1 'default to comma
  125. do
  126.   wb=b
  127.   a=instr b,buf,cr
  128.   if a=0 then exit do
  129.   cm=0
  130.   c++
  131.   do
  132.     wa=instr wb,buf,dlm
  133.     if wa=0 or wa>a then exit do
  134.     cm++
  135.     if cm>cols then cols=cm
  136.     cw=wa-wb
  137.     if cw > ColWidth[cm] then ColWidth[cm]=cw
  138.     wb=wa+ld
  139.   end do
  140.   b=a+len cr
  141. end do
  142. rows=c
  143. '
  144. c=0
  145. for i=1 to cols
  146.   ColWidth[ i ]+=ColPad
  147.   c+=ColWidth[ i ]
  148. next
  149. TotWidth=c+len cr
  150. 'print ShowMetrics
  151. end method
  152.  
  153.  
  154. method ShowMetrics() as char*
  155. =============================
  156. pr="METRICS:" cr cr
  157. pr+=rows tab cols tab totwidth cr cr
  158. pr+="column" tab "spacing" cr
  159. for i=1 to cols
  160.   pr+=i tab ColWidth[ i ] cr
  161. next
  162. return pr
  163. end method
  164.  
  165.  
  166. method justify(char*j)
  167. ======================
  168. jus=j
  169. end method
  170.  
  171. method delimiter(char*j)
  172. ========================
  173. dlm=j
  174. end method
  175.  
  176. method endofline(char*j)
  177. ========================
  178. cr=j
  179. end method
  180.  
  181.  
  182. method layout() as char*
  183. ========================
  184. sys a, b, wa, wb, wl, cm, lpos, cpos
  185. bufo=space Rows*TotWidth
  186. a=1 : b=1
  187. do
  188.   wb=b
  189.   a=instr(b,buf,cr)
  190.   if a=0 then exit do
  191.   cm=0
  192.   cpos=1
  193.   do
  194.     wa=instr(wb,buf,dlm)
  195.     if wa=0 or wa>a then exit do
  196.     '
  197.    cm++
  198.     '
  199.    'JUSTIFICATION
  200.    '
  201.    wl=wa-wb
  202.     p=lpos+cpos 'default "L" LEFT ALIGN
  203.    '
  204.    select case asc(jus,cm)
  205.       case "R" : p=lpos+cpos+ColWidth[cm]-wl-Colpad
  206.       case "C" : p=lpos+cpos+( ColWidth[cm]-wl-Colpad )*.5
  207.     end select
  208.     '
  209.    mid bufo,p, mid buf,wb,wl
  210.     cpos+=colwidth[cm]
  211.     wb=wa+ld
  212.   end do
  213.   b=a+len cr
  214.   lpos+=TotWidth
  215.   if lpos<len(bufo) then mid bufo,lpos-1,cr
  216. end do
  217. return bufo
  218. end method
  219.  
  220. end class
  221.  
  222. '#recordof AlignedText
  223.  
  224. AlignedText atxt
  225.  
  226. function AlignText(char *in,*ju,*dl,*cr) as char*
  227. =================================================
  228. atxt.justify         ju
  229. atxt.delimiter       dl
  230. atxt.endofline       cr
  231. atxt.SetText         in
  232. return               atxt.layout
  233. end function
  234.  
  235. sub finish()
  236. '===========
  237.  terminate
  238. end sub
  239.  
  240. function link(sys n) as sys
  241. '==========================
  242.  select n
  243.   case 0 : return @finish
  244.   case 1 : return @reverse
  245.   case 2 : return @getword
  246.   case 3 : return @AlignText
  247.   end select
  248. end function
  249.  
  250. end extern
  251.  
  252. addr link
  253.  
Title: Re: DLLC
Post by: Charles Pegge on November 05, 2014, 01:51:52 PM

Yes there is a glitch somewhere. It is environmental and needs deep investigation...
Title: Re: DLLC
Post by: JRS on November 05, 2014, 02:24:26 PM
Wild guess - It there a chance that the SB version is starting with a CR/NL and it's giving your function fits?

Title: Re: DLLC
Post by: Charles Pegge on November 05, 2014, 02:28:47 PM
Got mine working.

It seems to be null terminators.

Code: [Select]
s="""
Given;;a;;text;;file;;of;;many;;lines,;;where;;fields;;within;;a;;line;;
are;;delineated;;by;;a;;single;;'dollar';;character,;;write;;a;;program
that;;aligns;;each;;column;;of;;fields;;by;;ensuring;;that;;words;;in;;each;;
column;;are;;separated;;by;;at;;least;;one;;space.
Further,;;allow;;for;;each;;word;;in;;a;;column;;to;;be;;either;;left;;
justified,;;right;;justified,;;or;;center;;justified;;within;;its;;column.
""" & chr(0)

  lf=chr(10) & chr(0)
  dl=";;" & chr(0)
  fm="LLLLCCCRRRRR" & chr(0)
  t=dllcall(AlignText,s,fm,dl,lf )
  print t & "\n"

PS: only lf requires a null terminator
Title: Re: DLLC
Post by: JRS on November 05, 2014, 02:32:18 PM
SB by default includes a NULL string terminator when passing literal strings.

Code: Script BASIC
  1. PRINT O2::ColumnAlign(quo, "LLLLCCCRRRRR", ";;", "\n" & CHR(0))
  2.  

Still gives an exception error.
Title: Re: DLLC
Post by: Charles Pegge on November 05, 2014, 02:47:56 PM

both of these work:

Code: Script BASIC
  1.   lf="\n" & chr(0)
  2.   dl=";;"
  3.   fm="LLLLCCCRRRRR"
  4.   ' t=dllcall(AlignText,s,fm,dl,lf )
  5.  t=dllcall(AlignText,s,"LLLLCCCRRRRR",";;","\n" & chr(0) )
  6.   print t & "\n"
  7.  
Title: Re: DLLC
Post by: JRS on November 05, 2014, 02:54:26 PM
Code: Script BASIC
  1. PRINT O2::ColumnAlign(quo, "LLLLCCCRRRRR", ";;", "\n" & CHR(0) & CHR(0))
  2.  

Still gives an exception error.

If you don't want to flood this thread with a debugging session, I'm on the AllBASIC IRC.

I put your code in O2.inc and still getting the exception error. I'm thinking it has something to do with the O2 source reacting differently using the virtual DLL method rather than a standard O2 compile.

Have you tried the SB method on your end?

Title: Re: DLLC
Post by: Charles Pegge on November 05, 2014, 04:11:06 PM
John, I have your code working. I had to include the module inside testo2.sb. and change some filepaths to match my system.

Since I am launching by icon: line input u at the end

Code: Script BASIC
  1. MODULE O2
  2.  
  3. include "dllcinc.sb"
  4.  
  5. oxy=dllfile("/scriptbasic/modules/oxygen.dll")
  6.  
  7. o2_basic = dllproc( oxy, "o2_basic i =(c*source) " )
  8. o2_exec  = dllproc( oxy, "o2_exec  i =(i call)   " )
  9. o2_error = dllproc( oxy, "o2_error c*=()         " )
  10. o2_errno = dllproc( oxy, "o2_errno i =()         " )
  11. o2_len   = dllproc( oxy, "o2_len   i =()         " )
  12. o2_mode  = dllproc( oxy, "o2_mode     (i mode)   " )
  13.  
  14. dllcall o2_mode,1
  15.  
  16. OPEN "O2.src" FOR INPUT AS #1
  17. src = INPUT(LOF(1), 1)
  18. CLOSE(1)
  19.  
  20. a = oxygen(src)
  21. Finish  = dllproc(a,"Finish ()", dllcald(a,0))
  22. Reverse = dllproc(a,"Reverse (c*value)", dllcald(a,1))
  23. Words   = dllproc(a,"getword c* = (c* strraw, i* start)", dllcald(a,2))
  24. Align   = dllproc(a,"AlignText c* = (c* in, c* ju, C* dl, c* cr)", dllcald(a,3))
  25.  
  26. FUNCTION oxygen(src)
  27.   dllcall o2_basic,src
  28.   IF (dllcall(o2_errno)<> 0) THEN
  29.     dllprnt dllcall(o2_error)
  30.     a = 0
  31.   ELSE
  32.     a = dllcall(o2_exec,0)
  33.   END IF
  34.   oxygen = a
  35. END FUNCTION
  36.  
  37. FUNCTION Done
  38.   rtnval = dllcall(Finish)
  39.   dllfile
  40.   Done = rtnval
  41. END FUNCTION
  42.  
  43. ' SB wrapper functions
  44.  
  45. FUNCTION RevStr(strarg)
  46.   dllcall(Reverse, strarg)
  47.   RevStr = strarg
  48. END FUNCTION
  49.  
  50. FUNCTION GetWords(strarg, longarg)
  51.   GetWords = dllcall(Words, strarg, longarg)
  52. END FUNCTION
  53.  
  54. FUNCTION ColumnAlign(in_str, just_str, dlm_str, eol_str)
  55.   ColumnAlign = dllcall(Align, in_str, just_str, dlm_str, eol_str)
  56. END FUNCTION
  57.  
  58. END MODULE
  59.  
  60.  
  61.  
  62.  FOR x = 65 TO 90
  63.   alpha &= CHR(x)
  64.  NEXT
  65.  PRINT alpha,"\n"
  66.  PRINT O2::RevStr(alpha),"\n"
  67.  p = 1
  68.  Next_Word:
  69.   wd = O2::GetWords("abc(d[xx]+7/6,\"qwerty\")", p)
  70.   IF wd <> "" THEN
  71.     PRINT wd,"\n"
  72.     GOTO Next_Word
  73.   END IF
  74.  quo = """
  75. Given;;a;;text;;file;;of;;many;;lines,;;where;;fields;;within;;a;;line;;
  76. are;;delineated;;by;;a;;single;;'dollar';;character,;;write;;a;;program
  77. that;;aligns;;each;;column;;of;;fields;;by;;ensuring;;that;;words;;in;;each;;
  78. column;;are;;separated;;by;;at;;least;;one;;space.
  79. Further,;;allow;;for;;each;;word;;in;;a;;column;;to;;be;;either;;left;;
  80. justified,;;right;;justified,;;or;;center;;justified;;within;;its;;column.
  81. """
  82.  PRINT O2::ColumnAlign(quo, "LLLLCCCRRRRR", ";;", "\n"  )
  83.  
  84.  O2::Done()
  85.  
  86. line input u
  87.  
Title: Re: DLLC
Post by: JRS on November 05, 2014, 04:17:02 PM
What does this O2 statement do?

AlignedText atxt

Please send me your working SB version to try here. I need to run out for a bit but I'll try it as soon as I get back.

Thanks for all the help!!!

Title: Re: DLLC
Post by: JRS on November 05, 2014, 04:26:54 PM
Your code also exception errors. I'm wondering if this has something to do with the new Oxygen.dll I'm using.

Got to go, I'm late.
Title: Re: DLLC
Post by: Charles Pegge on November 05, 2014, 04:47:17 PM
My version with DLLC and Oxygen attached:

.
Title: Re: DLLC
Post by: JRS on November 05, 2014, 07:45:57 PM
My problem might have been I was using an older copy of the source that didn't have the new Align Text code.  ::)

testo2.sb
Code: Script BASIC
  1. IMPORT O2.inc
  2.  
  3. FOR x = 65 TO 90
  4.   alpha &= CHR(x)
  5. NEXT
  6. PRINT alpha,"\n"
  7. PRINT O2::RevStr(alpha),"\n"
  8. p = 1
  9. Next_Word:
  10.   wd = O2::GetWords("abc(d[xx]+7/6,\"qwerty\")", p)
  11.   IF wd <> "" THEN
  12.     PRINT wd,"\n"
  13.     GOTO Next_Word
  14.   END IF
  15. q = """
  16. Given;;a;;text;;file;;of;;many;;lines,;;where;;fields;;within;;a;;line;;
  17. are;;delineated;;by;;a;;single;;'dollar';;character,;;write;;a;;program
  18. that;;aligns;;each;;column;;of;;fields;;by;;ensuring;;that;;words;;in;;each;;
  19. column;;are;;separated;;by;;at;;least;;one;;space.
  20. Further,;;allow;;for;;each;;word;;in;;a;;column;;to;;be;;either;;left;;
  21. justified,;;right;;justified,;;or;;center;;justified;;within;;its;;column.
  22. """
  23. PRINT O2::ColumnAlign(q, "LLLLCCCRRRRR", ";;", "\n")
  24.  
  25. O2::Done()
  26.  

O2.inc
Code: Script BASIC
  1. MODULE O2
  2.  
  3. include "dllcinc.sb"
  4.  
  5. oxy=dllfile("/scriptbasic/Debugger/modules/oxygen.dll")
  6.  
  7. o2_basic = dllproc( oxy, "o2_basic i =(c*source) " )
  8. o2_exec  = dllproc( oxy, "o2_exec  i =(i call)   " )
  9. o2_error = dllproc( oxy, "o2_error c*=()         " )
  10. o2_errno = dllproc( oxy, "o2_errno i =()         " )
  11. o2_len   = dllproc( oxy, "o2_len   i =()         " )
  12. o2_mode  = dllproc( oxy, "o2_mode     (i mode)   " )
  13.  
  14. dllcall o2_mode,1
  15.  
  16. OPEN "/scriptbasic/Debugger/include/O2.src" FOR INPUT AS #1
  17. src = INPUT(LOF(1), 1)
  18. CLOSE(1)
  19.  
  20. a = oxygen(src)
  21. Finish  = dllproc(a,"Finish ()", dllcald(a,0))
  22. Reverse = dllproc(a,"Reverse (c*value)", dllcald(a,1))
  23. Words   = dllproc(a,"getword c* = (c* strraw, i* start)", dllcald(a,2))
  24. ColAlign   = dllproc(a,"AlignText c* = (c* in, c* ju, C* dl, c* cr)", dllcald(a,3))
  25.  
  26. FUNCTION oxygen(src)
  27.   dllcall o2_basic,src
  28.   IF (dllcall(o2_errno)<> 0) THEN
  29.     dllprnt dllcall(o2_error)
  30.     a = 0
  31.   ELSE
  32.     a = dllcall(o2_exec,0)
  33.   END IF
  34.   oxygen = a
  35. END FUNCTION
  36.  
  37. FUNCTION Done
  38.   rtnval = dllcall(Finish)
  39.   dllfile
  40.   Done = rtnval
  41. END FUNCTION
  42.  
  43. ' SB wrapper functions
  44.  
  45. FUNCTION RevStr(strarg)
  46.   dllcall(Reverse, strarg)
  47.   RevStr = strarg
  48. END FUNCTION
  49.  
  50. FUNCTION GetWords(strarg, longarg)
  51.   GetWords = dllcall(Words, strarg, longarg)
  52. END FUNCTION
  53.  
  54. FUNCTION ColumnAlign(in_str, just_str, dlm_str, eol_str)
  55.   ColumnAlign = dllcall(ColAlign, in_str, just_str, dlm_str, eol_str)
  56. END FUNCTION
  57.  
  58. END MODULE
  59.  

O2.src
Code: OxygenBasic
  1. ' O2 source
  2.  
  3. extern
  4.  
  5. function reverse(char*s)
  6. '=======================
  7.  addr ecx,s
  8.   mov edx,0
  9.  .rlen
  10.   mov al,[ecx]
  11.   cmp al,0
  12.   jz xlen
  13.   inc edx
  14.   inc ecx
  15.   jmp rlen
  16.  .xlen
  17.   ;
  18.   addr ecx,s
  19.   add  edx,ecx
  20.   dec ecx
  21.   ;
  22.  .rswap
  23.   inc ecx
  24.   dec edx
  25.   cmp edx,ecx
  26.   jle xswap
  27.   mov al,[ecx]
  28.   mov ah,[edx]
  29.   mov [ecx],ah
  30.   mov [edx],al
  31.   jmp rswap
  32.  .xswap
  33.   end function
  34.  
  35. function getword(char*ss,sys*b) as char*
  36. '=======================================
  37. if b=0 then b=1
  38. byte s at @ss
  39. byte c,d
  40. sys bb,bc
  41. static char z[128]
  42. a=0
  43. bb=b
  44.  
  45. 'SKIP LEADING SPACES
  46. do
  47.   c=s[b]
  48.   select c
  49.    case 33 to 255,0 : exit do 'SKIP SPACE
  50.  end select
  51.   b++
  52. end do
  53. bc=b
  54.  '
  55. 'QUOTES
  56. select c
  57.  case 34,39
  58.    do
  59.      b+=1
  60.      d=s[b]
  61.      if d=0 or d=c then b+=1 : jmp fwd done
  62.    end do
  63. end select
  64. 'WORDS AND SYMBOLS
  65. do
  66.   c=s[b]
  67.   select c
  68.   case 0 to 32    : exit do
  69.   case 35         : jmp fwd more
  70.   case 33 to 47   : 'symbols
  71.  case 48 to 57   : jmp fwd more 'numbers
  72.  case 58 to 64   : 'symbols
  73.  case 65 to 90   : jmp fwd more 'capitals
  74.  case 95         : jmp fwd more 'underscore
  75.  case 91 to 96   : 'symbols
  76.  case 97 to 122  : jmp fwd more 'lower case
  77.  case 123 to 127 : 'symbols
  78.  case 128 to 255 : jmp fwd more 'higher ascii
  79. end select
  80.  
  81. if b=bc then b++
  82.   exit do
  83.  
  84.   more:
  85.   b++
  86. end do
  87.  
  88. done:
  89.  
  90. if b > bb then
  91.   z=mid ss,bc,b-bc
  92. else
  93.   z = ""
  94. end if
  95. return z
  96.  
  97. end function
  98.  
  99. =================
  100. Class AlignedText
  101. =================
  102.  
  103. indexbase 1
  104.  
  105. string  buf, bufo, pr, cr, tab, jus, dlm
  106. sys     Cols, Rows, ColWidth[0x100], TotWidth, ColPad, ld
  107.  
  108. method SetText(char*s)
  109. ======================
  110. if not len cr then cr=chr(13,10)
  111. tab=chr(9)
  112. if not len jus then jus=string 200,"L"
  113. buf=s
  114. measure
  115. end method
  116.  
  117.  
  118. method measure()
  119. ================
  120. sys a, b, wa, wb, cm, c, cw
  121. a=1 : b=1
  122. Cols=0 : Rows=0 : ColPad=3
  123. ld=len dlm
  124. if not ld then dlm="," : ld=1 'default to comma
  125. do
  126.   wb=b
  127.   a=instr b,buf,cr
  128.   if a=0 then exit do
  129.   cm=0
  130.   c++
  131.   do
  132.     wa=instr wb,buf,dlm
  133.     if wa=0 or wa>a then exit do
  134.     cm++
  135.     if cm>cols then cols=cm
  136.     cw=wa-wb
  137.     if cw > ColWidth[cm] then ColWidth[cm]=cw
  138.     wb=wa+ld
  139.   end do
  140.   b=a+len cr
  141. end do
  142. rows=c
  143. '
  144. c=0
  145. for i=1 to cols
  146.   ColWidth[ i ]+=ColPad
  147.   c+=ColWidth[ i ]
  148. next
  149. TotWidth=c+len cr
  150. 'print ShowMetrics
  151. end method
  152.  
  153.  
  154. method ShowMetrics() as char*
  155. =============================
  156. pr="METRICS:" cr cr
  157. pr+=rows tab cols tab totwidth cr cr
  158. pr+="column" tab "spacing" cr
  159. for i=1 to cols
  160.   pr+=i tab ColWidth[ i ] cr
  161. next
  162. return pr
  163. end method
  164.  
  165.  
  166. method justify(char*j)
  167. ======================
  168. jus=j
  169. end method
  170.  
  171. method delimiter(char*j)
  172. ========================
  173. dlm=j
  174. end method
  175.  
  176. method endofline(char*j)
  177. ========================
  178. cr=j
  179. end method
  180.  
  181.  
  182. method layout() as char*
  183. ========================
  184. sys a, b, wa, wb, wl, cm, lpos, cpos
  185. bufo=space Rows*TotWidth
  186. a=1 : b=1
  187. do
  188.   wb=b
  189.   a=instr(b,buf,cr)
  190.   if a=0 then exit do
  191.   cm=0
  192.   cpos=1
  193.   do
  194.     wa=instr(wb,buf,dlm)
  195.     if wa=0 or wa>a then exit do
  196.     '
  197.    cm++
  198.     '
  199.    'JUSTIFICATION
  200.    '
  201.    wl=wa-wb
  202.     p=lpos+cpos 'default "L" LEFT ALIGN
  203.    '
  204.    select case asc(jus,cm)
  205.       case "R" : p=lpos+cpos+ColWidth[cm]-wl-Colpad
  206.       case "C" : p=lpos+cpos+( ColWidth[cm]-wl-Colpad )*.5
  207.     end select
  208.     '
  209.    mid bufo,p, mid buf,wb,wl
  210.     cpos+=colwidth[cm]
  211.     wb=wa+ld
  212.   end do
  213.   b=a+len cr
  214.   lpos+=TotWidth
  215.   if lpos<len(bufo) then mid bufo,lpos-1,cr
  216. end do
  217. return bufo
  218. end method
  219.  
  220. end class
  221.  
  222. '#recordof AlignedText
  223.  
  224. AlignedText atxt
  225.  
  226. function AlignText(char *in,*ju,*dl,*cr) as char*
  227. =================================================
  228. atxt.justify         ju
  229. atxt.delimiter       dl
  230. atxt.endofline       cr
  231. atxt.SetText         in
  232. return               atxt.layout
  233. end function
  234.  
  235. sub finish()
  236. '===========
  237.  terminate
  238. end sub
  239.  
  240. function link(sys n) as sys
  241. '==========================
  242.  select n
  243.   case 0 : return @finish
  244.   case 1 : return @reverse
  245.   case 2 : return @getword
  246.   case 3 : return @AlignText
  247.   end select
  248. end function
  249.  
  250. end extern
  251.  
  252. addr link
  253.  

Output

C:\scriptbasic\o2dev>scriba testo2.sb
ABCDEFGHIJKLMNOPQRSTUVWXYZ
ZYXWVUTSRQPONMLKJIHGFEDCBA
abc
(
d
[
xx
]
+
7
/
6
,
"qwerty"
)

  Given        a            text         file       of       many        lines,        where   fields   within        a   line
  are          delineated   by           a        single    'dollar'   character,      write        a
  that         aligns       each         column     of       fields        by       ensuring     that    words       in   each
  column       are          separated    by         at       least        one
  Further,     allow        for          each      word        in          a          column       to       be   either   left
  justified,   right        justified,   or       center   justified     within          its

C:\scriptbasic\o2dev>
Title: Re: DLLC
Post by: JRS on November 05, 2014, 09:58:50 PM
Charles,

This thread (http://www.oxygenbasic.org/forum/index.php?topic=294.0) seems to have complementing functions to what we have already gathered. Do you think this should be included as well?

Title: Re: DLLC
Post by: Charles Pegge on November 06, 2014, 01:06:00 AM
Hi John,

It has a different GetWord parser, adapted for plain text.

refactored for char*

Code: OxygenBasic
  1.   function GetTextWord(char*s, sys *i) as char*
  2.   =============================================
  3.   static string w
  4.   sys           a,j
  5.   byte          b at @s
  6.   j=i
  7.   i--
  8.   @b--
  9.   do
  10.     i++
  11.     @b++
  12.     select b
  13.     case 0            : exit do      'end of string
  14.    case 33 to 255    : exit do      'non white space
  15.    end select
  16.   end do
  17.   '
  18.  j=i
  19.   '
  20.  do
  21.     select b
  22.     case 0 to 32 : exit do
  23.     end select
  24.     i++
  25.     @b++
  26.   end do
  27.   '
  28.  if i>j
  29.     w=mid s,j,i-j
  30.     return w
  31.   else
  32.     return ""
  33.   end if
  34.   '
  35.  end function
  36.  
Title: Re: DLLC
Post by: JRS on November 06, 2014, 01:09:01 AM
Nice!

I was thinking that the column align function could use a format option. Does O2 have a numeric format feature built into the syntax?
Title: Re: DLLC
Post by: JRS on November 06, 2014, 01:34:55 AM
I'm getting a strange error with the new GetTextWord function.


C:\scriptbasic\o2dev>scriba testo2.sb
 ; ASM ERR:     >!!  Unidentified instruction: >
 ; AFTER:       ._here_
 ; LINE:        290

C:\scriptbasic\o2dev>


testo2.sb
Code: Script BASIC
  1. IMPORT O2.inc
  2.  
  3. FOR x = 65 TO 90
  4.   alpha &= CHR(x)
  5. NEXT
  6. PRINT alpha,"\n"
  7. PRINT O2::RevStr(alpha),"\n"
  8. p = 1
  9. Next_Word:
  10.   wd = O2::GetWords("abc(d[xx]+7/6,\"qwerty\")", p)
  11.   IF wd <> "" THEN
  12.     PRINT wd,"\n"
  13.     GOTO Next_Word
  14.   END IF
  15. q = """
  16. Given;;a;;text;;file;;of;;many;;lines,;;where;;fields;;within;;a;;line;;
  17. are;;delineated;;by;;a;;single;;'dollar';;character,;;write;;a;;program
  18. that;;aligns;;each;;column;;of;;fields;;by;;ensuring;;that;;words;;in;;each;;
  19. column;;are;;separated;;by;;at;;least;;one;;space.
  20. Further,;;allow;;for;;each;;word;;in;;a;;column;;to;;be;;either;;left;;
  21. justified,;;right;;justified,;;or;;center;;justified;;within;;its;;column.
  22. """
  23. PRINT O2::ColumnAlign(q, "LLLLCCCRRRRR", ";;", "\n")
  24. p = 1
  25. Next_Text_Word:
  26.   wd = O2::GetTextWords("The quick brown fox jumped over the lazy dog's back 1234567890 times.", p)
  27.   IF wd <> "" THEN
  28.     PRINT wd,"\n"
  29.     GOTO Next_Text_Word
  30.   END IF
  31.  
  32. O2::Done()
  33.  

O2.inc
Code: Script BASIC
  1. MODULE O2
  2.  
  3. include "dllcinc.sb"
  4.  
  5. oxy=dllfile("/scriptbasic/Debugger/modules/oxygen.dll")
  6.  
  7. o2_basic = dllproc( oxy, "o2_basic i =(c*source) " )
  8. o2_exec  = dllproc( oxy, "o2_exec  i =(i call)   " )
  9. o2_error = dllproc( oxy, "o2_error c*=()         " )
  10. o2_errno = dllproc( oxy, "o2_errno i =()         " )
  11. o2_len   = dllproc( oxy, "o2_len   i =()         " )
  12. o2_mode  = dllproc( oxy, "o2_mode     (i mode)   " )
  13.  
  14. dllcall o2_mode,1
  15.  
  16. OPEN "/scriptbasic/Debugger/include/O2.src" FOR INPUT AS #1
  17. src = INPUT(LOF(1), 1)
  18. CLOSE(1)
  19.  
  20. a = oxygen(src)
  21. Finish  = dllproc(a,"Finish ()", dllcald(a,0))
  22. Reverse = dllproc(a,"Reverse (c*value)", dllcald(a,1))
  23. Words   = dllproc(a,"getword c* = (c* strraw, i* start)", dllcald(a,2))
  24. ColAlign   = dllproc(a,"AlignText c* = (c* in, c* ju, C* dl, c* cr)", dllcald(a,3))
  25. Text = dllproc(a,"GetTextWord c* = (c* in_str, i* start)", dllcald(a,4))
  26.  
  27. FUNCTION oxygen(src)
  28.   dllcall o2_basic,src
  29.   IF (dllcall(o2_errno)<> 0) THEN
  30.     dllprnt dllcall(o2_error)
  31.     a = 0
  32.   ELSE
  33.     a = dllcall(o2_exec,0)
  34.   END IF
  35.   oxygen = a
  36. END FUNCTION
  37.  
  38. FUNCTION Done
  39.   rtnval = dllcall(Finish)
  40.   dllfile
  41.   Done = rtnval
  42. END FUNCTION
  43.  
  44. ' SB wrapper functions
  45.  
  46. FUNCTION RevStr(strarg)
  47.   dllcall(Reverse, strarg)
  48.   RevStr = strarg
  49. END FUNCTION
  50.  
  51. FUNCTION GetWords(strarg, longarg)
  52.   GetWords = dllcall(Words, strarg, longarg)
  53. END FUNCTION
  54.  
  55. FUNCTION ColumnAlign(in_str, just_str, dlm_str, eol_str)
  56.   ColumnAlign = dllcall(ColAlign, in_str, just_str, dlm_str, eol_str)
  57. END FUNCTION
  58.  
  59. FUNCTION GetTextWords(in_str, start)
  60.   GetTextWords = dllcall(Text, in_str, start)
  61. END FUNCTION
  62.  
  63. END MODULE
  64.  

O2.src
Code: OxygenBasic
  1. ' O2 source
  2.  
  3. extern
  4.  
  5. function reverse(char*s)
  6. '=======================
  7.  addr ecx,s
  8.   mov edx,0
  9.  .rlen
  10.   mov al,[ecx]
  11.   cmp al,0
  12.   jz xlen
  13.   inc edx
  14.   inc ecx
  15.   jmp rlen
  16.  .xlen
  17.   ;
  18.   addr ecx,s
  19.   add  edx,ecx
  20.   dec ecx
  21.   ;
  22.  .rswap
  23.   inc ecx
  24.   dec edx
  25.   cmp edx,ecx
  26.   jle xswap
  27.   mov al,[ecx]
  28.   mov ah,[edx]
  29.   mov [ecx],ah
  30.   mov [edx],al
  31.   jmp rswap
  32.  .xswap
  33.   end function
  34.  
  35. function getword(char*ss,sys*b) as char*
  36. '=======================================
  37. if b=0 then b=1
  38. byte s at @ss
  39. byte c,d
  40. sys bb,bc
  41. static char z[128]
  42. a=0
  43. bb=b
  44.  
  45. 'SKIP LEADING SPACES
  46. do
  47.   c=s[b]
  48.   select c
  49.    case 33 to 255,0 : exit do 'SKIP SPACE
  50.  end select
  51.   b++
  52. end do
  53. bc=b
  54.  '
  55. 'QUOTES
  56. select c
  57.  case 34,39
  58.    do
  59.      b+=1
  60.      d=s[b]
  61.      if d=0 or d=c then b+=1 : jmp fwd done
  62.    end do
  63. end select
  64. 'WORDS AND SYMBOLS
  65. do
  66.   c=s[b]
  67.   select c
  68.   case 0 to 32    : exit do
  69.   case 35         : jmp fwd more
  70.   case 33 to 47   : 'symbols
  71.  case 48 to 57   : jmp fwd more 'numbers
  72.  case 58 to 64   : 'symbols
  73.  case 65 to 90   : jmp fwd more 'capitals
  74.  case 95         : jmp fwd more 'underscore
  75.  case 91 to 96   : 'symbols
  76.  case 97 to 122  : jmp fwd more 'lower case
  77.  case 123 to 127 : 'symbols
  78.  case 128 to 255 : jmp fwd more 'higher ascii
  79. end select
  80.  
  81. if b=bc then b++
  82.   exit do
  83.  
  84.   more:
  85.   b++
  86. end do
  87.  
  88. done:
  89.  
  90. if b > bb then
  91.   z=mid ss,bc,b-bc
  92. else
  93.   z = ""
  94. end if
  95. return z
  96.  
  97. end function
  98.  
  99. =================
  100. Class AlignedText
  101. =================
  102.  
  103. indexbase 1
  104.  
  105. string  buf, bufo, pr, cr, tab, jus, dlm
  106. sys     Cols, Rows, ColWidth[0x100], TotWidth, ColPad, ld
  107.  
  108. method SetText(char*s)
  109. ======================
  110. if not len cr then cr=chr(13,10)
  111. tab=chr(9)
  112. if not len jus then jus=string 200,"L"
  113. buf=s
  114. measure
  115. end method
  116.  
  117.  
  118. method measure()
  119. ================
  120. sys a, b, wa, wb, cm, c, cw
  121. a=1 : b=1
  122. Cols=0 : Rows=0 : ColPad=3
  123. ld=len dlm
  124. if not ld then dlm="," : ld=1 'default to comma
  125. do
  126.   wb=b
  127.   a=instr b,buf,cr
  128.   if a=0 then exit do
  129.   cm=0
  130.   c++
  131.   do
  132.     wa=instr wb,buf,dlm
  133.     if wa=0 or wa>a then exit do
  134.     cm++
  135.     if cm>cols then cols=cm
  136.     cw=wa-wb
  137.     if cw > ColWidth[cm] then ColWidth[cm]=cw
  138.     wb=wa+ld
  139.   end do
  140.   b=a+len cr
  141. end do
  142. rows=c
  143. '
  144. c=0
  145. for i=1 to cols
  146.   ColWidth[ i ]+=ColPad
  147.   c+=ColWidth[ i ]
  148. next
  149. TotWidth=c+len cr
  150. 'print ShowMetrics
  151. end method
  152.  
  153.  
  154. method ShowMetrics() as char*
  155. =============================
  156. pr="METRICS:" cr cr
  157. pr+=rows tab cols tab totwidth cr cr
  158. pr+="column" tab "spacing" cr
  159. for i=1 to cols
  160.   pr+=i tab ColWidth[ i ] cr
  161. next
  162. return pr
  163. end method
  164.  
  165.  
  166. method justify(char*j)
  167. ======================
  168. jus=j
  169. end method
  170.  
  171. method delimiter(char*j)
  172. ========================
  173. dlm=j
  174. end method
  175.  
  176. method endofline(char*j)
  177. ========================
  178. cr=j
  179. end method
  180.  
  181.  
  182. method layout() as char*
  183. ========================
  184. sys a, b, wa, wb, wl, cm, lpos, cpos
  185. bufo=space Rows*TotWidth
  186. a=1 : b=1
  187. do
  188.   wb=b
  189.   a=instr(b,buf,cr)
  190.   if a=0 then exit do
  191.   cm=0
  192.   cpos=1
  193.   do
  194.     wa=instr(wb,buf,dlm)
  195.     if wa=0 or wa>a then exit do
  196.     '
  197.    cm++
  198.     '
  199.    'JUSTIFICATION
  200.    '
  201.    wl=wa-wb
  202.     p=lpos+cpos 'default "L" LEFT ALIGN
  203.    '
  204.    select case asc(jus,cm)
  205.       case "R" : p=lpos+cpos+ColWidth[cm]-wl-Colpad
  206.       case "C" : p=lpos+cpos+( ColWidth[cm]-wl-Colpad )*.5
  207.     end select
  208.     '
  209.    mid bufo,p, mid buf,wb,wl
  210.     cpos+=colwidth[cm]
  211.     wb=wa+ld
  212.   end do
  213.   b=a+len cr
  214.   lpos+=TotWidth
  215.   if lpos<len(bufo) then mid bufo,lpos-1,cr
  216. end do
  217. return bufo
  218. end method
  219.  
  220. end class
  221.  
  222. '#recordof AlignedText
  223.  
  224. AlignedText atxt
  225.  
  226. function AlignText(char *in,*ju,*dl,*cr) as char*
  227. =================================================
  228. atxt.justify         ju
  229. atxt.delimiter       dl
  230. atxt.endofline       cr
  231. atxt.SetText         in
  232. return               atxt.layout
  233. end function
  234.  
  235. function GetTextWord(char*s, sys *i) as char*
  236. =============================================
  237. static string w
  238. sys           a,j
  239. byte          b at @s
  240. j=i
  241. i--
  242. @b--
  243. do
  244.   i++
  245.   @b++
  246.   select b
  247.   case 0            : exit do      'end of string
  248.  case 33 to 255    : exit do      'non white space
  249.  end select
  250. end do
  251. '
  252. j=i
  253. '
  254. do
  255.   select b
  256.   case 0 to 32 : exit do
  257.   end select
  258.   i++
  259.   @b++
  260. end do
  261. '
  262. if i>j
  263.   w=mid s,j,i-j
  264.   return w
  265. else
  266.   return ""
  267. end if
  268. '
  269. end function
  270.  
  271. sub finish()
  272. '===========
  273.  terminate
  274. end sub
  275.  
  276. function link(sys n) as sys
  277. '==========================
  278.  select n
  279.   case 0 : return @finish
  280.   case 1 : return @reverse
  281.   case 2 : return @getword
  282.   case 3 : return @AlignText
  283.   case 4 : return @GetTextWord
  284.   end select
  285. end function
  286.  
  287. end extern
  288.  
  289. addr link
  290.  
Title: Re: DLLC
Post by: Charles Pegge on November 06, 2014, 04:30:27 AM
Does src require a null terminator?
Title: Re: DLLC
Post by: JRS on November 06, 2014, 09:31:15 AM
Everything worked fine until I added the GetTextWord function.

Quote
Does src require a null terminator?

That's something you would have to ask the QxygenBasic author.  ???

Title: Re: DLLC
Post by: Charles Pegge on November 06, 2014, 09:44:03 AM
I can't reproduce the error, John. It looks like junk at the end of the source, caused by an overrun.
Title: Re: DLLC
Post by: JRS on November 06, 2014, 10:02:35 AM
Quote
I can't reproduce the error, John. It looks like junk at the end of the source, caused by an overrun.

Good to know! I'll see what I can do on this end to clean up the O2 source file.

1. Tabs to spaces
2. Remove trailing spaces.
3. Only one extra blank line at the end of the source.


C:\scriptbasic\o2dev>scriba testo2.sb
 ; ASM ERR:     ╕ m!!  Unidentified instruction: ╕
 ; AFTER:       ._here_
 ; LINE:        290


C:\scriptbasic\o2dev>


Is there a chance we have overrun your internal buffer with this new function addition?

Title: Re: DLLC
Post by: JRS on November 06, 2014, 10:13:56 AM
It's something to do with loading the source from the file. I made the source as a string in O2.inc and it worked. (sort of) No exception /O2 error.


C:\scriptbasic\o2dev>scriba testo2.sb
ABCDEFGHIJKLMNOPQRSTUVWXYZ
ZYXWVUTSRQPONMLKJIHGFEDCBA
abc
(
d
[
xx
]
+
7
/
6
,
"qwerty"
)

 Given        a            text         file       of       many        lines,        where   fields   within        a   line
 are          delineated   by           a        single   'dollar'    character,      write        a
 that         aligns       each         column     of       fields        by       ensuring     that    words       in   each
 column       are          separated    by         at       least         one
 Further,     allow        for          each      word        in          a          column       to       be   either   left
 justified,   right        justified,   or       center   justified     within          its                                      The

 qu
ick
 br
own
 fo
x j
ump
ed
ove
r t
he
laz
y d
og'
s b
ack
 12
345
678
90
tim
es.

C:\scriptbasic\o2dev>

Title: Re: DLLC
Post by: JRS on November 06, 2014, 10:31:20 AM
I know what the problem is. I had this same issue on Koding.com. I had to put a SLEEP(1) after the file close to flush to remainder of the file into the string. I may try using the T ext. module  load string function instead of native INPUT.

Update

Putting the SB CLOSE(1) in the Finish function solved the problem and the SLEEP(1) wasn't needed. It seems that CLOSE() right after a INPUT needs some time.
Title: Re: DLLC
Post by: Charles Pegge on November 06, 2014, 10:57:58 AM
Thanks John, useful to know about the file buffering.
Title: Re: DLLC
Post by: JRS on November 06, 2014, 11:32:58 AM
Is the GetTextWords() function broken or am I not understanding how it works?
Title: Re: DLLC
Post by: Aurel on November 06, 2014, 12:41:11 PM
I think that work as you can see from screenshot...
// this time i use latest oxygen.dll    ;)
Code: [Select]
function GetTextWord(char*s, sys *i) as char*
      =============================================
      static string w
      sys           a,j
      byte          b at @s
      j=i
      i--
      @b--
      do
        i++
        @b++
        select b
        case 0            : exit do      'end of string
       case 33 to 255    : exit do      'non white space
       end select
      end do
      '
     j=i
      '
     do
        select b
        case 0 to 32 : exit do
        end select
        i++
        @b++
      end do
      '
     if i>j
        w=mid s,j,i-j
        return w
      else
        return ""
      end if
      '
end function

 print GetTextWord("Aurel",3)   

.
Title: Re: DLLC
Post by: Charles Pegge on November 06, 2014, 12:57:42 PM
I see the parsing problem now!

 missing
  @b+=i

Code: OxygenBasic
  1.   function GetTextWord(char*s, sys*i) as char*
  2.   ============================================
  3.   static string w
  4.   sys           j
  5.   byte          b at strptr s
  6.   j=i
  7.   i--
  8.   @b+=i
  9.   @b--
  10.   do
  11.     i++
  12.     @b++
  13.     select b
  14.     case 0            : exit do      'end of string
  15.    case 33 to 255    : exit do      'non white space
  16.    end select
  17.   end do
  18.   '
  19.  j=i
  20.   '
  21.  do
  22.     select b
  23.     case 0 to 32 : exit do
  24.     end select
  25.     i++
  26.     @b++
  27.   end do
  28.   '
  29.  if i>j
  30.     w=mid s,j,i-j
  31.   else
  32.     w=""
  33.   end if
  34.   return w
  35.   '
  36.  end function
  37.  
Title: Re: DLLC
Post by: JRS on November 06, 2014, 01:06:53 PM
Thanks Charles!

testo2.sb
Code: Script BASIC
  1. IMPORT O2.inc
  2.  
  3. FOR x = 65 TO 90
  4.   alpha &= CHR(x)
  5. NEXT
  6. PRINT alpha,"\n"
  7. PRINT O2::RevStr(alpha),"\n"
  8. p = 1
  9. Next_Word:
  10.   wd = O2::GetWords("abc(d[xx]+7/6,\"qwerty\")", p)
  11.   IF wd <> "" THEN
  12.     PRINT wd,"\n"
  13.     GOTO Next_Word
  14.   END IF
  15. q = """
  16. Given;;a;;text;;file;;of;;many;;lines,;;where;;fields;;within;;a;;line;;
  17. are;;delineated;;by;;a;;single;;'dollar';;character,;;write;;a;;program
  18. that;;aligns;;each;;column;;of;;fields;;by;;ensuring;;that;;words;;in;;each;;
  19. column;;are;;separated;;by;;at;;least;;one;;space.
  20. Further,;;allow;;for;;each;;word;;in;;a;;column;;to;;be;;either;;left;;
  21. justified,;;right;;justified,;;or;;center;;justified;;within;;its;;column.
  22. """
  23. PRINT O2::ColumnAlign(q, "LLLLCCCRRRRR", ";;", "\n"),"\n"
  24. p = 1
  25. Next_Text_Word:
  26.   wd = O2::GetTextWords("The quick brown fox jumped over the lazy dog's back 1234567890 times.", p)
  27.   IF wd <> "" THEN
  28.     PRINT wd,"\n"
  29.     GOTO Next_Text_Word
  30.   END IF
  31.  
  32. O2::Done()
  33.  

O2.inc
Code: Script BASIC
  1. MODULE O2
  2.  
  3. include "dllcinc.sb"
  4.  
  5. oxy=dllfile("/scriptbasic/Debugger/modules/oxygen.dll")
  6.  
  7. o2_basic = dllproc( oxy, "o2_basic i =(c*source) " )
  8. o2_exec  = dllproc( oxy, "o2_exec  i =(i call)   " )
  9. o2_error = dllproc( oxy, "o2_error c*=()         " )
  10. o2_errno = dllproc( oxy, "o2_errno i =()         " )
  11. o2_len   = dllproc( oxy, "o2_len   i =()         " )
  12. o2_mode  = dllproc( oxy, "o2_mode     (i mode)   " )
  13.  
  14. dllcall o2_mode,1
  15.  
  16. OPEN "/scriptbasic/Debugger/include/O2.src" FOR INPUT AS #1
  17. src = INPUT(LOF(1), 1)
  18.  
  19. a = oxygen(src)
  20. Finish  = dllproc(a,"Finish ()", dllcald(a,0))
  21. Reverse = dllproc(a,"Reverse (c*value)", dllcald(a,1))
  22. Words   = dllproc(a,"getword c* = (c* strraw, i* start)", dllcald(a,2))
  23. ColAlign   = dllproc(a,"AlignText c* = (c* in, c* ju, C* dl, c* cr)", dllcald(a,3))
  24. Text = dllproc(a,"GetTextWord c* = (c* in_str, i* start)", dllcald(a,4))
  25.  
  26. FUNCTION oxygen(src)
  27.   dllcall o2_basic,src
  28.   IF (dllcall(o2_errno)<> 0) THEN
  29.     dllprnt dllcall(o2_error)
  30.     a = 0
  31.   ELSE
  32.     a = dllcall(o2_exec,0)
  33.   END IF
  34.   oxygen = a
  35. END FUNCTION
  36.  
  37. FUNCTION Done
  38.   rtnval = dllcall(Finish)
  39.   dllfile
  40.   CLOSE(1)
  41.   Done = rtnval
  42. END FUNCTION
  43.  
  44. ' SB wrapper functions
  45.  
  46. FUNCTION RevStr(strarg)
  47.   dllcall(Reverse, strarg)
  48.   RevStr = strarg
  49. END FUNCTION
  50.  
  51. FUNCTION GetWords(strarg, longarg)
  52.   GetWords = dllcall(Words, strarg, longarg)
  53. END FUNCTION
  54.  
  55. FUNCTION ColumnAlign(in_str, just_str, dlm_str, eol_str)
  56.   ColumnAlign = dllcall(ColAlign, in_str, just_str, dlm_str, eol_str)
  57. END FUNCTION
  58.  
  59. FUNCTION GetTextWords(in_str, start)
  60.   GetTextWords = dllcall(Text, in_str, start)
  61. END FUNCTION
  62.  
  63. END MODULE
  64.  

O2.src
Code: OxygenBasic
  1. ' O2 source
  2.  
  3. extern
  4.  
  5. function reverse(char*s)
  6. '=======================
  7.  addr ecx,s
  8.   mov edx,0
  9.  .rlen
  10.   mov al,[ecx]
  11.   cmp al,0
  12.   jz xlen
  13.   inc edx
  14.   inc ecx
  15.   jmp rlen
  16.  .xlen
  17.   ;
  18.   addr ecx,s
  19.   add  edx,ecx
  20.   dec ecx
  21.   ;
  22.  .rswap
  23.   inc ecx
  24.   dec edx
  25.   cmp edx,ecx
  26.   jle xswap
  27.   mov al,[ecx]
  28.   mov ah,[edx]
  29.   mov [ecx],ah
  30.   mov [edx],al
  31.   jmp rswap
  32.  .xswap
  33.   end function
  34.  
  35. function getword(char*ss,sys*b) as char*
  36. '=======================================
  37. if b=0 then b=1
  38. byte s at @ss
  39. byte c,d
  40. sys bb,bc
  41. static char z[128]
  42. a=0
  43. bb=b
  44.  
  45. 'SKIP LEADING SPACES
  46. do
  47.   c=s[b]
  48.   select c
  49.    case 33 to 255,0 : exit do 'SKIP SPACE
  50.  end select
  51.   b++
  52. end do
  53. bc=b
  54.  '
  55. 'QUOTES
  56. select c
  57.  case 34,39
  58.    do
  59.      b+=1
  60.      d=s[b]
  61.      if d=0 or d=c then b+=1 : jmp fwd done
  62.    end do
  63. end select
  64. 'WORDS AND SYMBOLS
  65. do
  66.   c=s[b]
  67.   select c
  68.   case 0 to 32    : exit do
  69.   case 35         : jmp fwd more
  70.   case 33 to 47   : 'symbols
  71.  case 48 to 57   : jmp fwd more 'numbers
  72.  case 58 to 64   : 'symbols
  73.  case 65 to 90   : jmp fwd more 'capitals
  74.  case 95         : jmp fwd more 'underscore
  75.  case 91 to 96   : 'symbols
  76.  case 97 to 122  : jmp fwd more 'lower case
  77.  case 123 to 127 : 'symbols
  78.  case 128 to 255 : jmp fwd more 'higher ascii
  79. end select
  80.  
  81. if b=bc then b++
  82.   exit do
  83.  
  84.   more:
  85.   b++
  86. end do
  87.  
  88. done:
  89.  
  90. if b > bb then
  91.   z=mid ss,bc,b-bc
  92. else
  93.   z = ""
  94. end if
  95. return z
  96.  
  97. end function
  98.  
  99. =================
  100. Class AlignedText
  101. =================
  102.  
  103. indexbase 1
  104.  
  105. string  buf, bufo, pr, cr, tab, jus, dlm
  106. sys     Cols, Rows, ColWidth[0x100], TotWidth, ColPad, ld
  107.  
  108. method SetText(char*s)
  109. ======================
  110. if not len cr then cr=chr(13,10)
  111. tab=chr(9)
  112. if not len jus then jus=string 200,"L"
  113. buf=s
  114. measure
  115. end method
  116.  
  117.  
  118. method measure()
  119. ================
  120. sys a, b, wa, wb, cm, c, cw
  121. a=1 : b=1
  122. Cols=0 : Rows=0 : ColPad=3
  123. ld=len dlm
  124. if not ld then dlm="," : ld=1 'default to comma
  125. do
  126.   wb=b
  127.   a=instr b,buf,cr
  128.   if a=0 then exit do
  129.   cm=0
  130.   c++
  131.   do
  132.     wa=instr wb,buf,dlm
  133.     if wa=0 or wa>a then exit do
  134.     cm++
  135.     if cm>cols then cols=cm
  136.     cw=wa-wb
  137.     if cw > ColWidth[cm] then ColWidth[cm]=cw
  138.     wb=wa+ld
  139.   end do
  140.   b=a+len cr
  141. end do
  142. rows=c
  143. '
  144. c=0
  145. for i=1 to cols
  146.   ColWidth[ i ]+=ColPad
  147.   c+=ColWidth[ i ]
  148. next
  149. TotWidth=c+len cr
  150. 'print ShowMetrics
  151. end method
  152.  
  153.  
  154. method ShowMetrics() as char*
  155. =============================
  156. pr="METRICS:" cr cr
  157. pr+=rows tab cols tab totwidth cr cr
  158. pr+="column" tab "spacing" cr
  159. for i=1 to cols
  160.   pr+=i tab ColWidth[ i ] cr
  161. next
  162. return pr
  163. end method
  164.  
  165.  
  166. method justify(char*j)
  167. ======================
  168. jus=j
  169. end method
  170.  
  171. method delimiter(char*j)
  172. ========================
  173. dlm=j
  174. end method
  175.  
  176. method endofline(char*j)
  177. ========================
  178. cr=j
  179. end method
  180.  
  181.  
  182. method layout() as char*
  183. ========================
  184. sys a, b, wa, wb, wl, cm, lpos, cpos
  185. bufo=space Rows*TotWidth
  186. a=1 : b=1
  187. do
  188.   wb=b
  189.   a=instr(b,buf,cr)
  190.   if a=0 then exit do
  191.   cm=0
  192.   cpos=1
  193.   do
  194.     wa=instr(wb,buf,dlm)
  195.     if wa=0 or wa>a then exit do
  196.     '
  197.    cm++
  198.     '
  199.    'JUSTIFICATION
  200.    '
  201.    wl=wa-wb
  202.     p=lpos+cpos 'default "L" LEFT ALIGN
  203.    '
  204.    select case asc(jus,cm)
  205.       case "R" : p=lpos+cpos+ColWidth[cm]-wl-Colpad
  206.       case "C" : p=lpos+cpos+( ColWidth[cm]-wl-Colpad )*.5
  207.     end select
  208.     '
  209.    mid bufo,p, mid buf,wb,wl
  210.     cpos+=colwidth[cm]
  211.     wb=wa+ld
  212.   end do
  213.   b=a+len cr
  214.   lpos+=TotWidth
  215.   if lpos<len(bufo) then mid bufo,lpos-1,cr
  216. end do
  217. return bufo
  218. end method
  219.  
  220. end class
  221.  
  222. '#recordof AlignedText
  223.  
  224. AlignedText atxt
  225.  
  226. function AlignText(char *in,*ju,*dl,*cr) as char*
  227. =================================================
  228. atxt.justify         ju
  229. atxt.delimiter       dl
  230. atxt.endofline       cr
  231. atxt.SetText         in
  232. return               atxt.layout
  233. end function
  234.  
  235. function GetTextWord(char*s, sys*i) as char*
  236. '===========================================
  237. static string w
  238. sys           j
  239. byte          b at strptr s
  240. j=i
  241. i--
  242. @b+=i
  243. @b--
  244. do
  245.   i++
  246.   @b++
  247.   select b
  248.     case 0            : exit do      'end of string
  249.    case 33 to 255    : exit do      'non white space
  250.  end select
  251. end do
  252. j=i
  253. do
  254.   select b
  255.     case 0 to 32 : exit do
  256.   end select
  257.   i++
  258.   @b++
  259. end do
  260. if i>j
  261.   w=mid s,j,i-j
  262. else
  263.   w=""
  264. end if
  265. return w
  266.  
  267. end function
  268.  
  269. sub finish()
  270. '===========
  271.  terminate
  272. end sub
  273.  
  274. function link(sys n) as sys
  275. '==========================
  276.  select n
  277.   case 0 : return @finish
  278.   case 1 : return @reverse
  279.   case 2 : return @getword
  280.   case 3 : return @AlignText
  281.   case 4 : return @GetTextWord
  282.   end select
  283. end function
  284.  
  285. end extern
  286.  
  287. addr link
  288.  

Output

C:\scriptbasic\o2dev>scriba testo2.sb
ABCDEFGHIJKLMNOPQRSTUVWXYZ
ZYXWVUTSRQPONMLKJIHGFEDCBA
abc
(
d
[
xx
]
+
7
/
6
,
"qwerty"
)

 Given        a            text         file       of       many        lines,        where   fields   within        a   line
 are          delineated   by           a        single   'dollar'    character,      write        a
 that         aligns       each         column     of       fields        by       ensuring     that    words       in   each
 column       are          separated    by         at       least         one
 Further,     allow        for          each      word        in          a          column       to       be   either   left
 justified,   right        justified,   or       center   justified     within          its
The
quick
brown
fox
jumped
over
the
lazy
dog's
back
1234567890
times.

C:\scriptbasic\o2dev>
Title: Re: DLLC
Post by: JRS on November 06, 2014, 01:19:56 PM
Code: Script BASIC
  1. SPLITA "The quick brown fox jumped over the lazy dog's back 1234567890 times." BY " " TO a
  2. FOR x = 0 to UBOUND(a)
  3.   PRINT a[x],"\n"
  4. NEXT  
  5.  

In one line.
Code: Script BASIC
  1. PRINT REPLACE("The quick brown fox jumped over the lazy dog's back 1234567890 times.", " ", "\n"),"\n"
  2.  


C:\scriptbasic\o2dev>scriba altmeth.sb
The
quick
brown
fox
jumped
over
the
lazy
dog's
back
1234567890
times.

C:\scriptbasic\o2dev>


Sometimes making the right call can save time and effort.  :)
Title: Re: DLLC
Post by: JRS on November 07, 2014, 12:11:38 AM
Charles,

I'm going to remove the GetTextWord() function as it would be a duplication what SB already does well. I still would like to have numeric formating for the AlignText() function. It would be great if you had an array sort function laying around.  :)

I have two goals with creating this library with you.

1. To provide a library of functions that enhance the syntax of SB dynamically with JIT functions provided by O2.

2. Create a library of functions that show the features of O2 and provides a test suite to run against current O2 builds.
Title: Re: DLLC
Post by: Charles Pegge on November 07, 2014, 07:24:13 AM
Hi John,

Proposed format:



'params
' source string
' format string
' n padding spaces between columns
' field delimiter string
' end of line string

'FORMAT STRING:
'L left align (default)
'C centre align
'R right align

' 1...n optional number fixing column width.
         (data exceeding the column width is right-clipped


PRINT O2::ColumnAlign(quo, "L16 LCR R20", 1, ";;", "\n"  )
Title: Re: DLLC
Post by: JRS on November 07, 2014, 08:05:57 AM
That looks good. Will we be able to have text and numerics and formatting only applies to number columns?
Title: Re: DLLC
Post by: Charles Pegge on November 07, 2014, 08:43:35 AM
Yes, L is the preferred alignment for alpha text, and R for number text.

Where a width is not specified, a column will be adjusted to accommodate its widest member.

A metrics function is needed to return column offsets and widths for processing in SB
Title: Re: DLLC
Post by: JRS on November 07, 2014, 08:55:08 AM
This might be a good time to start interacting with SB arrays. Dave's engine code does a great job of parsing, returning SB arrays. My next goal is array sort and that would mean passing O2 a SB array and it returning (in the original array argument) the sorted results.
Title: Re: DLLC
Post by: Aurel on November 07, 2014, 10:16:43 AM
Charles...
did i miss something ?
Function GetTextWord (char,int) extract words in loop ..right?
Is there function like Parse(delimiter,text) or in another words is there a easy way to modify
function  GetTextWord  to function buffer = Parse(delimiter,text)
or something similar already exists in examples.
if not there is no problem ..don't worry  ;)
Title: Re: DLLC
Post by: JRS on November 07, 2014, 01:07:06 PM
Charles,

This is my first attempt at using SB arrays in an extension module.

Windows - Iup::Info
C:\SB3\test>sb3 iupinfo.sb
SYSTEMLANGUAGE: English (United States)
DRIVER: Win32
SYSTEM: WinXP
SYSTEMLOCALE: 1252  (ANSI - Latin I)
COMPUTERNAME: JRS-C997F91780C
USERNAME: John
MONITORSINFO: 0 0 1280 800

SCREENSIZE: 1280x766
SCREENDEPTH: 32
VIRTUALSCREEN: 0 0 1280 800
DLGFGCOLOR:   0   0   0
DLGBGCOLOR: 236 233 216
DEFAULTFONT: Tahoma,  8
DEFAULTFONTSIZE: 8
TXTFGCOLOR:   0   0   0
TXTBGCOLOR: 255 255 255
C:\SB3\test>

Linux - Iup::Info
jrs@laptop:~/sb/test$ scriba iupinfo.sb
SYSTEMLANGUAGE: en-us
DRIVER: GTK
SYSTEM: Linux
SYSTEMLOCALE: UTF-8
COMPUTERNAME: laptop
USERNAME: jrs
MONITORSINFO: 0 0 1280 800

SCREENSIZE: 1280x749
SCREENDEPTH: 24
VIRTUALSCREEN: 0 0 1280 800
DLGFGCOLOR:   0   0   0
DLGBGCOLOR: 220 218 213
DEFAULTFONT: Ubuntu 11
DEFAULTFONTSIZE: 11
TXTFGCOLOR:   0   0   0
TXTBGCOLOR: 255 255 255
jrs@laptop:~/sb/test$


Code: [Select]
IMPORT iup.bas
IUP::Open()

IUP::Info(INFO)

PRINT "SYSTEMLANGUAGE: ",INFO{"SYSTEMLANGUAGE"},"\n"
PRINT "DRIVER: ",INFO{"DRIVER"},"\n"
PRINT "SYSTEM: ",INFO{"SYSTEM"},"\n"
PRINT "SYSTEMLOCALE: ",INFO{"SYSTEMLOCALE"},"\n"
PRINT "COMPUTERNAME: ",INFO{"COMPUTERNAME"},"\n"
PRINT "USERNAME: ", INFO{"USERNAME"},"\n"
PRINT "MONITORSINFO: ",INFO{"MONITORSINFO"},"\n"
PRINT "SCREENSIZE: ",INFO{"SCREENSIZE"},"\n"
PRINT "SCREENDEPTH: ",INFO{"SCREENDEPTH"},"\n"
PRINT "VIRTUALSCREEN: ",INFO{"VIRTUALSCREEN"},"\n"
PRINT "DLGFGCOLOR: ",INFO{"DLGFGCOLOR"},"\n"
PRINT "DLGBGCOLOR: ",INFO{"DLGBGCOLOR"},"\n"
PRINT "DEFAULTFONT: ",INFO{"DEFAULTFONT"},"\n"
PRINT "DEFAULTFONTSIZE: ",INFO{"DEFAULTFONTSIZE"},"\n"
PRINT "TXTFGCOLOR: ",INFO{"TXTFGCOLOR"},"\n"
PRINT "TXTBGCOLOR: ",INFO{"TXTBGCOLOR"},"\n"

INFO IUP Function
Code: C
  1. #define ELEMENTS(x) (sizeof (x) / sizeof *(x))
  2.  
  3. besFUNCTION(PuiInfo)
  4.   VARIABLE Argument;
  5.   unsigned long __refcount_;
  6.   LEFTVALUE Lval;
  7.   char buffer[1024];
  8.   int i;
  9.   const char *glbvar[] = {
  10.   "SYSTEMLANGUAGE",
  11.   "DRIVER",
  12.   "SYSTEM",
  13.   "SYSTEMLOCALE",
  14.   "COMPUTERNAME",
  15.   "USERNAME",
  16.   "MONITORSINFO",
  17.   "SCREENSIZE",
  18.   "SCREENDEPTH",
  19.   "VIRTUALSCREEN",
  20.   "DLGFGCOLOR",
  21.   "DLGBGCOLOR",
  22.   "DEFAULTFONT",
  23.   "DEFAULTFONTSIZE",
  24.   "TXTFGCOLOR",
  25.   "TXTBGCOLOR"
  26.   };
  27.  
  28.   besRETURNVALUE = NULL;
  29.  
  30.   Argument = besARGUMENT(1);
  31.   besLEFTVALUE(Argument,Lval);
  32.   besRELEASE(*Lval);
  33.   *Lval = NULL;
  34.  
  35.   *Lval = besNEWARRAY(0,ELEMENTS(glbvar)*2-1);
  36.   if( *Lval == NULL )return COMMAND_ERROR_MEMORY_LOW;
  37.  
  38.   for (i =0; i < ELEMENTS(glbvar);i++) {
  39.     ARRAYVALUE(*Lval,2*i) = besNEWSTRING(strlen(glbvar[i]));
  40.     memcpy(STRINGVALUE(ARRAYVALUE(*Lval,2*i)),glbvar[i],strlen(glbvar[i]));
  41.     memset(buffer,0,1024);
  42.     strcpy(buffer, IupGetGlobal(glbvar[i]));
  43.     ARRAYVALUE(*Lval,2*i+1) = besNEWSTRING(strlen(buffer));
  44.     memcpy(STRINGVALUE(ARRAYVALUE(*Lval,2*i+1)),buffer,strlen(buffer));
  45.   }
  46.  
  47.   besALLOC_RETURN_LONG;
  48.   LONGVALUE(besRETURNVALUE) = -1;
  49. besEND
  50.  
Title: Re: DLLC
Post by: JRS on November 07, 2014, 07:09:50 PM
Charles,

I was thinking maybe something like this for a FORMAT string would be more SB friendly.

Code: Script BASIC
  1. fmtstr = "L,L,C,L,R~-###,###.00~,R"
  2.  

The text alignment is independent of the numeric formating if applied. Integers may not require formatting and justification only. SB uses a "%~-###,###.00~" mask format as an alternate to the standard "%d" C style.

Quote from: SB docs
An alternate format BASIC-like for numbers has the form %~format~ where format can be:

# Digit or space

0 Digit or zero

^ Stores a number in exponential format. Unlike QB's USING format this is a place-holder like the #.

. The position of the decimal point.

, Separator.

- Stores minus if the number is negative.

+ Stores the sign of the number.
Title: Re: DLLC
Post by: Mike Lobanovsky on November 08, 2014, 01:59:48 AM
Gentlemen,

In my humble opinion there is nothing better than a standard C language printf() primitive. It is present on every platform in every implementation of a C runtime that's there as a system library. An average implementation of printf() function is more than 500 lines of C code long.

Do you really think it's worth trying to beat the stake?

(P.S. When answering, please consider that we're talking about a language feature here rather that a simple know-how tip)
Title: Re: DLLC
Post by: Charles Pegge on November 08, 2014, 03:55:51 AM
Hi Aurel,

Quote
Is there function like Parse(delimiter,text) or in another words is there a easy way to modify
function  GetTextWord  to function buffer = Parse(delimiter,text)

I use several specialised word parsers in the o2 compiler. Their behaviour is set for specific tasks, but they all have the same params: string s, i byref.

The most important one for raw Basic is tword(s,i)

They also set various global state variables

ascw : ascii of first character
ascn : ascii of next word / end of line
swd  : start position
lenw : length of word


i       : (byref) position of next word / end of line

complete set of parsing functions (FB)

src/o2lexi.bas
Title: Re: DLLC
Post by: Charles Pegge on November 08, 2014, 04:08:34 AM
Hi John,

I'll focus on the text-table processing aspect for now. It will be a very useful generic for databases, as well as text tabulation.

There will be functions for extracting rows and columns which will provide useful outputs, compatible with the SB splitter / array-maker.
Title: Re: DLLC
Post by: Charles Pegge on November 08, 2014, 04:19:08 AM

Hi Mike,

Yes, we can pass the whole printf, or rather, sprintf stuff onto MSVCRT :)

For Basic number formats, I quite like 0 and ##.## placeholder notation, but in a utility function rather than core o2.
Title: Re: DLLC
Post by: JRS on November 08, 2014, 09:49:53 AM
Sounds good. SB's splitter/array maker works well gluing together diverse matrix structures. (index, associative or a combination of both) Thanks for the update!

I'm going to be digging into Dave's SB array handling code to get a better handle how it works. Hopefully all O2 will have to do is the sort part.

This will also enhance the TinyScheme SB interface dealing with returned Lisp lists strings to be processed further.

Just thinking a None alignment option/place holder might be more flexible.

OR

,, = None or Default action taken.

Code: Script BASIC
  1. fmtstr = "L,,,R~-##.00~,R~-##,###.00~,R"
  2.  

Quote
Hide messages posted by members on my ignore list.

Ignore list, nice!
Title: Re: DLLC
Post by: JRS on November 08, 2014, 06:21:26 PM
Quote
n padding spaces between columns

If this parameter is 0 (zero) then could you use a TAB character? This would be handy as my SPLIT delimiter.
Title: Re: DLLC
Post by: JRS on November 08, 2014, 11:44:44 PM
Charles,

Here is a quick prototype in SB of the direction I think your going. Can you confirm?

Code: Script BASIC
  1.  
  2. FUNCTION FormatLine(ln)
  3.   SPLITA ln BY dlm TO col
  4.   FOR x = 0 to UBOUND(col)
  5.     IF fmt[x,0] = "L" THEN
  6.       tmp = LEFT(col[x] & STRING(fmt[x,1]," "),fmt[x,1])
  7.       IF spc = 0 THEN
  8.         tmp &= "\t"
  9.       ELSE
  10.         tmp &= STRING(spc," ")
  11.       END IF
  12.       rs &= tmp
  13.     ELSE IF fmt[x,0] = "R" THEN  
  14.       tmp = RIGHT(STRING(fmt[x,2]," ") & FORMAT(fmt[x,2],col[x]), fmt[x,1])
  15.       IF spc = 0 THEN
  16.         tmp &= "\t"
  17.       ELSE
  18.         tmp &= STRING(spc," ")
  19.       END IF
  20.       rs &= tmp
  21.     END IF
  22.   NEXT
  23.   FormatLine = rs
  24. END FUNCTION            
  25.  
  26. spc = 1
  27. dlm = ","
  28. rs = ""
  29.  
  30. fmt[0,0]="L"
  31. fmt[0,1]=10
  32. fmt[1,0]="R"
  33. fmt[1,1]=10
  34. fmt[1,2]="%~$-###.00~"
  35.  
  36. PRINT FormatLine("John,1.5"),"\n"
  37.  


jrs@laptop:~/sb/sb22/test$ scriba fmtline.sb
John       $   1.50
jrs@laptop:~/sb/sb22/test$



Title: Re: DLLC
Post by: Charles Pegge on November 09, 2014, 01:13:46 AM
Hi John,

Yes Similar, but column maximums are collected first. then overridden by any specified widths.

I think padding might be better expressed as a string. This will satisfy the requirements of both data layouts and displayed text layouts. For instance, using vertical bars between columns:

" | "
Title: Re: DLLC
Post by: JRS on November 09, 2014, 08:39:19 AM
The | (vertical bar) sounds like a good delimiter for the format string.

I'll cleanup my by the line method with column widths as a requirement and using my format array. I also plan to make this a comma-separated values (http://tools.ietf.org/html/rfc4180) (CSV) input string only. ("A String",123,...")

FYI - SB has a secrete weapon (undocumented) called SPLITAQ. (for spliting CSV strings)  :)

Quote from: SB source
SPLITAQ

SPLITAQ string BY string QUOTE string TO array

Split a string into an array using the second string as delimiter.
The delimited fields may optionally be quoted with the third string.
If the string to be split has zero length the array becomes undefined.
When the delimiter is a zero length string each array element will contain a
single character of the string.

Leading and trailing delimiters are accepted and return an empty element
in the array. For example :-

Code: Script BASIC
  1.    SPLITAQ ",'A,B',C," BY "," QUOTE "'" TO Result
  2.  
   will generate
Code: Script BASIC
  1.                  Result[0] = ""
  2.                  Result[1] = "A,B"
  3.                  Result[2] = "C"
  4.                  Result[3] = ""
  5.  

Note that this kind of handling of trailing and leading empty elements is different
from the handling of the same by the command SPLIT and SPLITA which do ignore
those empty elements. This command is useful to handle lines exported as CSV from
Excel or similar application.

The QUOTE string is really a string and need not be a single character. If there is an
unmatched quote string in the string to be split then the rest of the string until its end
is considered quoted.
Title: T.bas - CSV Line Formatter
Post by: JRS on November 09, 2014, 01:20:56 PM
Here is my CSV line formatter. It should be self explanatory. I'm going to add this to the existing T.bas (Tools) extension module include file.

Quote from: 4 Mike
As an alternative to the BASIC like format mask, you are free to use the SB printf style format mask.

%[flags][width][.precision]type  type = can only be "dioxXueEfgGsc".

Code: Script BASIC
  1. ' result = FormatLine(in_str, fmt_str, quo_char, num_spc) Note: num_spc = -1 uses TAB
  2.  
  3. FUNCTION FormatLine(ln,fmtstr,qc,nsp)
  4.   SPLITAQ ln BY "," QUOTE qc TO col
  5.   SPLITA fmtstr BY "|" TO fmtcmd
  6.   rs = ""
  7.   FOR x = 0 to UBOUND(col)
  8.     SPLITA fmtcmd[x] BY ":" TO fmt
  9.     IF fmt[0] = "L" THEN
  10.       tmp = LEFT(col[x] & STRING(fmt[1]," "),fmt[1])
  11.       GOSUB Margin
  12.     ELSE IF fmt[0] = "R" THEN
  13.       IF fmt[2] <> undef THEN
  14.         tmp = FORMAT(fmt[2],col[x])
  15.       ELSE
  16.         tmp = col[x]
  17.       END IF
  18.       tmp = RIGHT(STRING(fmt[1]," ") & tmp, fmt[1])
  19.       GOSUB Margin
  20.     ELSE IF fmt[0] = "C" THEN
  21.       pad = fmt[1] - LEN(col[x])
  22.       pboth = pad \ 2
  23.       prt = pad % 2
  24.       tmp = STRING(pboth," ") & col[x] & STRING(pboth," ") & STRING(prt," ")
  25.       GOSUB Margin
  26.     END IF
  27.   NEXT
  28.   GOTO Done
  29.  
  30.   Margin:
  31.   IF nsp = -1 THEN
  32.     tmp &= "\t"
  33.   ELSE
  34.     tmp &= STRING(nsp," ")
  35.   END IF
  36.   rs &= tmp  
  37.   RETURN
  38.  
  39.   Done:
  40.   FormatLine = rs
  41. END FUNCTION
  42.  
  43. amt = "|C:6|R:10:%~#,###.00~"
  44. fmtstr = "L:20|R:5" & amt & amt & amt & amt
  45. PRINT FormatLine("\"John Spikowski\",123,30,10.5,60,20.75,90,35.25,120,1234.99",fmtstr,"\"",0),"\n"
  46.  


jrs@laptop:~/sb/sb22/test$ time scriba fmtline.sb
John Spikowski      123  30     10.50  60     20.75  90     35.25 120  1,234.99

real   0m0.008s
user   0m0.008s
sys    0m0.000s
jrs@laptop:~/sb/sb22/test$


If using nsp = -1 (TAB character)

John Spikowski         123     30        10.50     60        20.75     90        35.25    120     1,234.99


FYI

If you wish to Skip a column, you can do the following. (or just about anything else besides L,C,R) || also works.

Code: Script BASIC
  1. fmtstr = "L:20|S|C:6|R:10:%~#,###.00~|C:6|R:10:%~#,###.00~|C:6|R:10:%~#,###.00~|C:6|R:10:%~#,###.00~"
  2.  

John Spikowski        30     10.50  60     20.75  90     35.25 120  1,234.99



Title: Re: T.bas - CSV Line Formatter
Post by: JRS on November 09, 2014, 08:47:55 PM
While searching for some .csv sample data to play with, I ran into this data set. It had the right amount of columns and rows.

(https://lh6.googleusercontent.com/ozf9CYNgPuAFnPbFshFWWiYNPU5w4P9gWU9IagLqGubeK_9EHI1u8R130imbh1sjdhDMbjq-CCudDIT08Y4P6eIbF3H1ptG85Tnbo02Abqu1qQcw18A)

Code: Script BASIC
  1. ' result = FormatLine(in_str, fmt_str, quo_char, num_spc) Note: num_spc = -1 uses TAB
  2.  
  3. FUNCTION FormatLine(ln,fmtstr,qc,nsp)
  4.   SPLITAQ ln BY "," QUOTE qc TO col
  5.   SPLITA fmtstr BY "|" TO fmtcmd
  6.   rs = ""
  7.   FOR x = 0 to UBOUND(col)
  8.     SPLITA fmtcmd[x] BY ":" TO fmt
  9.     IF fmt[0] = "L" THEN
  10.       tmp = LEFT(col[x] & STRING(fmt[1]," "),fmt[1])
  11.       GOSUB Margin
  12.     ELSE IF fmt[0] = "R" THEN
  13.       IF fmt[2] <> undef THEN
  14.         tmp = FORMAT(fmt[2],col[x])
  15.       ELSE
  16.         tmp = col[x]
  17.       END IF
  18.       tmp = RIGHT(STRING(fmt[1]," ") & tmp, fmt[1])
  19.       GOSUB Margin
  20.     ELSE IF fmt[0] = "C" THEN
  21.       pad = fmt[1] - LEN(col[x])
  22.       pboth = pad \ 2
  23.       prt = pad % 2
  24.       tmp = STRING(pboth," ") & col[x] & STRING(pboth," ") & STRING(prt," ")
  25.       GOSUB Margin
  26.     END IF
  27.   NEXT
  28.   GOTO Done
  29.  
  30.   Margin:
  31.   IF nsp = -1 THEN
  32.     tmp &= "\t"
  33.   ELSE
  34.     tmp &= STRING(nsp," ")
  35.   END IF
  36.   rs &= tmp  
  37.   RETURN
  38.  
  39.   Done:
  40.   FormatLine = rs
  41. END FUNCTION
  42.  
  43. OPEN "SacramentocrimeJanuary2006.csv" FOR INPUT AS #1
  44. OPEN "sac.fmt" FOR OUTPUT AS #2
  45. fmtstr = "L:15|L:30|R:4|L:4|R:6|L:35|L:6|R:10:%~-##0.0000~|R:10:%~-##0.0000~"
  46. LINE INPUT #1, hdr
  47. WHILE NOT EOF(1)
  48.   LINE INPUT #1, csvln
  49.   csvln = CHOMP(csvln)
  50.   PRINT #2, FormatLine(csvln,fmtstr,"",2),"\n"
  51. WEND  
  52.  
  53. CLOSE(1)
  54. CLOSE(2)
  55.  

Output (7584 rows)

jrs@laptop:~/sb/sb22/test$ time scriba fmtline.sb

real   0m0.454s
user   0m0.415s
sys    0m0.036s
jrs@laptop:~/sb/sb22/test$

Code: [Select]
1/1/06 0:00      3108 OCCIDENTAL DR                 3  3C      1115  10851(A)VC TAKE VEH W/O OWNER        2404       38.5504   -121.3914 
1/1/06 0:00      2082 EXPEDITION WAY                5  5A      1512  459 PC  BURGLARY RESIDENCE           2204       38.4735   -121.4902 
1/1/06 0:00      4 PALEN CT                         2  2A       212  10851(A)VC TAKE VEH W/O OWNER        2404       38.6578   -121.4621 
1/1/06 0:00      22 BECKFORD CT                     6  6C      1443  476 PC PASS FICTICIOUS CHECK         2501       38.5068   -121.4270 
1/1/06 0:00      3421 AUBURN BLVD                   2  2A       508  459 PC  BURGLARY-UNSPECIFIED         2299       38.6374   -121.3846 
Title: Re: DLLC
Post by: Charles Pegge on November 11, 2014, 04:32:44 AM
Hi John,

I've added sort-by-column and CSV support to the o2 table_utils. Also using MSVCRT sprintf to format numerics, if a format string is defined per column.

To read/write CSV, the delimiter is simply set to "," including the double quotes.

To get printf formats, the format specifier is included between quotes: '%f'

Thus a format string might look like this

" L 20  L 20 C R 20 '%e'  R 20 '%f' "

I'm still testing...
Title: Re: DLLC
Post by: JRS on November 11, 2014, 08:12:07 AM
Sort-by-column, cool! The standard C format masks should put a smile on Mike face. Looking forward to giving it a try.

Title: T.bas CSV to SQLite3
Post by: JRS on November 11, 2014, 11:07:03 PM
Charles,

Here is a prototype of my CSV2SQL to be function. I almost gave up on it because it was so slow until I discovered TRANSACTION.

Quote
By default SQLite will evaluate every INSERT / UPDATE statement within a unique transaction. If performing a large number of inserts, it's advisable to wrap your operation in a transaction:

Code: Script BASIC
  1. IMPORT sqlite.bas
  2.  
  3. OPEN "SacramentocrimeJanuary2006.csv" FOR INPUT AS #1
  4. db = sqlite::open("sac116.db")
  5. fmtstr = "SSISISIRR"
  6. LINE INPUT #1, hdr
  7. hdr = CHOMP(hdr)
  8. SPLITA hdr BY "," TO col
  9. SPLITA fmtstr BY "" TO typ
  10. lastcol = UBOUND(col)
  11. sql = "CREATE TABLE crime ("
  12. FOR x = 0 TO lastcol
  13.   tmp = ""
  14.   IF typ[x] = "S" THEN
  15.     tstr = " TEXT"
  16.   ELSE IF typ[x] = "I" THEN
  17.     tstr = " INTEGER"
  18.   ELSE IF typ[x] = "R" THEN
  19.     tstr = " REAL"
  20.   END IF
  21.   tmp &= col[x] & tstr
  22.   IF x <> lastcol THEN tmp &= ", "
  23.   sql &= tmp
  24. NEXT
  25. sql &= ");"
  26. sqlite::execute(db, sql)
  27. sqlite::execute(db, "BEGIN TRANSACTION")
  28. WHILE NOT EOF(1)
  29.   sql = "INSERT INTO crime VALUES ("
  30.   LINE INPUT #1, csvln
  31.   csvln = CHOMP(csvln)
  32.   SPLITAQ csvln BY "," QUOTE "" TO col
  33.   FOR x = 0 TO lastcol
  34.     IF typ[x] = "S" THEN
  35.       tmp = "'" & col[x] & "'"
  36.     ELSE
  37.       tmp = col[x]
  38.     END IF
  39.     IF x <> lastcol THEN tmp &= ", "
  40.     sql &= tmp
  41.   NEXT
  42.   sql &= ");"
  43.   sqlite::execute(db, sql)
  44. WEND
  45. sqlite::execute(db, "END TRANSACTION")
  46. sqlite::close(db)
  47. CLOSE(1)
  48.  

Output

jrs@laptop:~/sb/sb22/test$ time scriba csv2sql.sb

real   0m0.763s
user   0m0.457s
sys   0m0.016s
jrs@laptop:~/sb/sb22/test$ sqlite3
SQLite version 3.8.2 2013-12-06 14:53:30
Enter ".help" for instructions
Enter SQL statements terminated with a ";"
sqlite> .open sac116.db
sqlite> SELECT COUNT(*) FROM crime;
7584
sqlite> .q
jrs@laptop:~/sb/sb22/test$

Title: Re: T.bas CSV to SQLite3
Post by: JRS on November 12, 2014, 06:08:48 PM
This is the CSV2SQL example posted earlier in a SUB. If you don't pass the database:table  name argument it will default to using the CSV file name and csv_import as the table name.

Note: I was getting weird filenames being generated with the sqlite::open() function. I  appended a NULL to the end and the problem went away. I need to look closer at AIR's ext. module code.

Update: I took a peek at the SQLite extension module code and noticed AIR used the SB String type rather than a Zstring argument definition. It now works fine without the extra NULL appended. The updated SQLite ext. module will be in the 2.2 release.

Code: Script BASIC
  1. IMPORT sqlite.bas
  2.  
  3. SUB CSV2SQL(csvfn, fmtstr, dbtbl, quo)
  4.   OPEN csvfn FOR INPUT AS #1
  5.   IF dbtbl = "" THEN
  6.     dbfn = LEFT(csvfn,INSTR(csvfn,".")) & "db"
  7.     tblfn = "csv_import"
  8.   ELSE
  9.     SPLIT dbtbl BY ":" TO dbfn, tblfn
  10.     dbfn &= ".db"
  11.   END IF    
  12.   db = sqlite::open(dbfn & CHR(0))
  13.   LINE INPUT #1, hdr
  14.   hdr = CHOMP(hdr)
  15.   SPLITA hdr BY "," TO col
  16.   SPLITA fmtstr BY "" TO typ
  17.   lastcol = UBOUND(col)
  18.   sql = "CREATE TABLE " & tblfn & "("
  19.   FOR x = 0 TO lastcol
  20.     tmp = ""
  21.     IF typ[x] = "S" THEN
  22.       tstr = " TEXT"
  23.     ELSE IF typ[x] = "I" THEN
  24.       tstr = " INTEGER"
  25.     ELSE IF typ[x] = "R" THEN
  26.       tstr = " REAL"
  27.     END IF
  28.     tmp &= col[x] & tstr
  29.     IF x <> lastcol THEN tmp &= ", "
  30.     sql &= tmp
  31.   NEXT
  32.   sql &= ");"
  33.   sqlite::execute(db, sql)
  34.   sqlite::execute(db, "BEGIN TRANSACTION")
  35.   WHILE NOT EOF(1)
  36.     sql = "INSERT INTO " & tblfn & " VALUES ("
  37.     LINE INPUT #1, csvln
  38.     csvln = CHOMP(csvln)
  39.     SPLITAQ csvln BY "," QUOTE "" TO col
  40.     FOR x = 0 TO lastcol
  41.       IF typ[x] = "S" THEN
  42.         tmp = "'" & col[x] & "'"
  43.       ELSE
  44.         tmp = col[x]
  45.       END IF
  46.       IF x <> lastcol THEN tmp &= ", "
  47.       sql &= tmp
  48.     NEXT
  49.     sql &= ");"
  50.     sqlite::execute(db, sql)
  51.   WEND
  52.   sqlite::execute(db, "END TRANSACTION")
  53.  
  54.   sqlite::close(db)
  55.   CLOSE(1)
  56. END SUB
  57.  
  58.  
  59. ' CSV2SQL "SacramentocrimeJanuary2006.csv", "SSISISIRR", "sac16:crime", ""
  60. CSV2SQL "SacramentocrimeJanuary2006.csv", "SSISISIRR", "", ""
  61.  

Title: Re: DLLC
Post by: JRS on November 13, 2014, 09:48:21 AM
Quote from: Charles
I'm still testing...

How things going with your O2 data processing library?

I'm pretty happy with the two new T.bas additions. It would be helpful if others would give the new SB routines a try on CSV files you may want to convert. (Works on Windows, Linux and Android native)

.
Title: Re: DLLC
Post by: Charles Pegge on November 13, 2014, 11:09:59 AM
Hi John,

We have text and numeric sorting, ascending and descending.

Also selectable multiple columns.

Pls correct me if I'm wrong but I don't think there is a way of passing variadic data via a module wrapper function. So dllcall has to be invoked directly for variadic procedures.
Title: Re: DLLC
Post by: JRS on November 13, 2014, 11:40:44 AM
All SB extension module functions are variadic based. Your right, there is no way to dynamically service a variadic defined function on the fly. (that I know) I had to emulate it in many of the IUP SB interface calls. If I remember correctly, this is how IUP is doing Vbox in their source. I just copied the code for most part.

Code: C
  1. /*
  2. Ihandle* IupVbox(Ihandle *child, ...);  <iup.h>
  3.  
  4. Creates a void container for composing elements vertically. It is a box that
  5. arranges the elements it contains from top to bottom.
  6. */
  7. besFUNCTION(PuiVbox)
  8.   VARIABLE Argument;
  9.   Ihandle *ih;
  10.   unsigned long i;
  11.   char *child;
  12.  
  13.   ih = IupVbox(NULL);
  14.  
  15.   for( i=1 ; i <= (unsigned)besARGNR ; i++ ){
  16.     Argument = besARGUMENT(i);
  17.     besDEREFERENCE(Argument);
  18.     memcpy(&child, STRINGVALUE(Argument), sizeof(child));
  19.     IupAppend(ih, child);
  20.     }
  21.  
  22.   besRETURN_POINTER(ih);
  23. besEND
  24.  
Title: Re: DLLC
Post by: JRS on November 13, 2014, 01:19:15 PM
Charles,

I modified one of my old IUP examples to show the variadic emulation of appending child controls to a IupVbox container.

Code: Script BASIC
  1. ' Script BASIC Rapid-Q form conversion
  2.  
  3. IMPORT iup.bas
  4.  
  5. ' SBIUP-Q INIT
  6.  
  7. Iup::Open()
  8. Iup::SetGlobal("DEFAULTFONT", "Sans, 7.5")
  9.  
  10. ' CREATE FORM
  11.  
  12. Form = Iup::Create("dialog")
  13.        Iup::SetAttributes(Form, "RASTERSIZE=320x240, TITLE=\"Form1\"")
  14.  
  15. '    vbx     = Iup::Create("vbox")
  16. '              Iup::Append(Form, vbx)
  17.  
  18.      Label1  = Iup::Create("label")
  19.                Iup::SetAttributes(Label1, "TITLE=\"Customer\", RASTERSIZE=55x13, FLOATING=YES, POSITION=\"19,19\"")
  20. '              Iup::Append(vbx, Label1)
  21.  
  22.      Edit1   = Iup::Create("text")
  23.                Iup::SetAttributes(Edit1, "RASTERSIZE=121x21, FLOATING=YES, POSITION=\"72,16\"")
  24. '              Iup::Append(vbx, Edit1)
  25.  
  26.      Button1 = Iup::Create("button")
  27.                Iup::SetAttributes(Button1, "TITLE=\"&Quit\", RASTERSIZE=75x25, FLOATING=YES, POSITION=\"107,164\"")
  28. '              Iup::Append(vbx, Button1)
  29.  
  30. ' Code to show variadic emulation
  31. vbx = Iup::Vbox(Label1, Edit1, Button1)
  32. Iup::Append(Form, vbx)
  33.  
  34.                
  35. ' SET CALLBACKS
  36.  
  37. Iup::SetCallback(Form, "CLOSE_CB", ADDRESS(Win_exit()))
  38.  
  39.  
  40. ' CALLBACKS FUNCTIONS
  41.  
  42. SUB Win_exit
  43.   Iup::ExitLoop = TRUE
  44. END SUB
  45.  
  46. ' MAIN
  47.  
  48. ' Iup::Show(Iup::LayoutDialog(Form))
  49. Iup::Show(Form)
  50. Iup::MainLoop
  51. Iup::Close
  52.  
Title: Re: T.bas CSV to SQLite3
Post by: JRS on November 13, 2014, 06:53:27 PM
I'm thinking of doing one last function to format a row from a SQLite return. Most of the work is already done and I should have something posted soon. FmtSQLRow

It turns out all I had to do is check if the passed data line was an array.

Code: Script BASIC
  1. ' result = FormatLine(in_str/array, fmt_str, quo_char, num_spc) Note: num_spc = -1 uses TAB
  2.  
  3. IMPORT sqlite.bas
  4.  
  5. FUNCTION FormatLine(ln,fmtstr,qc,nsp)
  6.   IF ISARRAY(ln) THEN
  7.     col = ln
  8.   ELSE
  9.     SPLITAQ ln BY "," QUOTE qc TO col
  10.   END IF
  11.      
  12.   SPLITA fmtstr BY "|" TO fmtcmd
  13.   rs = ""
  14.   FOR x = 0 to UBOUND(col)
  15.     SPLITA fmtcmd[x] BY ":" TO fmt
  16.     IF fmt[0] = "L" THEN
  17.       tmp = LEFT(col[x] & STRING(fmt[1]," "),fmt[1])
  18.       GOSUB Margin
  19.     ELSE IF fmt[0] = "R" THEN
  20.       IF fmt[2] <> undef THEN
  21.         tmp = FORMAT(fmt[2],col[x])
  22.       ELSE
  23.         tmp = col[x]
  24.       END IF
  25.       tmp = RIGHT(STRING(fmt[1]," ") & tmp, fmt[1])
  26.       GOSUB Margin
  27.     ELSE IF fmt[0] = "C" THEN
  28.       pad = fmt[1] - LEN(col[x])
  29.       pboth = pad \ 2
  30.       prt = pad % 2
  31.       tmp = STRING(pboth," ") & col[x] & STRING(pboth," ") & STRING(prt," ")
  32.       GOSUB Margin
  33.     END IF
  34.   NEXT
  35.   GOTO Done
  36.  
  37.   Margin:
  38.   IF nsp = -1 THEN
  39.     tmp &= "\t"
  40.   ELSE
  41.     tmp &= STRING(nsp," ")
  42.   END IF
  43.   rs &= tmp  
  44.   RETURN
  45.  
  46.   Done:
  47.   FormatLine = rs
  48. END FUNCTION
  49.  
  50. db = sqlite::Open("sac16.db")
  51. stmt = sqlite::Query(db,"SELECT * FROM crime LIMIT 1")
  52. sqlite::Row(stmt)
  53. sqlite::FetchArray(stmt,columns)
  54. fmtstr = "L:15|L:30|R:4|L:4|R:6|L:35|L:6|R:10:%~-##0.0000~|R:10:%~-##0.0000~"
  55. PRINT FormatLine(columns,fmtstr,"",2),"\n"
  56. sqlite::Close(db)
  57.  

Output
Code: [Select]
jrs@laptop:~/sb/sb22/test$ scriba fmtsqlrow.sb
1/1/06 0:00      3108 OCCIDENTAL DR                 3  3C      1115  10851(A)VC TAKE VEH W/O OWNER        2404       38.5504   -121.3914 
jrs@laptop:~/sb/sb22/test$
Title: Re: DLLC
Post by: Charles Pegge on November 15, 2014, 03:08:11 AM
Hi John,

I think string passing might be a better alternative to using variadics.

If you take table-processing far enough - then some form of SQL begins to emerge, in order to specify all the options. I wonder if there is a sweet spot for managing simple databases with lightweight code - say 20k of source code.
Title: Re: DLLC
Post by: JRS on November 15, 2014, 08:02:56 AM
Charles,

SQLite is hard to beat. The SB extension module static links the SQLite library in so there is zero dependencies. It's important on Android as there is no native Andorid Linux SQLite support. (only Java VM) You may want to look at SQLite for O2 rather than recreating the wheel in code. IMHO

Could SQLite be static linked into Oxygen.dll? (expanding the O2 keyword set) I believe SQLite can keep it's DB in memory rather than using a file. Maybe a wrapper include would be better.

John

.
Title: Re: DLLC
Post by: JRS on November 16, 2014, 09:07:55 AM
Charles,

I haven't lost interest in our DLLC library project and was hoping to get Rob involved with what we have cooking. (TinyScheme, SDL_gfx, SQLite3 and DLLC) Our BASIC children hooked on METHods.  :)

I'm trying to wrap up what is and isn't going to make it in the Script BASIC 2.2 release which I hope to have out by the first of the year.

John
Title: Re: DLLC
Post by: JRS on November 16, 2014, 12:47:20 PM
Charles,

It seems rather easy to use :memory: as the data base rather than a file.

Quote
An SQLite database is normally stored in a single ordinary disk file. However, in certain circumstances, the database might be stored in memory.

The most common way to force an SQLite database to exist purely in memory is to open the database using the special filename ":memory:". In other words, instead of passing the name of a real disk file into one of the sqlite3_open(), sqlite3_open16(), or sqlite3_open_v2() functions, pass in the string ":memory:". For example:

Code: C
  1.     rc = sqlite3_open(":memory:", &db);
  2.  

When this is done, no disk file is opened. Instead, a new database is created purely in memory. The database ceases to exist as soon as the database connection is closed. Every :memory: database is distinct from every other. So, opening two database connections each with the filename ":memory:" will create two independent in-memory databases.

The first run was using the :memory: feature and the second run using a normal file. Both were reading from a .csv disk file.

Output

jrs@laptop:~/sb/sb22/test$ time scriba csv2sql.sb

real   0m0.426s
user   0m0.327s
sys   0m0.000s
jrs@laptop:~/sb/sb22/test$ time scriba csv2sql.sb

real   0m0.627s
user   0m0.325s
sys   0m0.008s
jrs@laptop:~/sb/sb22/test$
Title: Re: DLLC
Post by: Charles Pegge on November 16, 2014, 02:09:52 PM
Hi John,

I see that SQLITE weighs in at around 700K. I'm after something much lighter, for managing simple data tables, that will work well with other inc/ utilities. The question is how much fuctionality can be obtained for 20kb?

(http://animals.phillipmartin.info/animal_elephant_mouse.gif)
Title: Re: DLLC
Post by: JRS on November 16, 2014, 02:28:07 PM
Fat people are happy people.  :P (I've been told. I'm 175 lbs / 6' 1" and have been so since I was 17 years old)

I'll take convenience, portability and ease of use over ball busting teeny weeny code any day.  8)

I guess I would have to be an ASM programmer to realize your 700KB concerns.

@ALL - You can find a SQLite3 example done in O2 in the examples/data processing folder of the O2 zip.

BTW SB < 700KB

Now that I figuured out how to get the speed I was looking for with SQLite, I plan to use it more instead of SB arrays when a sort option is required.
Title: Re: DLLC
Post by: JRS on November 16, 2014, 09:51:01 PM
Quote
WhiteDB is a lightweight database library operating fully in main memory.
Disk is used only for dumping/restoring database and logging.

Data is persistantly kept in the shared memory area: it is available simultaneously
to all processes and is kept intact even if no processes are currently using the
database.

WhiteDB has no server process. Data is read and written directly from/to memory,
no sockets are used between WhiteDB and the application using WhiteDB.

WhiteDB keeps data as N-tuples: each database record is a tuple of N elements.
Each element (record field) may have an arbitrary type amongst the types provided
by WhiteDB. Each record field contains exactly one integer (4 bytes or 8 bytes).
Datatypes which cannot be fit into one integer are allocated separately
and the record field contains an (encoded) pointer to the real data.

WhiteDB is written in pure C in a portable manner and should compile and function
without additional porting at least  under Linux (gcc) and Windows
(native Windows C compiler cl). It has Python and experimental Java bindings.

WhiteDB Project Site (http://whitedb.org/)

Here is the results of the speed test I ran that came with the distribution. Sorry Charles, still a 700KB engine.  :-\ Being lighting fast might be enough to ignore it's weight problem.

Quote
Record pointers: the foundation of a graph database

Searching for a related record through an index using the record id (the standard SQL way) is neither the fastest nor the simplest way of going through complex data structures.

It is much faster and easier to store direct pointers to records.

speed15.c builds a database of 10 million records and stores a pointer to the previously created record to the field 3 of each record. Essentially, all the 10 million records will form a long chain from the last record back to the first. Additionally, we store a pointer to the last record into field 2 of the very first record, to directly access the last record later.

The building test takes 0.66 seconds.

speed16.c traverses through the whole chain of backward pointers and counts the 10 million records in a list.

The traversal test takes 0.15 seconds: in other words, you can traverse almost 100 million records in a linked list in a second.


jrs@laptop:~/whitedb-0.7.3/Examples/speed$ gcc speed15.c -o speed15 -O2 -lwgdb
jrs@laptop:~/whitedb-0.7.3/Examples/speed$ time ./speed15
wg memory error: creating shared memory segment: Specified segment size too large or too small.
wg memory error: create_shared_memory failed.
db creation failed
jrs@laptop:~/whitedb-0.7.3/Examples/speed$ sudo sysctl kernel.shmmax=1000000000
kernel.shmmax = 1000000000
jrs@laptop:~/whitedb-0.7.3/Examples/speed$ time ./speed15
i 10000000

real   0m1.154s
user   0m0.660s
sys   0m0.492s
jrs@laptop:~/whitedb-0.7.3/Examples/speed$ gcc speed16.c -o speed16 -O2 -lwgdb
jrs@laptop:~/whitedb-0.7.3/Examples/speed$ time ./speed16
i 10000000

real   0m0.281s
user   0m0.194s
sys   0m0.087s
jrs@laptop:~/whitedb-0.7.3/Examples/speed$


FYI: I'm staying with SQLite3. I like SQL access syntax. It self documents as you code.