Compare commits
1 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| b143c6607a |
+14
-2
@@ -246,10 +246,22 @@ def score_bitmap_rescored(
|
||||
return np.maximum(0.0, out).astype(np.float32)
|
||||
|
||||
|
||||
_HAS_NP_BITCOUNT = hasattr(np, "bitwise_count")
|
||||
|
||||
|
||||
def popcount_density(spread: np.ndarray) -> np.ndarray:
|
||||
"""Conta bit set per pixel.
|
||||
|
||||
Order:
|
||||
1) Numba JIT parallel (preferito: piu veloce su 1080p, 0.5ms vs 1.6ms)
|
||||
2) numpy.bitwise_count (NumPy 2.0+, SIMD ma single-thread)
|
||||
3) Fallback numpy bit-shift puro
|
||||
"""
|
||||
spread_c = np.ascontiguousarray(spread, dtype=np.uint8)
|
||||
if HAS_NUMBA:
|
||||
return _jit_popcount_density(np.ascontiguousarray(spread, dtype=np.uint8))
|
||||
# Fallback
|
||||
return _jit_popcount_density(spread_c)
|
||||
if _HAS_NP_BITCOUNT:
|
||||
return np.bitwise_count(spread_c).astype(np.float32, copy=False)
|
||||
H, W = spread.shape
|
||||
out = np.zeros((H, W), dtype=np.float32)
|
||||
for b in range(8):
|
||||
|
||||
@@ -293,42 +293,8 @@ class LineShapeMatcher:
|
||||
kh=kh, kw=kw,
|
||||
cx_local=float(cx_local), cy_local=float(cy_local),
|
||||
))
|
||||
self._dedup_variants()
|
||||
return len(self.variants)
|
||||
|
||||
def _dedup_variants(self) -> int:
|
||||
"""Rimuove varianti con feature-set identico (post-quantizzazione).
|
||||
|
||||
Halcon-style: con angle range = (0, 360) e simmetrie del template,
|
||||
molte rotazioni producono lo stesso set quantizzato di feature.
|
||||
Es: quadrato a 0/90/180/270 deg → stesse features (modulo permutazione).
|
||||
Hash su feature ordinate (livello 0, full-res) elimina i duplicati.
|
||||
|
||||
Vantaggio: meno varianti = meno chiamate kernel JIT al top-level
|
||||
senza perdere copertura angolare effettiva. Per template asimmetrici
|
||||
non rimuove nulla.
|
||||
"""
|
||||
seen: dict[bytes, int] = {}
|
||||
kept: list[_Variant] = []
|
||||
removed = 0
|
||||
for var in self.variants:
|
||||
lvl0 = var.levels[0]
|
||||
order = np.lexsort((lvl0.bin, lvl0.dy, lvl0.dx))
|
||||
key = (
|
||||
lvl0.dx[order].tobytes()
|
||||
+ b"|" + lvl0.dy[order].tobytes()
|
||||
+ b"|" + lvl0.bin[order].tobytes()
|
||||
+ b"|" + str(round(var.scale, 4)).encode()
|
||||
)
|
||||
h = key # diretto, senza hash crypto (collision ok solo se identici)
|
||||
if h in seen:
|
||||
removed += 1
|
||||
continue
|
||||
seen[h] = len(kept)
|
||||
kept.append(var)
|
||||
self.variants = kept
|
||||
return removed
|
||||
|
||||
# --- Matching ------------------------------------------------------
|
||||
|
||||
def _response_map(self, gray: np.ndarray) -> np.ndarray:
|
||||
|
||||
Reference in New Issue
Block a user