Previous Thread
Next Thread
Print Thread
Segfault in Debian 12 but not EL7 #37825 24 Feb 25 02:06 PM
Joined: Nov 2006
Posts: 2,262
S
Stephen Funkhouser Offline OP
Member
OP Offline
Member
S
Joined: Nov 2006
Posts: 2,262
Here's a screencapture on how to run the test program and see segfault
https://www.mediafire.com/file/z4grxqwpu7v2f46/devdatimp_segfault_debian12.mp4/file

The attached LSX has devdatimp.run that xcalls datimp (a dialog selecting a csv and then matching the columns). You will trace.prints from contents of the included csv file. The dataimport.bsi module creates a DYNSTRUCT of the select columns with the coders desired names.

This program runs in EL7 with no errors, but segfaults in Debian12 .

I've tested from A-Shell Version 7.0.1756.2 to 7.0.1768.3.

Thanks in advance

Attached Files
devdatimp_segfault_debian12.7z.gpg (305.56 KB, 10 downloads)
SHA1: e94fa70947083e11ef65063df2a1f4c2456376f8

Stephen Funkhouser
Diversified Data Solutions
Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37826 24 Feb 25 03:09 PM
Joined: Jun 2001
Posts: 11,924
J
Jack McGregor Offline
Member
Offline
Member
J
Joined: Jun 2001
Posts: 11,924
I appear to be missing the DATCON subroutine.

Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37827 24 Feb 25 03:34 PM
Joined: Nov 2006
Posts: 2,262
S
Stephen Funkhouser Offline OP
Member
OP Offline
Member
S
Joined: Nov 2006
Posts: 2,262
Here's that is, sorry about that.

Attached Files
datcon.7z.gpg (41.49 KB, 17 downloads)
SHA1: 9105dbbaadfc3180926301622029b3a797e66f06

Stephen Funkhouser
Diversified Data Solutions
Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37828 24 Feb 25 04:53 PM
Joined: Jun 2001
Posts: 11,924
J
Jack McGregor Offline
Member
Offline
Member
J
Joined: Jun 2001
Posts: 11,924
For what it's worth, I can reproduce the problem. Might take a while to figure out...

Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37829 24 Feb 25 04:58 PM
Joined: Nov 2006
Posts: 2,262
S
Stephen Funkhouser Offline OP
Member
OP Offline
Member
S
Joined: Nov 2006
Posts: 2,262
At least, it's fairly easily reproducible. Good Luck!


Stephen Funkhouser
Diversified Data Solutions
Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37837 26 Feb 25 12:14 PM
Joined: Nov 2006
Posts: 2,262
S
Stephen Funkhouser Offline OP
Member
OP Offline
Member
S
Joined: Nov 2006
Posts: 2,262
Any updates?


