-------------------------------------------------------------------------------------------- [BUG/PRB.] VFP 9.0 FIX - FIX THE INLIST() FUNCTION FOR RUSHMORE QUERY OPTIMIZATION January 2025 -------------------------------------------------------------------------------------------- CCB 1. BUG: In vfp9 (and vfp6, vfp7, vfp8), when using UDFs (user-defined functions) in the INLIST() function for Rushmore Query Optimization, sometimes it will cause the memory access violation exception (exception code 0xC0000005). The bug occurs in the FOR clause when SET("OPTIMIZE")="ON" (the default setting), it does not occur in the FOR clause when SET("OPTIMIZE")="OFF". The bug does not occur in the WHERE clause when SET("OPTIMIZE")="ON" (the default setting) or SET("OPTIMIZE")="OFF". The bug occurs in the following commands (in the FOR clause): AVERAGE command BLANK command BROWSE command CALCULATE command CHANGE command COPY TO command COPY TO ARRAY command COUNT command DELETE command DISPLAY command EDIT command EXPORT command INDEX command LABEL command LIST command LOCATE command RECALL command REPLACE command REPLACE FROM ARRAY command REPORT command SCAN command SET FILTER command SORT command SUM command TOTAL command The bug does not occur in the following commands (in the WHERE clause): DELETE - SQL command SELECT - SQL command UPDATE - SQL command The bug also occurs when using some expressions which include field names (in the current work area) in the INLIST() function. The bug also occurs when using some functions which support an optional alias or work area in the INLIST() function: ALIAS() function BOF() function DELETED() function EOF() function FCOUNT() function FIELD() function FLOCK() function FOUND() function LUPDATE() function NDX() function RECCOUNT() function RECNO() function RECSIZE() function RLOCK() function ORDER() function FILTER() function HEADER() function USED() function ISBLANK() function GETFLDSTATE() function Using Rushmore Query Optimization to Speed Data Access (from dv_foxhelp9.chm). Creating Basic Optimizable Expressions A basic optimizable expression takes one of the two following forms: eIndex relOp eExp -or- eExpr relOp eIndex A basic optimizable expression has the following characteristics: eIndex exactly matches the expression on which an index is constructed. eExpr is any expression and can include variables and fields from other unrelated tables. relOp is one of the following relational operators: <, >, =, <=, >=, <>, #, ==, or !=. You can also use the ISNULL( ), BETWEEN( ), or INLIST( ) functions (or their SQL equivalents such as IS NULL, and so on). You can use BETWEEN( ) or INLIST( ) in the following two forms: BETWEEN(eIndex, eExpr, eExpr) -or- INLIST(eIndex, eExpr [, eExpr, eExpr, ...]) The bug occurs when using UDFs (user-defined functions) in the INLIST() function, for example, INDEX ON fld1 TAG fld1 LOCATE FOR INLIST(fld1,"aaa",subproc("bbb")) && no error. LOCATE FOR INLIST(fld1,"aaa",subproc("bbb"),"ccc") && no error. LOCATE FOR INLIST(fld1,"aaa",subproc("bbb"),"ccc","ddd") && causes the memory access violation exception (exception code 0xC0000005)! LOCATE FOR INLIST(fld1,"aaa",subproc("bbb"),subproc("ccc"),"ddd") && causes the memory access violation exception (exception code 0xC0000005)! LOCATE FOR INLIST(fld1,"aaa",subproc("bbb"),subproc("ccc"),subproc("ddd")) && causes the memory access violation exception (exception code 0xC0000005)! The bug also occurs when using some expressions which include field names (in the current work area) in the INLIST() function, for example, INDEX ON fld1 TAG fld1 LOCATE FOR INLIST(fld1,"aaa",fld2) && no error. LOCATE FOR INLIST(fld1,"aaa",fld2,"ccc") && no error. LOCATE FOR INLIST(fld1,"aaa",fld2,"ccc","ddd") && causes the memory access violation exception (exception code 0xC0000005)! LOCATE FOR INLIST(fld1,"aaa",fld2,fld2+fld3,"ddd") && causes the memory access violation exception (exception code 0xC0000005)! LOCATE FOR INLIST(fld1,"aaa",fld2,fld2+fld3,fld2+fld4) && causes the memory access violation exception (exception code 0xC0000005)! The bug also occurs when using some functions which support an optional alias or work area in the INLIST() function, for example, INDEX ON fld1 TAG fld1 LOCATE FOR INLIST(fld1,"aaa",FIELD(2)) && no error. LOCATE FOR INLIST(fld1,"aaa",FIELD(2),"ccc") && no error. LOCATE FOR INLIST(fld1,"aaa",FIELD(2),"ccc","ddd") && causes the memory access violation exception (exception code 0xC0000005)! LOCATE FOR INLIST(fld1,"aaa",FIELD(2),FIELD(3),"ddd") && causes the memory access violation exception (exception code 0xC0000005)! LOCATE FOR INLIST(fld1,"aaa",FIELD(2),FIELD(3),FIELD(4)) && causes the memory access violation exception (exception code 0xC0000005)! 2. CAUSE: There are some BUGs in the following code. 3. RESOLUTION: We can write some code to fix the BUG. Label59b7df :: mov eax , ebx ;0x0059b7df : 8bc3 call Fun49b895 ;0x0059b7e1 : e8af00f0ff test eax , eax ;0x0059b7e6 : 85c0 mov dword ptr [ ebp + 24 ] , eax ;0x0059b7e8 : 894518 je Label4767ef ;0x0059b7eb : 0f84feafedff mov esi , dword ptr [ esi + 8 ] ;0x0059b7f1 : 8b7608 mov dword ptr [ esp + 24 ] , 02h ;0x0059b7f4 : c744241802000000 nop ;0x0059b7fc : 90 Label59b7fd :: cmp dword ptr [esi] , 00h ;0x0059b7fd : 833e00 jne Label59b844 ;0x0059b800 : 7542 mov ecx , dword ptr [ esi + 4 ] ;0x0059b802 : 8b4e04 mov edx , dword ptr [ ebp + 24 ] ;0x0059b805 : 8b5518 mov eax , dword ptr [ Data9370f8 ] ;0x0059b808 : a1f8709300 mov eax , dword ptr [eax] ;0x0059b80d : 8b00 pushd 00h ;0x0059b80f : 6a00 push ecx ;0x0059b811 : 51 mov ecx , dword ptr [ eax + 0E4h ] ;0x0059b812 : 8b88e4000000 push edx ;0x0059b818 : 52 push edi ;0x0059b819 : 57 push ebx ;0x0059b81a : 53 call Fun77ec57 ;0x0059b81b : e837341e00 mov ecx , dword ptr [ esp + 28 ] ;0x0059b820 : 8b4c241c test ecx , ecx ;0x0059b824 : 85c9 je Label59b82f ;0x0059b826 : 7407 cmp eax , 01h ;0x0059b828 : 83f801 jne Label59b82f ;0x0059b82b : 7502 xor eax , eax ;0x0059b82d : 33c0 Label59b82f :: cmp dword ptr [ esp + 24 ] , eax ;0x0059b82f : 39442418 jl Label59b839 ;0x0059b833 : 7c04 mov dword ptr [ esp + 24 ] , eax ;0x0059b835 : 89442418 Label59b839 :: mov eax , dword ptr [ esp + 24 ] ;0x0059b839 : 8b442418 test eax , eax ;0x0059b83d : 85c0 mov esi , dword ptr [ esi + 8 ] ;0x0059b83f : 8b7608 jne Label59b7fd ;0x0059b842 : 75b9 ; ; --------------------------------------------------------------------------------- ; VFP 9.0 FIX - FIX THE INLIST() FUNCTION FOR RUSHMORE QUERY OPTIMIZATION ; June 2024 ; --------------------------------------------------------------------------------- ; CCB ; ; Fix the INLIST() function for Rushmore Query Optimization. ; ; 2024/6/10, by ccb ; cmp dword ptr vfpa_sys9158_data,0FFFFFFFFh je Label59b844 cmp dword ptr vfpa_sys9158_data,00h je Label59b844 test esi , esi je Label59b844 cmp dword ptr [esi] , 00h jne Label59b844 jmp Label59b7fd Label59b844 :: mov ecx , dword ptr [ ebp + 24 ] ;0x0059b844 : 8b4d18 mov eax , dword ptr [ Data9370f8 ] ;0x0059b847 : a1f8709300 mov eax , dword ptr [eax] ;0x0059b84c : 8b00 pushd 00h ;0x0059b84e : 6a00 push esi ;0x0059b850 : 56 push ecx ;0x0059b851 : 51 mov ecx , dword ptr [ eax + 0E4h ] ;0x0059b852 : 8b88e4000000 push edi ;0x0059b858 : 57 push ebx ;0x0059b859 : 53 call Fun77ec57 ;0x0059b85a : e8f8331e00 mov ecx , dword ptr [ esp + 28 ] ;0x0059b85f : 8b4c241c test ecx , ecx ;0x0059b863 : 85c9 je Label59b86e ;0x0059b865 : 7407 cmp eax , 01h ;0x0059b867 : 83f801 jne Label59b86e ;0x0059b86a : 7502 xor eax , eax ;0x0059b86c : 33c0 Label59b86e :: cmp dword ptr [ esp + 24 ] , eax ;0x0059b86e : 39442418 jl Label59b878 ;0x0059b872 : 7c04 mov dword ptr [ esp + 24 ] , eax ;0x0059b874 : 89442418 Label59b878 :: mov eax , dword ptr [ esp + 24 ] ;0x0059b878 : 8b442418 test eax , eax ;0x0059b87c : 85c0 jne Label59b896 ;0x0059b87e : 7516 mov eax , dword ptr [ ebp + 24 ] ;0x0059b880 : 8b4518 call Fun530453 ;0x0059b883 : e8cb4bf9ff mov dword ptr [ ebp + 24 ] , 00h ;0x0059b888 : c7451800000000 pop edi ;0x0059b88f : 5f pop esi ;0x0059b890 : 5e pop ebp ;0x0059b891 : 5d pop ebx ;0x0059b892 : 5b ret 0Ch ;0x0059b893 : c20c00 4. APPLIES TO: VFP 6.0.8167.0 VFP 6.0.8961.0 (SP5) VFP 7.0.0.9262 VFP 7.0.0.9465 (SP1) VFP 8.0.0.2521 VFP 8.0.0.3117 (SP1) VFP 9.0.0.2412 VFP 9.0.0.3504 (SP1) VFP 9.0.0.4611 (SP2) VFP 9.0.0.5015 (SP2) VFP 9.0.0.5411 (SP2) VFP 9.0.0.5721 (SP2) VFP 9.0.0.5815 (SP2) VFP 9.0.0.6303 (SP2) VFP 9.0.0.6602 (SP2) VFP 9.0.0.7423 (SP2) The bug has been fixed in VFP Advanced. 5. REFERENCE WEBSITES: 1, baiyujia.com: http://www.baiyujia.com http://www.baiyujia.com/vfpdocuments/f_vfp9fix326.asp http://www.baiyujia.com/vfpdocuments/f_vfp9fix328.asp http://www.baiyujia.com/vfpdocuments/f_vfp9fix329.asp http://www.baiyujia.com/vfpdocuments/f_vfp9fix330.asp http://www.baiyujia.com/vfpdocuments/f_vfp9fix331.asp http://www.baiyujia.com/vfpdocuments/f_vfp9fix340.asp http://www.baiyujia.com/vfpdocuments/f_vfp9fix341.asp 6. OTHER: For reference only, there is no guarantees. Any questions or suggestions, please send me an email at ccb2000@163.com. |