2 * GRUB -- GRand Unified Bootloader
3 * Copyright (c) 1999-2008 Igor Pavlov
4 * Copyright (C) 2008 Free Software Foundation, Inc.
6 * GRUB is free software: you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation, either version 3 of the License, or
9 * (at your option) any later version.
11 * GRUB is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License
17 * along with GRUB. If not, see <http://www.gnu.org/licenses/>.
21 * This code was taken from LZMA SDK 4.58 beta, and was slightly modified
22 * to adapt it to GRUB's requirement.
24 * See <http://www.7-zip.org>, for more information about LZMA.
32 #include <grub/lib/LzFind.h>
33 #include <grub/lib/LzHash.h>
35 #define kEmptyHashValue 0
36 #define kMaxValForNormalize ((UInt32)0xFFFFFFFF)
37 #define kNormalizeStepMin (1 << 10) /* it must be power of 2 */
38 #define kNormalizeMask (~(kNormalizeStepMin - 1))
39 #define kMaxHistorySize ((UInt32)3 << 30)
41 #define kStartMaxLen 3
43 static void LzInWindow_Free(CMatchFinder
*p
, ISzAlloc
*alloc
)
47 alloc
->Free(alloc
, p
->bufferBase
);
52 /* keepSizeBefore + keepSizeAfter + keepSizeReserv must be < 4G) */
54 static int LzInWindow_Create(CMatchFinder
*p
, UInt32 keepSizeReserv
, ISzAlloc
*alloc
)
56 UInt32 blockSize
= p
->keepSizeBefore
+ p
->keepSizeAfter
+ keepSizeReserv
;
59 p
->blockSize
= blockSize
;
62 if (p
->bufferBase
== 0 || p
->blockSize
!= blockSize
)
64 LzInWindow_Free(p
, alloc
);
65 p
->blockSize
= blockSize
;
66 p
->bufferBase
= (Byte
*)alloc
->Alloc(alloc
, (size_t)blockSize
);
68 return (p
->bufferBase
!= 0);
71 Byte
*MatchFinder_GetPointerToCurrentPos(CMatchFinder
*p
) { return p
->buffer
; }
72 Byte
MatchFinder_GetIndexByte(CMatchFinder
*p
, Int32 index
) { return p
->buffer
[index
]; }
74 UInt32
MatchFinder_GetNumAvailableBytes(CMatchFinder
*p
) { return p
->streamPos
- p
->pos
; }
76 void MatchFinder_ReduceOffsets(CMatchFinder
*p
, UInt32 subValue
)
78 p
->posLimit
-= subValue
;
80 p
->streamPos
-= subValue
;
83 static void MatchFinder_ReadBlock(CMatchFinder
*p
)
85 if (p
->streamEndWasReached
|| p
->result
!= SZ_OK
)
89 Byte
*dest
= p
->buffer
+ (p
->streamPos
- p
->pos
);
90 size_t size
= (p
->bufferBase
+ p
->blockSize
- dest
);
93 p
->result
= p
->stream
->Read(p
->stream
, dest
, &size
);
94 if (p
->result
!= SZ_OK
)
98 p
->streamEndWasReached
= 1;
101 p
->streamPos
+= (UInt32
)size
;
102 if (p
->streamPos
- p
->pos
> p
->keepSizeAfter
)
107 void MatchFinder_MoveBlock(CMatchFinder
*p
)
109 memmove(p
->bufferBase
,
110 p
->buffer
- p
->keepSizeBefore
,
111 (size_t)(p
->streamPos
- p
->pos
+ p
->keepSizeBefore
));
112 p
->buffer
= p
->bufferBase
+ p
->keepSizeBefore
;
115 int MatchFinder_NeedMove(CMatchFinder
*p
)
117 /* if (p->streamEndWasReached) return 0; */
118 return ((size_t)(p
->bufferBase
+ p
->blockSize
- p
->buffer
) <= p
->keepSizeAfter
);
121 void MatchFinder_ReadIfRequired(CMatchFinder
*p
)
123 if (p
->streamEndWasReached
)
125 if (p
->keepSizeAfter
>= p
->streamPos
- p
->pos
)
126 MatchFinder_ReadBlock(p
);
129 static void MatchFinder_CheckAndMoveAndRead(CMatchFinder
*p
)
131 if (MatchFinder_NeedMove(p
))
132 MatchFinder_MoveBlock(p
);
133 MatchFinder_ReadBlock(p
);
136 static void MatchFinder_SetDefaultSettings(CMatchFinder
*p
)
141 /* p->skipModeBits = 0; */
146 #define kCrcPoly 0xEDB88320
148 void MatchFinder_Construct(CMatchFinder
*p
)
154 MatchFinder_SetDefaultSettings(p
);
156 for (i
= 0; i
< 256; i
++)
160 for (j
= 0; j
< 8; j
++)
161 r
= (r
>> 1) ^ (kCrcPoly
& ~((r
& 1) - 1));
166 static void MatchFinder_FreeThisClassMemory(CMatchFinder
*p
, ISzAlloc
*alloc
)
168 alloc
->Free(alloc
, p
->hash
);
172 void MatchFinder_Free(CMatchFinder
*p
, ISzAlloc
*alloc
)
174 MatchFinder_FreeThisClassMemory(p
, alloc
);
175 LzInWindow_Free(p
, alloc
);
178 static CLzRef
* AllocRefs(UInt32 num
, ISzAlloc
*alloc
)
180 size_t sizeInBytes
= (size_t)num
* sizeof(CLzRef
);
181 if (sizeInBytes
/ sizeof(CLzRef
) != num
)
183 return (CLzRef
*)alloc
->Alloc(alloc
, sizeInBytes
);
186 int MatchFinder_Create(CMatchFinder
*p
, UInt32 historySize
,
187 UInt32 keepAddBufferBefore
, UInt32 matchMaxLen
, UInt32 keepAddBufferAfter
,
191 if (historySize
> kMaxHistorySize
)
193 MatchFinder_Free(p
, alloc
);
196 sizeReserv
= historySize
>> 1;
197 if (historySize
> ((UInt32
)2 << 30))
198 sizeReserv
= historySize
>> 2;
199 sizeReserv
+= (keepAddBufferBefore
+ matchMaxLen
+ keepAddBufferAfter
) / 2 + (1 << 19);
201 p
->keepSizeBefore
= historySize
+ keepAddBufferBefore
+ 1;
202 p
->keepSizeAfter
= matchMaxLen
+ keepAddBufferAfter
;
203 /* we need one additional byte, since we use MoveBlock after pos++ and before dictionary using */
204 if (LzInWindow_Create(p
, sizeReserv
, alloc
))
206 UInt32 newCyclicBufferSize
= (historySize
/* >> p->skipModeBits */) + 1;
208 p
->matchMaxLen
= matchMaxLen
;
210 p
->fixedHashSize
= 0;
211 if (p
->numHashBytes
== 2)
215 hs
= historySize
- 1;
221 /* hs >>= p->skipModeBits; */
222 hs
|= 0xFFFF; /* don't change it! It's required for Deflate */
225 if (p
->numHashBytes
== 3)
233 if (p
->numHashBytes
> 2) p
->fixedHashSize
+= kHash2Size
;
234 if (p
->numHashBytes
> 3) p
->fixedHashSize
+= kHash3Size
;
235 if (p
->numHashBytes
> 4) p
->fixedHashSize
+= kHash4Size
;
236 hs
+= p
->fixedHashSize
;
240 UInt32 prevSize
= p
->hashSizeSum
+ p
->numSons
;
242 p
->historySize
= historySize
;
244 p
->cyclicBufferSize
= newCyclicBufferSize
;
245 p
->numSons
= (p
->btMode
? newCyclicBufferSize
* 2 : newCyclicBufferSize
);
246 newSize
= p
->hashSizeSum
+ p
->numSons
;
247 if (p
->hash
!= 0 && prevSize
== newSize
)
249 MatchFinder_FreeThisClassMemory(p
, alloc
);
250 p
->hash
= AllocRefs(newSize
, alloc
);
253 p
->son
= p
->hash
+ p
->hashSizeSum
;
258 MatchFinder_Free(p
, alloc
);
262 static void MatchFinder_SetLimits(CMatchFinder
*p
)
264 UInt32 limit
= kMaxValForNormalize
- p
->pos
;
265 UInt32 limit2
= p
->cyclicBufferSize
- p
->cyclicBufferPos
;
268 limit2
= p
->streamPos
- p
->pos
;
269 if (limit2
<= p
->keepSizeAfter
)
275 limit2
-= p
->keepSizeAfter
;
279 UInt32 lenLimit
= p
->streamPos
- p
->pos
;
280 if (lenLimit
> p
->matchMaxLen
)
281 lenLimit
= p
->matchMaxLen
;
282 p
->lenLimit
= lenLimit
;
284 p
->posLimit
= p
->pos
+ limit
;
287 void MatchFinder_Init(CMatchFinder
*p
)
290 for(i
= 0; i
< p
->hashSizeSum
; i
++)
291 p
->hash
[i
] = kEmptyHashValue
;
292 p
->cyclicBufferPos
= 0;
293 p
->buffer
= p
->bufferBase
;
294 p
->pos
= p
->streamPos
= p
->cyclicBufferSize
;
296 p
->streamEndWasReached
= 0;
297 MatchFinder_ReadBlock(p
);
298 MatchFinder_SetLimits(p
);
301 static UInt32
MatchFinder_GetSubValue(CMatchFinder
*p
)
303 return (p
->pos
- p
->historySize
- 1) & kNormalizeMask
;
306 void MatchFinder_Normalize3(UInt32 subValue
, CLzRef
*items
, UInt32 numItems
)
309 for (i
= 0; i
< numItems
; i
++)
311 UInt32 value
= items
[i
];
312 if (value
<= subValue
)
313 value
= kEmptyHashValue
;
320 static void MatchFinder_Normalize(CMatchFinder
*p
)
322 UInt32 subValue
= MatchFinder_GetSubValue(p
);
323 MatchFinder_Normalize3(subValue
, p
->hash
, p
->hashSizeSum
+ p
->numSons
);
324 MatchFinder_ReduceOffsets(p
, subValue
);
327 static void MatchFinder_CheckLimits(CMatchFinder
*p
)
329 if (p
->pos
== kMaxValForNormalize
)
330 MatchFinder_Normalize(p
);
331 if (!p
->streamEndWasReached
&& p
->keepSizeAfter
== p
->streamPos
- p
->pos
)
332 MatchFinder_CheckAndMoveAndRead(p
);
333 if (p
->cyclicBufferPos
== p
->cyclicBufferSize
)
334 p
->cyclicBufferPos
= 0;
335 MatchFinder_SetLimits(p
);
338 static UInt32
* Hc_GetMatchesSpec(UInt32 lenLimit
, UInt32 curMatch
, UInt32 pos
, const Byte
*cur
, CLzRef
*son
,
339 UInt32 _cyclicBufferPos
, UInt32 _cyclicBufferSize
, UInt32 cutValue
,
340 UInt32
*distances
, UInt32 maxLen
)
342 son
[_cyclicBufferPos
] = curMatch
;
345 UInt32 delta
= pos
- curMatch
;
346 if (cutValue
-- == 0 || delta
>= _cyclicBufferSize
)
349 const Byte
*pb
= cur
- delta
;
350 curMatch
= son
[_cyclicBufferPos
- delta
+ ((delta
> _cyclicBufferPos
) ? _cyclicBufferSize
: 0)];
351 if (pb
[maxLen
] == cur
[maxLen
] && *pb
== *cur
)
354 while(++len
!= lenLimit
)
355 if (pb
[len
] != cur
[len
])
359 *distances
++ = maxLen
= len
;
360 *distances
++ = delta
- 1;
369 UInt32
* GetMatchesSpec1(UInt32 lenLimit
, UInt32 curMatch
, UInt32 pos
, const Byte
*cur
, CLzRef
*son
,
370 UInt32 _cyclicBufferPos
, UInt32 _cyclicBufferSize
, UInt32 cutValue
,
371 UInt32
*distances
, UInt32 maxLen
)
373 CLzRef
*ptr0
= son
+ (_cyclicBufferPos
<< 1) + 1;
374 CLzRef
*ptr1
= son
+ (_cyclicBufferPos
<< 1);
375 UInt32 len0
= 0, len1
= 0;
378 UInt32 delta
= pos
- curMatch
;
379 if (cutValue
-- == 0 || delta
>= _cyclicBufferSize
)
381 *ptr0
= *ptr1
= kEmptyHashValue
;
385 CLzRef
*pair
= son
+ ((_cyclicBufferPos
- delta
+ ((delta
> _cyclicBufferPos
) ? _cyclicBufferSize
: 0)) << 1);
386 const Byte
*pb
= cur
- delta
;
387 UInt32 len
= (len0
< len1
? len0
: len1
);
388 if (pb
[len
] == cur
[len
])
390 if (++len
!= lenLimit
&& pb
[len
] == cur
[len
])
391 while(++len
!= lenLimit
)
392 if (pb
[len
] != cur
[len
])
396 *distances
++ = maxLen
= len
;
397 *distances
++ = delta
- 1;
406 if (pb
[len
] < cur
[len
])
424 static void SkipMatchesSpec(UInt32 lenLimit
, UInt32 curMatch
, UInt32 pos
, const Byte
*cur
, CLzRef
*son
,
425 UInt32 _cyclicBufferPos
, UInt32 _cyclicBufferSize
, UInt32 cutValue
)
427 CLzRef
*ptr0
= son
+ (_cyclicBufferPos
<< 1) + 1;
428 CLzRef
*ptr1
= son
+ (_cyclicBufferPos
<< 1);
429 UInt32 len0
= 0, len1
= 0;
432 UInt32 delta
= pos
- curMatch
;
433 if (cutValue
-- == 0 || delta
>= _cyclicBufferSize
)
435 *ptr0
= *ptr1
= kEmptyHashValue
;
439 CLzRef
*pair
= son
+ ((_cyclicBufferPos
- delta
+ ((delta
> _cyclicBufferPos
) ? _cyclicBufferSize
: 0)) << 1);
440 const Byte
*pb
= cur
- delta
;
441 UInt32 len
= (len0
< len1
? len0
: len1
);
442 if (pb
[len
] == cur
[len
])
444 while(++len
!= lenLimit
)
445 if (pb
[len
] != cur
[len
])
456 if (pb
[len
] < cur
[len
])
475 ++p->cyclicBufferPos; \
477 if (++p->pos == p->posLimit) MatchFinder_CheckLimits(p);
479 #define MOVE_POS_RET MOVE_POS return offset;
481 static void MatchFinder_MovePos(CMatchFinder
*p
) { MOVE_POS
; }
483 #define GET_MATCHES_HEADER2(minLen, ret_op) \
484 UInt32 lenLimit; UInt32 hashValue; const Byte *cur; UInt32 curMatch; \
485 lenLimit = p->lenLimit; { if (lenLimit < minLen) { MatchFinder_MovePos(p); ret_op; }} \
488 #define GET_MATCHES_HEADER(minLen) GET_MATCHES_HEADER2(minLen, return 0)
489 #define SKIP_HEADER(minLen) GET_MATCHES_HEADER2(minLen, continue)
491 #define MF_PARAMS(p) p->pos, p->buffer, p->son, p->cyclicBufferPos, p->cyclicBufferSize, p->cutValue
493 #define GET_MATCHES_FOOTER(offset, maxLen) \
494 offset = (UInt32)(GetMatchesSpec1(lenLimit, curMatch, MF_PARAMS(p), \
495 distances + offset, maxLen) - distances); MOVE_POS_RET;
497 #define SKIP_FOOTER \
498 SkipMatchesSpec(lenLimit, curMatch, MF_PARAMS(p)); MOVE_POS;
500 static UInt32
Bt2_MatchFinder_GetMatches(CMatchFinder
*p
, UInt32
*distances
)
503 GET_MATCHES_HEADER(2)
505 curMatch
= p
->hash
[hashValue
];
506 p
->hash
[hashValue
] = p
->pos
;
508 GET_MATCHES_FOOTER(offset
, 1)
511 UInt32
Bt3Zip_MatchFinder_GetMatches(CMatchFinder
*p
, UInt32
*distances
)
514 GET_MATCHES_HEADER(3)
516 curMatch
= p
->hash
[hashValue
];
517 p
->hash
[hashValue
] = p
->pos
;
519 GET_MATCHES_FOOTER(offset
, 2)
522 static UInt32
Bt3_MatchFinder_GetMatches(CMatchFinder
*p
, UInt32
*distances
)
524 UInt32 hash2Value
, delta2
, maxLen
, offset
;
525 GET_MATCHES_HEADER(3)
529 delta2
= p
->pos
- p
->hash
[hash2Value
];
530 curMatch
= p
->hash
[kFix3HashSize
+ hashValue
];
532 p
->hash
[hash2Value
] =
533 p
->hash
[kFix3HashSize
+ hashValue
] = p
->pos
;
538 if (delta2
< p
->cyclicBufferSize
&& *(cur
- delta2
) == *cur
)
540 for (; maxLen
!= lenLimit
; maxLen
++)
541 if (cur
[(ptrdiff_t)maxLen
- delta2
] != cur
[maxLen
])
543 distances
[0] = maxLen
;
544 distances
[1] = delta2
- 1;
546 if (maxLen
== lenLimit
)
548 SkipMatchesSpec(lenLimit
, curMatch
, MF_PARAMS(p
));
552 GET_MATCHES_FOOTER(offset
, maxLen
)
555 static UInt32
Bt4_MatchFinder_GetMatches(CMatchFinder
*p
, UInt32
*distances
)
557 UInt32 hash2Value
, hash3Value
, delta2
, delta3
, maxLen
, offset
;
558 GET_MATCHES_HEADER(4)
562 delta2
= p
->pos
- p
->hash
[ hash2Value
];
563 delta3
= p
->pos
- p
->hash
[kFix3HashSize
+ hash3Value
];
564 curMatch
= p
->hash
[kFix4HashSize
+ hashValue
];
566 p
->hash
[ hash2Value
] =
567 p
->hash
[kFix3HashSize
+ hash3Value
] =
568 p
->hash
[kFix4HashSize
+ hashValue
] = p
->pos
;
572 if (delta2
< p
->cyclicBufferSize
&& *(cur
- delta2
) == *cur
)
574 distances
[0] = maxLen
= 2;
575 distances
[1] = delta2
- 1;
578 if (delta2
!= delta3
&& delta3
< p
->cyclicBufferSize
&& *(cur
- delta3
) == *cur
)
581 distances
[offset
+ 1] = delta3
- 1;
587 for (; maxLen
!= lenLimit
; maxLen
++)
588 if (cur
[(ptrdiff_t)maxLen
- delta2
] != cur
[maxLen
])
590 distances
[offset
- 2] = maxLen
;
591 if (maxLen
== lenLimit
)
593 SkipMatchesSpec(lenLimit
, curMatch
, MF_PARAMS(p
));
599 GET_MATCHES_FOOTER(offset
, maxLen
)
602 static UInt32
Hc4_MatchFinder_GetMatches(CMatchFinder
*p
, UInt32
*distances
)
604 UInt32 hash2Value
, hash3Value
, delta2
, delta3
, maxLen
, offset
;
605 GET_MATCHES_HEADER(4)
609 delta2
= p
->pos
- p
->hash
[ hash2Value
];
610 delta3
= p
->pos
- p
->hash
[kFix3HashSize
+ hash3Value
];
611 curMatch
= p
->hash
[kFix4HashSize
+ hashValue
];
613 p
->hash
[ hash2Value
] =
614 p
->hash
[kFix3HashSize
+ hash3Value
] =
615 p
->hash
[kFix4HashSize
+ hashValue
] = p
->pos
;
619 if (delta2
< p
->cyclicBufferSize
&& *(cur
- delta2
) == *cur
)
621 distances
[0] = maxLen
= 2;
622 distances
[1] = delta2
- 1;
625 if (delta2
!= delta3
&& delta3
< p
->cyclicBufferSize
&& *(cur
- delta3
) == *cur
)
628 distances
[offset
+ 1] = delta3
- 1;
634 for (; maxLen
!= lenLimit
; maxLen
++)
635 if (cur
[(ptrdiff_t)maxLen
- delta2
] != cur
[maxLen
])
637 distances
[offset
- 2] = maxLen
;
638 if (maxLen
== lenLimit
)
640 p
->son
[p
->cyclicBufferPos
] = curMatch
;
646 offset
= (UInt32
)(Hc_GetMatchesSpec(lenLimit
, curMatch
, MF_PARAMS(p
),
647 distances
+ offset
, maxLen
) - (distances
));
651 UInt32
Hc3Zip_MatchFinder_GetMatches(CMatchFinder
*p
, UInt32
*distances
)
654 GET_MATCHES_HEADER(3)
656 curMatch
= p
->hash
[hashValue
];
657 p
->hash
[hashValue
] = p
->pos
;
658 offset
= (UInt32
)(Hc_GetMatchesSpec(lenLimit
, curMatch
, MF_PARAMS(p
),
659 distances
, 2) - (distances
));
663 static void Bt2_MatchFinder_Skip(CMatchFinder
*p
, UInt32 num
)
669 curMatch
= p
->hash
[hashValue
];
670 p
->hash
[hashValue
] = p
->pos
;
676 void Bt3Zip_MatchFinder_Skip(CMatchFinder
*p
, UInt32 num
)
682 curMatch
= p
->hash
[hashValue
];
683 p
->hash
[hashValue
] = p
->pos
;
689 static void Bt3_MatchFinder_Skip(CMatchFinder
*p
, UInt32 num
)
696 curMatch
= p
->hash
[kFix3HashSize
+ hashValue
];
697 p
->hash
[hash2Value
] =
698 p
->hash
[kFix3HashSize
+ hashValue
] = p
->pos
;
704 static void Bt4_MatchFinder_Skip(CMatchFinder
*p
, UInt32 num
)
708 UInt32 hash2Value
, hash3Value
;
711 curMatch
= p
->hash
[kFix4HashSize
+ hashValue
];
712 p
->hash
[ hash2Value
] =
713 p
->hash
[kFix3HashSize
+ hash3Value
] = p
->pos
;
714 p
->hash
[kFix4HashSize
+ hashValue
] = p
->pos
;
720 static void Hc4_MatchFinder_Skip(CMatchFinder
*p
, UInt32 num
)
724 UInt32 hash2Value
, hash3Value
;
727 curMatch
= p
->hash
[kFix4HashSize
+ hashValue
];
728 p
->hash
[ hash2Value
] =
729 p
->hash
[kFix3HashSize
+ hash3Value
] =
730 p
->hash
[kFix4HashSize
+ hashValue
] = p
->pos
;
731 p
->son
[p
->cyclicBufferPos
] = curMatch
;
737 void Hc3Zip_MatchFinder_Skip(CMatchFinder
*p
, UInt32 num
)
743 curMatch
= p
->hash
[hashValue
];
744 p
->hash
[hashValue
] = p
->pos
;
745 p
->son
[p
->cyclicBufferPos
] = curMatch
;
751 void MatchFinder_CreateVTable(CMatchFinder
*p
, IMatchFinder
*vTable
)
753 vTable
->Init
= (Mf_Init_Func
)MatchFinder_Init
;
754 vTable
->GetIndexByte
= (Mf_GetIndexByte_Func
)MatchFinder_GetIndexByte
;
755 vTable
->GetNumAvailableBytes
= (Mf_GetNumAvailableBytes_Func
)MatchFinder_GetNumAvailableBytes
;
756 vTable
->GetPointerToCurrentPos
= (Mf_GetPointerToCurrentPos_Func
)MatchFinder_GetPointerToCurrentPos
;
759 vTable
->GetMatches
= (Mf_GetMatches_Func
)Hc4_MatchFinder_GetMatches
;
760 vTable
->Skip
= (Mf_Skip_Func
)Hc4_MatchFinder_Skip
;
762 else if (p
->numHashBytes
== 2)
764 vTable
->GetMatches
= (Mf_GetMatches_Func
)Bt2_MatchFinder_GetMatches
;
765 vTable
->Skip
= (Mf_Skip_Func
)Bt2_MatchFinder_Skip
;
767 else if (p
->numHashBytes
== 3)
769 vTable
->GetMatches
= (Mf_GetMatches_Func
)Bt3_MatchFinder_GetMatches
;
770 vTable
->Skip
= (Mf_Skip_Func
)Bt3_MatchFinder_Skip
;
774 vTable
->GetMatches
= (Mf_GetMatches_Func
)Bt4_MatchFinder_GetMatches
;
775 vTable
->Skip
= (Mf_Skip_Func
)Bt4_MatchFinder_Skip
;