Stephen Funkhouser
Diversified Data Solutions
Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37839 26 Feb 25 01:25 PM
Joined: Jun 2001
Posts: 11,924
J
Jack McGregor Offline
Member
Offline
Member
J
Joined: Jun 2001
Posts: 11,924
Sorry this is turning out to be difficult to pin down. One reason is the problem changes after recompilation (which I didn't recognize at first). This remains an investigation in progress, but here's my thought process at the moment...

Using the version you sent (which was compiled using compiler edit 1046 (A-Shell 7.0.1757.5), the crash occurs in the NEXT $$i statement below:
Code
0191ca  	FOREACH $$i IN $columns(startkey$)
0190c3  		IF (fn'str_startswith(search$=.KEY($$i), substr$=startkey$) = 1) THEN
0190e0  			$columns(.KEY($$i)) = .NULL
0190ec  		ELSE
0190f0  			EXIT
0190f4  		ENDIF
0190f4  	NEXT $$i

I tried to drill down to the actual point of failure, but it occurs deep inside the standard template library code implementing the ordered maps. However, looking at the code more closely, it appears to commit the faux pas of manipulating the index (or more explicitly, deleting a key-value pair) from the collection being iterated. The fact that the fallout from that differs across platforms/architectures is not entirely shocking. See Writeable Iterators for a brief mention of the issue in our doc. Searching for "deleting key-value pairs while iterating a collection" will turn up some more general comments about the problem and workarounds in languages such as Python and Java.

Interestingly though, when I recompile the program I get (error 70 - invalid dynstruct reference) here...
Code
IF (fn'dataimport_iter_csv(hndl=hndl'first, iter'hndl=iter'hndl, mode=0) = 1) THEN ...
...
FUNCTION fn'dataimport_iter_csv(hndl as B3:INPUTONLY,iter'hndl as B3,mode as B1:INPUTONLY,datimp'dynx="" as DYNSTRUCT:OUTPUTONLY) as f8

That was caused by a loophole in the handling of the initialization of the DYNSTRUCT datimp'dynx to "". However, after closing that loophole, the error moves to this function...
Code
01a94e  PRIVATE FUNCTION fn'dataimport_iter_csv_next_set'dynst(hndl as B3:INPUTONLY,iter'hndl as B3,$columns() as ordmap(varstr;varx),csv'ary() as x0,datimp'dynx="" as DYNSTRUCT,dummy=0 as b1:OUTPUTONLY)
01a9da  MAP1 func$	,s,128,"fn'dataimport_iter_csv_next_set'dynst()"
01aa08  MAP1 col'st	,DATAIMPORT_COLUMNS_ST
01aa08  !>>      MAP2 col'st.column'ct,f,8
01aa08  !>>      MAP2 col'st.label$,s,128
01aa08  !>>      MAP2 col'st.field'name$,s,128
01aa08  !>>      MAP2 col'st.required,BOOLF,8
01aa08  !>>      MAP2 col'st.selected'col,f,8
01aa08  
01aa08  	XCALL FILL, datimp'dynx, ""
01aa17  	FOREACH $$i IN $columns()
01aa41  		col'st = $$i
01aa49  		IF (col'st.selected'col > 0) THEN
01aa55  			datimp'dynx.@col'st.field'name$ = csv'ary(col'st.selected'col)
01aa66  		ENDIF
01aa66  	NEXT $$i
01aa7c  	XPUTARG @datimp'dynx
01aa85  ENDFUNCTION[code]

It seems to think that datimp'dynx.@col'st'field'name$ is a reference to an uninitialized dynstruct, but it's not easy to tell if that's a valid error or not without inserting some code. It occurs to me that it might be nice if there was a variation of the .ISDEF(ds), perhaps .ISBOUND(ds) to simplify testing if dynstruct was initialized/bound to something.

Well that's as far as I've gotten.

Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37841 26 Feb 25 04:56 PM
Joined: Jun 2001
Posts: 11,924
J
Jack McGregor Offline
Member
Offline
Member
J
Joined: Jun 2001
Posts: 11,924
I think we might be close to the bottom on this...

I decided to implement the .ISBOUND(ds) function, making it easier to insert simple traces to track the binding status of the datimp'dynx variable in your routine. The last three traces I get are before the error 70 are (with the value at the end of each line indicating the result of .ISBOUND(datimp'dynx) .)..
Code
main 2-datimp'dynx -1
   func$=[fn'dataimport_iter_csv()],  mode=[1], 10-datimp'dynx 0
     func$=[fn'dataimport_iter_csv_next_set'dynst()], 20-datimp'dynx 0

The relevant code...
Code
1b95d  		IF (fn'dataimport_iter_csv(hndl=hndl'first, iter'hndl=iter'hndl, mode=0) = 1) THEN
01b97e              trace.print (2) "main 2-datimp'dynx", .isbound(datimp'dynx)  ! [jm]
01b9bd  			DO WHILE (fn'dataimport_iter_csv(hndl=hndl'first, iter'hndl=iter'hndl, mode=1, datimp'dynx=datimp'dynx) = 2)
01b9da  Trace.Print "TESTDATIMP stock'number=["+datimp'dynx.stock'number+"] desc=["+datimp'dynx.description+"] test=["+datimp'dynx.test'col'3+"] size=["+datimp'dynx.size+"] "
01ba6f              trace.print (2) "main 3-datimp'dynx", .isbound(datimp'dynx)  ! [jm]
01baae  			LOOP

The "main 2-datimp'dynx -1" trace shows that the datimp'dynx is bound (-1) after the call to fn'dataimport_iter_csv(), as expected. But the next trace shows it unbound (0). That code is here...
Code
01a5b2  FUNCTION fn'dataimport_iter_csv(hndl as B3:INPUTONLY,iter'hndl as B3,mode as B1:INPUTONLY,datimp'dynx="" as DYNSTRUCT:OUTPUTONLY) as f8
...
01a66b      trace.print (2) func$, mode, "10-datimp'dynx", .isbound(datimp'dynx)  ! [jm]
01a6d0  	IF (mode = 0) THEN
...
01a7e0  	ELSEIF (mode = 1) THEN
01a7f0  		ch'csv = $s_ch'csv(iter'hndl)
01a7fe  		INPUT CSV #ch'csv, csv'ary()
01a80c  		IF EOF(ch'csv) THEN
01a815  			XCALL FILL, datimp'dynx, ""
01a824  			mode = 2
01a82c  			.fn = 3
01a834  		ELSE
01a838  			CALL fn'dataimport_iter_csv_next_set'dynst(hndl=hndl, iter'hndl=iter'hndl, $columns()=$s_columns(),csv'ary()=csv'ary(),datimp'dynx=datimp'dynx)
01a85c              trace.print (2) "11-datimp'dynx", .isbound(datimp'dynx)  ! [jm]
01a897  			mode = 1
01a89f  			.fn = 2
01a8a7  		ENDIF
01a8a7  		XPUTARG @datimp'dynx
01a8b0  	ENDIF

So we passed a bound dynstruct to fn'dataimport_iter_csv(), but it gets received as unbound (based on the trace at location 01a66b).

What happened? The :OUTPUTONLY qualifier on the function parameter list. So the dynstruct starts out again as uninitialized, and when it gets passed in that state to fn'dataimport_iter_csv_next_set'dynst(), an error occurs when that function tries to reference field level members. (Note that we don't ever get the trace at 01a85c, so the error is in that routine, specifically:
Code
<DEVDATIMP:1abb9> Trapped Basic Error #70 (Invalid dynstruct reference)

here:
Code
01aa62  PRIVATE FUNCTION fn'dataimport_iter_csv_next_set'dynst(hndl as B3:INPUTONLY,iter'hndl as B3,$columns() as ordmap(varstr;varx),csv'ary() as x0,datimp'dynx="" as DYNSTRUCT,dummy=0 as b1:OUTPUTONLY)
...
01ab1c      trace.print (2) func$, "20-datimp'dynx", .isbound(datimp'dynx)  ! [jm]
01ab6c  	XCALL FILL, datimp'dynx, ""
01ab7b  	FOREACH $$i IN $columns()
01aba5  		col'st = $$i
01abad  		IF (col'st.selected'col > 0) THEN
01abb9  			datimp'dynx.@col'st.field'name$ = csv'ary(col'st.selected'col)
01abca  		ENDIF
01abca  	NEXT $$i
01abe0  	XPUTARG @datimp'dynx
01abe9      trace.print (2) func$, "20-datimp'dynx", .isbound(datimp'dynx)  ! [jm]
01ac39  ENDFUNCTION

Why did it ever work? Because of compiler edit 1047 (one edit beyond the one you had previously compiled this program with)...
Quote

1. Compiler bug fixes (edit 1047):

- Specifying a default value for a parameter in a function declaration was effectively overriding the outputonly clause.

- The :outputonly attribute was making it impossible to specify that parameter by name in a DYNFUNC() call.

(This issue turned out to be complicated due to historical / legacy / compatibility factors, leading to a series of subsequent updates, one of which introduced the ++PRAGMA OVERRIDE_OUTPUTONLY feature which effectively ignores the :OUTPUTONLY qualifier so that the parameters revert to the pre-1047 behavior, i.e. acting like in/out. With that in place, I start getting the variations of the original error you reported (described in my prior post) caused by deleting members of the collection while iterating it. While that leads to a SEGFAULT under Debian 12, in Windows I managed to trap an "incompatible iterator" error running it through a debugger, both of which are consistent with the known problem in the underlying system calls.

Perhaps the compiler can be made smart enough to trap that situation as an error, but for now, you have to avoid it. And although the 7.0.1770.0 update introduces the handy .ISBOUND(ds) function, it doesn't really change the behavior of the version you have relative to this issue. So my position, at least at this moment, is to suggest that you:
  • Either add the ++PRAGMA OVERRIDE_OUTPUTONLY or manually remove those qualifiers from the dynstruct function parameters.
  • Revise the code in fn'dataimport_iter_csv_end_remove'xref() to avoid deleting items from the collection while iterating it.

Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37843 27 Feb 25 06:05 AM
Joined: Nov 2006
Posts: 2,262
S
Stephen Funkhouser Offline OP
Member
OP Offline
Member
S
Joined: Nov 2006
Posts: 2,262
I'll fix this to not do the deletion in the iteration. Could this be made a compiler error? Seems like behavior we don't want to allow as it causing Segfault.

Kyle Vaughn reported the error 70 - invalid dynstruct reference to me and as I was helping debug I ran into the segfault first.

Anything you need from me?


Stephen Funkhouser
Diversified Data Solutions
Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37844 27 Feb 25 09:44 AM
Joined: Jun 2001
Posts: 11,924
J
Jack McGregor Offline
Member
Offline
Member
J
Joined: Jun 2001
Posts: 11,924
I'm going to see if I can get the compiler to flag this today. I think I have all the necessary information. But one detail I forgot to clarify in my previous post -- the 1770.0 update contains one other related fix to avoid generating error #70 on the assignment of the default value "" to the dynstruct parameter, i.e.
Code
function fn'foo(ds="" as dynstruct:outputonly, ...) 

While it's debatable whether assigning a literal to an unbound dynstruct should generate an error, (since it doesn't really make sense), this is a special case where the purpose of the assignment is purely to make the parameter optional, i.e. to provide flexibility to call the function with only the named parameters of interest. So you'll need the update in any case.

Stay tuned...

Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37845 27 Feb 25 10:14 AM
Joined: Jun 2001
Posts: 11,924
J
Jack McGregor Offline
Member
Offline
Member
J
Joined: Jun 2001
Posts: 11,924
Actually, looking into this now, I can see some problems with the compiler being able to flag illegal map update operations during iteration, while allowing legal ones. The documentation (see Writeable iterators) clarifies that it's ok to update the map being iterated, as long as it doesn't affect the keys, offering this example of two ways to do it..
Code
foreach $$i in $capitals()
    $capitals(.key($$i)) = "Beautiful " + ucs($$i)    ! method 1
! or...
    $$i = "Beautiful " + ucs($$i)                     ! method 2
next $$i

The first method presents multiple difficulties, since the key argument, .key($$i)) could theoretically be an ordinary string variable, impossible for the compiler to evaluate at runtime. We could require that only legal key is the .key($$i)) function as in the example, or that you always use method 2. But even with the simpler method 2, I'm not sure the compiler can rule out the possibility that the value being assigned doesn't equate to .NULL.

I suppose it could detect the most straightforward cases (i.e. ".NULL" appearing literally in the expression), i.e. not let the perfect be the enemy of the good.

Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37846 27 Feb 25 10:56 AM
Joined: Nov 2006
Posts: 2,262
S
Stephen Funkhouser Offline OP
Member
OP Offline
Member
S
Joined: Nov 2006
Posts: 2,262
The explicit .NULL makes sense.

I don't have a strong opinion on method 1 vs 2.

Could the run-time be updated log and basic error when the key is illegally changed while iterating?


Stephen Funkhouser
Diversified Data Solutions
Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37847 27 Feb 25 12:16 PM
Joined: Jun 2001
Posts: 11,924
J
Jack McGregor Offline
Member
Offline
Member
J
Joined: Jun 2001
Posts: 11,924
I think I've successfully updated the compiler to treat an assignment of an explicit .NULL to the collection being iterated. So your sample program now triggers this error:
Code
?Illegal collection update during iteration (see Writeable Iterators) (+22685) - $columns(.KEY($$i)) = .NULL

As previously mentioned though, the compiler can't very well detect an assignment to a string variable which turns out to evaluate to .NULL at runtime.

On the runtime side, we can pick up some of those errors where the value being assigned turns out to be .NULL, provided that the lvalue is either explicitly an iterator (e.g. $$i = value$) or explicitly uses the .key() function, e.g. ($columns(.KEY($$i)) = value$). What I don't have a good solution for is the case of a generic-looking assignment, e.g. $m(key$) = value$ that happens to occur while iterating the target collection. That would require sub-classing the collection classes to add some kind of iteration-in-progress state variable, which seems doable but more potentially disruptive than I want to get into right now. The new compiler and runtime tests though should cover nearly all of the exposure though, especially if you stick with using the explicit iterator when making assignments to a collection during iteration.

I still have a lot of testing to do though, to make sure this didn't have side effects with the MLIST and GRIDMAP collections. Hopefully I'll be able to post an beta update this afternoon.

Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37848 27 Feb 25 01:57 PM
Joined: Jun 2001
Posts: 11,924
J
Jack McGregor Offline
Member
Offline
Member
J
Joined: Jun 2001
Posts: 11,924

Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37849 05 Mar 25 03:53 PM
Joined: Nov 2006
Posts: 2,262
S
Stephen Funkhouser Offline OP
Member
OP Offline
Member
S
Joined: Nov 2006
Posts: 2,262
This doesn't resolve the error 70 - invalid dynstruct reference issue.


Stephen Funkhouser
Diversified Data Solutions
Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37850 05 Mar 25 04:51 PM
Joined: Jun 2001
Posts: 11,924
J
Jack McGregor Offline
Member
Offline
Member
J
Joined: Jun 2001
Posts: 11,924
Are you sure you removed the :OUTPUTONLY from the fn'dataimport_iter_csv_next_set'dynst() function declaration ...
Code
019110  PRIVATE FUNCTION fn'dataimport_iter_csv_next_set'dynst(hndl as B3:INPUTONLY,iter'hndl as B3,$columns() as ordmap(varstr;varx),csv'ary() as x0,datimp'dynx="" as DYNSTRUCT,dummy=0 as b1:OUTPUTONLY)


You either need to do that, or add a ++PRAGMA OVERRIDE_OUTPUTONLY statement somewhere. The reason you didn't get the error previously (under -el7) was that the earlier version was in fact ignoring the :OUTPUTONLY modifier when there was a default value, but we subsequently decided that didn't make sense.

Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37851 06 Mar 25 06:35 AM
Joined: Nov 2006
Posts: 2,262
S
Stephen Funkhouser Offline OP
Member
OP Offline
Member
S
Joined: Nov 2006
Posts: 2,262
I am still seeing with

Code
PRIVATE FUNCTION fn'dataimport_iter_csv_next_set'dynst(hndl as HANDLE:INPUTONLY,iter'hndl as ITER_HANDLE,$columns() as ordmap(varstr;varx),csv'ary() as x0,datimp'dynx="" as DYNSTRUCT,dummy=0 as DUMMY:OUTPUTONLY) 



Also happens when

Code
PRIVATE FUNCTION fn'dataimport_iter_csv_next_set'dynst(hndl as HANDLE:INPUTONLY,iter'hndl as ITER_HANDLE,$columns() as ordmap(varstr;varx),csv'ary() as x0,datimp'dynx as DYNSTRUCT)




Stephen Funkhouser
Diversified Data Solutions
Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37853 06 Mar 25 09:23 AM
Joined: Jun 2001
Posts: 11,924
J
Jack McGregor Offline
Member
Offline
Member
J
Joined: Jun 2001
Posts: 11,924
Something is out of sync between us. I started over with the original devdatimp.lsx you emailed (763-775-201-752), and recompiled it under 7.0.1770.0. Both Windows and Debian 12 generate the same compil error (?Illegal collection update during iteration (see Writeable Iterators) (+22683) - $columns(.KEY($$i)) = .NULL) in this routine...
Code
018fde  PRIVATE FUNCTION fn'dataimport_iter_csv_end_remove'xref(hndl as B3:INPUTONLY,iter'hndl as B3,$columns() as ordmap(varstr;varx),dummy=0 as b1:OUTPUTONLY)
019043  MAP1 func$	,s,128,"fn'dataimport_iter_csv_end_remove'xref()"
019072  MAP1 startkey$	,s,0
019072  
019072  	startkey$ = iter'hndl USING "######Z"
019084  	startkey$ += "-"
019090  	FOREACH $$i IN $columns(startkey$)
0190c3  		IF (fn'str_startswith(search$=.KEY($$i), substr$=startkey$) = 1) THEN
0190e0  			$columns(.KEY($$i)) = .NULL
0190ec  		ELSE
0190f0  			EXIT
0190f4  		ENDIF
0190f4  	NEXT $$i
01910a  ENDFUNCTION

That's due to the new compile feature(?) to treat obvious deletions from an ordmap during iteration as compile errors rather than wait for them to show up as some kind of segfault or other unexpected runtime behavior. To work around the compiler complaint (without worrying about what it's going to do to the algorithm logic), I just changed the offending statement to:
Code
0190e0  			$columns(.KEY($$i)) = ""    ! [jm] .NULL

(Setting the value to "" leaves the key-value pair in the index, avoiding the problems with changing the index during iteration.) The recompiled RUN file hash is 771-517-632-117. Running it results in this trace:
Code
TESTDATIMP: - datimp'dynx.column is not defined

And this error:
Code
[p13719-1]<DEVDATIMP:1aa55> Trapped Basic Error #70 (Invalid dynstruct reference) at location counter &h1AA55, last proc/func: fn'dataimport'bsi_'_dataimport_iter_csv_next_set'dynst
[p13719-1]<DEVDATIMP:1aa55> Call stack trace, from program DEVDATIMP :
     From loc 1b78c, Func() @1a53e
     From loc 1a75f, Call Proc() @1a94e

The error #70 problem there is due to the :OUTPUTONLY qualifier in the following function (upstream of the one you modified above), which causes the datimp'dynx dynstruct to get reinitialized to "", wiping out it's previous binding...
Code
FUNCTION fn'dataimport_iter_csv(hndl as B3:INPUTONLY,iter'hndl as B3,mode as B1:INPUTONLY,datimp'dynx="" as DYNSTRUCT:OUTPUTONLY) as f8

If I remove the :OUTPUTONLY qualifier on the DYNSTRUCT and recompile (generating 204-100-215-514), it runs without error.
Or, to take the simpler approach, putting the :OUTPUTONLY qualifier back the way it was but adding ++PRAGMA OVERRIDE_OUTPUTONLY results in 115-622-043-726 and also runs without error.

I'm ignoring here the difference between what you wanted to do in the fn'dataimport_iter_csv_end_remove'xref() function (deleting items from the map) vs just assigning their values to "". If you really wanted to remove items from the map in that function, you'd have to come up with an alternate method, probably involving identifying the keys to be removed during the iteration, but then removing them outside the FOREACH iteration loop.

If your code has diverged from the version you sent me last week, making the above hard to match up to yours, another suggestion would be to use the new .ISBOUND(ds) function in a trace to identify where the dynstruct loses its binding.

Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37854 06 Mar 25 09:43 AM
Joined: Nov 2006
Posts: 2,262
S
Stephen Funkhouser Offline OP
Member
OP Offline
Member
S
Joined: Nov 2006
Posts: 2,262
I fixed the delete while iterating issue but missed the change to fn'dataimport_iter_csv(). Removing the OUTPUTONLY from there does resolve the 70 - invalid dynstruct reference issue.

Maybe this needs to be in a separate post, but we feel that OUTPUTONLY should be valid with DYNSTRUCT. They should behave as close to DEFSTRUCTS as possible. That means they should be bound to the local variable and be empty just like a DEFSTRUCT is.

How do-able is that? Do you disagree that's not desirable behavior?

If that is not the desired behavior, it should be a compile error to use OUTPUTONLY with a Dynstruct.

Thanks in advance!


Stephen Funkhouser
Diversified Data Solutions
Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37855 06 Mar 25 10:34 AM
Joined: Jun 2001
Posts: 11,924
J
Jack McGregor Offline
Member
Offline
Member
J
Joined: Jun 2001
Posts: 11,924
I'm not sure I understand your position. The :OUTPUTONLY qualifier is perfectly valid with a DYNSTRUCT param, but only makes sense if you don't want the caller's DYNSTRUCT value, or its binding, to be passed in. That doesn't interfere with your ability to bind the structure within the function and pass it back. If you wanted to bind the structure in the calling routine and pass it in to the function, you can do that by removing the :OUTPUTONLY. Here's an example showing both variations...
Code
defstruct ST_FOO
    map2 bar,s,10
endstruct

map1 FOO, dynstruct

    call fn'foo'bind(foo=FOO)  ! passing un-bound FOO
                               ! comes back bound and assigned

    ? "After call to fn'foo'bind(), FOO.bar = "; FOO.bar
    call fn'foo2(foo=FOO)      ! passing bound FOO
    end

function fn'foo'bind(foo="" as dynstruct:outputonly)
    map1 stfoo, ST_FOO
    .bindstruct foo, stfoo
    foo.bar = "Martini"
    xputarg @foo
endfunction

function fn'foo2(foo="" as dynstruct)
    ? "in fn'foo2(), foo.bar = "; foo.bar
endfunction


Code
.run dyn4
After call to fn'foo'bind(), FOO.bar = Martini
in fn'foo2(), foo.bar = Martini

Note that the ="" default value is irrelevant in the above case; I added it just to illustrate that it doesn't interfere. So what am I missing?

Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37856 06 Mar 25 11:17 AM
Joined: Nov 2006
Posts: 2,262
S
Stephen Funkhouser Offline OP
Member
OP Offline
Member
S
Joined: Nov 2006
Posts: 2,262
That still doesn't behave like a DEFSTRUCT does with OUTPUTONLY. Yes, we want it to be bound when passed, but allow OUTPUTONLY for the dynstruct to be bound in the caller, but have no value. That's exactly what happens for DEFSTRUCT with OUTPUTONLY.


Stephen Funkhouser
Diversified Data Solutions
Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37857 06 Mar 25 11:31 AM
Joined: Jun 2001
Posts: 11,924
J
Jack McGregor Offline
Member
Offline
Member
J
Joined: Jun 2001
Posts: 11,924
Per our phone conversation (after my post above), we've agreed that the :OUTPUTONLY qualifier should only disable the passing in of values bound to the DYNSTRUCT, but not the binding of the DYNSTRUCT to DEFSTRUCT. Although that wouldn't affect the example above, this modified version illustrates where it would come into play, in fn'foo3() ...
Code
defstruct ST_FOO
    map2 bar,s,10
endstruct

map1 FOO, dynstruct

    call fn'foo'bind(foo=FOO)  ! passing un-bound FOO
                               ! comes back bound and assigned

    ? "After call to fn'foo'bind(), FOO.bar = "; FOO.bar
    call fn'foo2(foo=FOO)      ! passing bound FOO
    call lfn'foo3(foo=FOO)     
    end

function fn'foo'bind(foo="" as dynstruct:outputonly)
    map1 stfoo, ST_FOO
    .bindstruct foo, stfoo
    foo.bar = "Martini"
    xputarg @foo
endfunction

function fn'foo2(foo="" as dynstruct)
    ? "in fn'foo2(), foo.bar = "; foo.bar
endfunction

function fn'foo3(foo="" as dynstruct:outputonly)
    ? "in fn'foo2(), foo.bar = "; foo.bar    ! this would be empty/null due to :outputonly
    foo.bar = "Margarita"                    ! but this would work, since the foo struct binding unaffected by the :outputonly
endfunction

It's on the to-do list, but probably won't be released until after this weekend.

Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37859 09 Mar 25 03:06 PM
Joined: Jun 2001
Posts: 11,924
J
Jack McGregor Offline
Member
Offline
Member
J
Joined: Jun 2001
Posts: 11,924
I think the dynstruct / outputonly issue should be resolved in compiler edit 1064 / A-Shell 7.0.1770.1 ...

compil-7.0.1064-w32.zip
ash-7.0.1770.1-w32-upd.zip
ash-7.0.1770.1-w32c-upd.zip
ash-7.0.1770.1-el7-upd.tz
ash-7.0.1770.1-d12-x86_64-upd.tz
ash70notes.txt

Last edited by Jack McGregor; 10 Mar 25 08:59 AM. Reason: Updated links
Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37946 28 Mar 25 04:42 PM
Joined: Aug 2016
Posts: 403
J
John Andreasen Online Content
Member
Online Content
Member
J
Joined: Aug 2016
Posts: 403
The update related to detecting the deleting of elements during iteration seems to have an issue. I was able to reproduce with the code below. A-Shell 7.0.1770.4 EL7
Code
++PRAGMA PRIVATE_BEGIN
PRIVATE DIMX $ordership'pallet'item_fk_order_pallet_iter_dyn'key,ordmap(varstr;varstr)
++PRAGMA PRIVATE_END

MAIN'ROUTINE:
	$ordership'pallet'item_fk_order_pallet_iter_dyn'key("1-1") = "_"
	$ordership'pallet'item_fk_order_pallet_iter_dyn'key("1-2") = "_"
	$ordership'pallet'item_fk_order_pallet_iter_dyn'key("2-1") = "_"
	$ordership'pallet'item_fk_order_pallet_iter_dyn'key("2-2") = "_"
	CALL fn'ordership'pallet'item_idx_fk_order_pallet_iter_dyn'key_garbage'collect(hndl=1)
	END

!-------------------------------------------------------------------------
!Function: fn'ordership'pallet'item_idx_fk_order_pallet_iter_dyn'key_garbage'collect()
! delete all dyn'key elements for given handle
!
!Parameters:
!--------------------------
! hndl [ITER_HANDLE, i/o] - variable with handle value
!Privates:
!--------------------------
! $ordership'pallet'item_fk_order_pallet_iter_dyn'key()
!-------------------------------------------------------------------------
FUNCTION fn'ordership'pallet'item_idx_fk_order_pallet_iter_dyn'key_garbage'collect(hndl as f8:INPUTONLY,dummy=0 as b1:OUTPUTONLY)
MAP1 func$	,s,0,"fn'ordership'pallet'item_idx_fk_order_pallet_iter_dyn'key_garbage'collect()"
DIMX $gc	,ordmap(varstr;varstr)                                         ! key by dyn'key

	trace.print " " + func$ + " hndl=["+hndl+"] "
	FOREACH $$i IN $ordership'pallet'item_fk_order_pallet_iter_dyn'key(hndl)
		!IF (fn'str_startswith(.key($$i), hndl) = TRUE) THEN
		IF VAL(.KEY($$i)[1;1]) = hndl THEN
			trace.print " " + func$ + " GATHER .key($$i)=["+.key($$i)+"] val=["+edit$(STR($$i), 40)+"]"
			$gc(.key($$i)) = ""
		ELSE
			EXIT
		ENDIF
	NEXT $$i
	FOREACH $$i IN $gc()
		trace.print " " + func$ + " DELETE .key($$i)=["+.key($$i)+"] val=["+edit$(STR($$i), 40)+"]"
		$ordership'pallet'item_fk_order_pallet_iter_dyn'key(.key($$i)) = .NULL
	NEXT $$i
ENDFUNCTION

The operation of deleting the element in the second loop causes a BASIC error.

Last edited by John Andreasen; 28 Mar 25 04:43 PM.
Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37948 29 Mar 25 05:21 PM
Joined: Jun 2001
Posts: 11,924
J
Jack McGregor Offline
Member
Offline
Member
J
Joined: Jun 2001
Posts: 11,924
I'm afraid you've found a crack in the digital firmament. The runtime time detection that was added in order to be able to report an error instead of running the risk of a segmentation fault was far from perfect. It fails to detect workarounds, like assigning .key($$i) to a variable and then using it as the key, or coping .NULL to a variable and using it to delete the key-value pair. At the time, I figured it was better than nothing. But obviously in this case, it's worse than nothing, since your key-value deletion during iteration of a different map is perfectly safe. I have to think about whether it's practical for the runtime system to detect that the map from which the key-value pair is being deleted is not the one being iterated. It may be best to just rely on the compiler to try to flag them before the runtime system gets a chance to see them.

In the meantime though, the workaround is simple enough (just copy the .key($$i) value to a variable and use it for the deletion) ...
Code
map1 key$, s, 20
...
   FOREACH $$i IN $gc()
       key$ = .key($$i)
       $ordership'pallet'item_fk_order_pallet_iter_dyn'key(key$)) = .NULL
   NEXT $$i

Although this is totally off the subject, for what it's worth, I'd like to make sure that you are aware of a couple of syntactic shortcuts related to your code:

The first is that the auto-defined symbol ABC_CURRENT_ROUTINE$ is automatically set to the name of the current function, so there's no real need for your func$ variable definition. (I realize that it's probably your code generator doing this, so weren't not talking about saving wear and tear on your fingers, but it might make the code slightly cleaner to look at.)

The second is that the trace.print statement will automatically output the names of variables specified, along with their values, so for example, instead of :
Code
trace.print " " + func$ + " hndl=["+hndl+"] " 

you could instead use
Code
trace.print ABC_CURRENT_ROUTINE$, hndl

The output of the second version is virtually identical to the first, e.g.
Code
  fn'ordership'pallet'item_idx_fk_order_pallet_iter_dyn'key_garbage'collect  hndl=[1]

Note that variables on the trace.print line get expanded to var=[value], while literal text, symbols and expressions are just output as is.

Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37950 31 Mar 25 09:05 AM
Joined: Aug 2016
Posts: 403
J
John Andreasen Online Content
Member
Online Content
Member
J
Joined: Aug 2016
Posts: 403
Thanks for the reply. You are correct that this is templated code. So, that is where the func$ mapping is from. I am glad you mentioned the keyword though as I was unaware or had forgotten about it as it is common for us to always map the func$ variable and set it to either a string literal of the function name or .LAST_ROUTINE if it is in a program where we can count on the pragma TRACK_LAST_ROUTINE being enabled. As for the detection, just let us know what you decide.

Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37951 31 Mar 25 10:53 AM
Joined: Jun 2001
Posts: 11,924
J
Jack McGregor Offline
Member
Offline
Member
J
Joined: Jun 2001
Posts: 11,924
I figured out a way to make the check for an illegal deletion from a collection during iteration more precise. But as it would close off the back door workaround, so I need to be certain that it isn't going to block any otherwise legal/legitimate operations. Figure on an update in a couple of days.

Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37955 02 Apr 25 07:53 AM
Joined: Nov 2006
Posts: 2,262
S
Stephen Funkhouser Offline OP
Member
OP Offline
Member
S
Joined: Nov 2006
Posts: 2,262
Sounds good. Waiting to give it another go.


Stephen Funkhouser
Diversified Data Solutions
Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37956 02 Apr 25 11:36 AM
Joined: Jun 2001
Posts: 11,924
J
Jack McGregor Offline
Member
Offline
Member
J
Joined: Jun 2001
Posts: 11,924
Ok, here's a new-and-improved test version (7.0.1770.5) which implements the test (for an attempt to delete a collection element while iterating the collection) at a lower level, where it won't be fooled by code like the example above. But note that it also won't be fooled by my suggested workaround of using another variable in place of the .key($$i) or .NULL references.

It still issues that same error 67 (?Invalid collection operation or reference), which perhaps isn't as clear and precise as it could be (as opposed to, say, "?Illegal attempt to delete an element of collection during iteration"), but I wasn't sure it was worth christening a new error code. For now, let me just repeat that this is a limitation of the underlying Standard Template Library implementation of collections, not an ASB issue per se.

ash70notes.txt
ash-7.0.1770.5-w32-upd.zip
ash-7.0.1770.5-el7-upd.tz
ash-7.0.1770.5-d12-x86_64-upd.tz

Re: Segfault in Debian 12 but not EL7 [Re: Stephen Funkhouser] #37957 03 Apr 25 03:37 PM
Joined: Aug 2016
Posts: 403
J
John Andreasen Online Content
Member
Online Content
Member
J
Joined: Aug 2016
Posts: 403
I am now seeing code that is triggering the BASIC error that I think should not. It is reproducible with this example.
Code
DIMX $test,ordmapm(varstr;varstr)
MAP1 key$,s,0

MAIN'ROUTINE:
	$test("1") = "_"
	$test("1") = "_"
	$test("2") = "_"
	key$ = "1"
	DO WHILE NOT .ISNULL($test(key$))
		$test(key$) = .NULL
	LOOP
	END


Moderated by  Jack McGregor, Ty Griffin 

Powered by UBB.threads™ PHP Forum Software 7.7.3