Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
I
imagej-elphel
Project
Project
Details
Activity
Releases
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
3
Issues
3
List
Board
Labels
Milestones
Wiki
Wiki
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Commits
Issue Boards
Open sidebar
Elphel
imagej-elphel
Commits
dfe9ae9d
Commit
dfe9ae9d
authored
May 07, 2017
by
Andrey Filippov
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
selecting no-conflict clusters
parent
da537ae9
Changes
1
Show whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
361 additions
and
2 deletions
+361
-2
TileSurface.java
src/main/java/TileSurface.java
+361
-2
No files found.
src/main/java/TileSurface.java
View file @
dfe9ae9d
...
...
@@ -2867,6 +2867,357 @@ public class TileSurface {
sdfa_instance
.
showArrays
(
img_clust
,
imageTilesX
,
imageTilesY
,
true
,
title
,
titles
);
}
public
boolean
[][]
growClusterOnce
(
// used from inside threads
final
boolean
[][]
cluster_local_pure
,
final
int
[]
window
,
final
int
[]
img_indices
,
final
int
[]
surf_indices
,
final
int
debugLevel
,
final
int
dbg_X
,
final
int
dbg_Y
)
{
int
num_tiles
=
img_indices
.
length
;
int
width
=
window
[
2
];
int
height
=
window
[
3
];
boolean
[][]
grown_cluster
=
new
boolean
[
num_tiles
][];
for
(
int
neTile
=
0
;
neTile
<
num_tiles
;
neTile
++){
if
(
cluster_local_pure
[
neTile
]
!=
null
){
grown_cluster
[
neTile
]
=
cluster_local_pure
[
neTile
].
clone
();
}
}
int
dbg_ntiles1
=
0
;
if
(
debugLevel
>
0
)
{
for
(
int
neTile
=
0
;
neTile
<
num_tiles
;
neTile
++){
if
(
grown_cluster
[
neTile
]
!=
null
){
for
(
int
nl
=
0
;
nl
<
grown_cluster
[
neTile
].
length
;
nl
++){
if
(
grown_cluster
[
neTile
][
nl
])
dbg_ntiles1
++;
}
}
}
System
.
out
.
println
(
"growClusterOnce(): number of tiles = "
+
dbg_ntiles1
);
}
final
TileNeibs
tnWindow
=
new
TileNeibs
(
width
,
height
);
for
(
int
neTile0
=
0
;
neTile0
<
num_tiles
;
neTile0
++)
if
(
cluster_local_pure
[
neTile0
]
!=
null
){
for
(
int
nl0
=
0
;
nl0
<
cluster_local_pure
[
neTile0
].
length
;
nl0
++){
if
(
cluster_local_pure
[
neTile0
][
nl0
]){
// source should be single-layer, but ...
int
nSurfTile0
=
surf_indices
[
neTile0
];
int
[]
neibs
=
tileData
[
nSurfTile0
][
nl0
].
getNeighbors
();
for
(
int
dir
=
0
;
dir
<
tnWindow
.
dirs
;
dir
++)
{
int
nl1
=
neibs
[
dir
];
if
(
nl1
<
0
){
if
(
debugLevel
>-
1
)
{
System
.
out
.
println
(
"growClusterOnce(): Expected 8 neighbors for tile nSurfTile0="
+
nSurfTile0
+
" neibs["
+
dir
+
"] = "
+
nl1
);
}
}
else
{
int
neTile1
=
tnWindow
.
getNeibIndex
(
neTile0
,
dir
);
if
(
neTile1
>=
0
)
{
if
(
((
cluster_local_pure
[
neTile1
]
==
null
)
||
!
cluster_local_pure
[
neTile1
][
nl1
])
&&
((
grown_cluster
[
neTile1
]
==
null
)
||
!
grown_cluster
[
neTile1
][
nl1
]))
{
if
(
grown_cluster
[
neTile1
]
==
null
)
{
int
nSurfTile1
=
surf_indices
[
neTile1
];
// should never be null as it is connected from nSurfTile0
grown_cluster
[
neTile1
]
=
new
boolean
[
tileData
[
nSurfTile1
].
length
];
}
grown_cluster
[
neTile1
][
nl1
]
=
true
;
}
}
}
}
}
}
}
if
(
debugLevel
>
0
)
{
int
dbg_ntiles2
=
0
;
for
(
int
neTile
=
0
;
neTile
<
num_tiles
;
neTile
++){
if
(
grown_cluster
[
neTile
]
!=
null
){
for
(
int
nl
=
0
;
nl
<
grown_cluster
[
neTile
].
length
;
nl
++){
if
(
grown_cluster
[
neTile
][
nl
])
dbg_ntiles2
++;
}
}
}
System
.
out
.
println
(
"growClusterOnce(): new number of tiles = "
+
dbg_ntiles2
+
" (was "
+
dbg_ntiles1
+
")"
);
}
return
grown_cluster
;
}
public
int
[][]
mergeNoConflict
(
final
int
[][]
matchedGrown
,
final
int
[][]
clusters_grown
,
final
int
[][]
clusters_pure
,
final
int
debugLevel
,
final
int
dbg_X
,
final
int
dbg_Y
)
{
if
(
debugLevel
>-
1
){
System
.
out
.
println
(
"mergeNoConflict(): groups: "
+
matchedGrown
.
length
);
}
final
int
num_grown
=
matchedGrown
.
length
;
final
int
[][][]
pre_merged_subs
=
new
int
[
num_grown
][][];
final
Thread
[]
threads
=
ImageDtt
.
newThreadArray
(
threadsMax
);
final
AtomicInteger
ai
=
new
AtomicInteger
(
0
);
for
(
int
ithread
=
0
;
ithread
<
threads
.
length
;
ithread
++)
{
threads
[
ithread
]
=
new
Thread
()
{
public
void
run
()
{
for
(
int
iGClust
=
ai
.
getAndIncrement
();
iGClust
<
num_grown
;
iGClust
=
ai
.
getAndIncrement
())
{
int
num_subs
=
matchedGrown
[
iGClust
].
length
;
if
(
num_subs
<
2
){
// nothing to split
pre_merged_subs
[
iGClust
]
=
new
int
[
num_subs
][];
if
(
num_subs
>
0
)
{
// should always be so?
pre_merged_subs
[
iGClust
][
0
]
=
matchedGrown
[
iGClust
].
clone
();
}
}
else
{
//>= 2 clusters
int
nGClust
=
iGClust
+
1
;
int
dl
=
((
debugLevel
>
-
1
)
&&
(
num_subs
>
3
))
?
3
:
0
;
if
(
dl
>
0
){
System
.
out
.
println
(
"mergeNoConflict(): nGClust: "
+
nGClust
+
" num_subs= "
+
num_subs
);
}
// create window for the grown cluster, it will include all subs
int
[]
window
=
getClusterBBox
(
nGClust
,
// 1-based
0
,
// border,
clusters_grown
);
int
[][]
tile_indices
=
getClusterBBoxIndices
(
window
,
// final int [] window,
0
);
// border); // final int border) // maybe 0, actual value just saves time
int
[]
img_tile_indices
=
tile_indices
[
0
];
int
[]
surf_tile_indices
=
tile_indices
[
1
];
int
num_tiles
=
img_tile_indices
.
length
;
int
[]
clust_sizes
=
new
int
[
num_subs
];
boolean
[][][]
subs_pure
=
new
boolean
[
num_subs
][][];
boolean
[][][]
subs_grown
=
new
boolean
[
num_subs
][][];
for
(
int
nSub
=
0
;
nSub
<
num_subs
;
nSub
++){
subs_pure
[
nSub
]
=
extractCluster
(
matchedGrown
[
iGClust
][
nSub
],
// final int numToSplit, // 1-based
0
,
// border, // final int border,
clusters_pure
,
// final int [][] clusters,
window
,
// final int [] window,
img_tile_indices
,
// final int [] bbox_indices,
debugLevel
,
// final int debugLevel,
dbg_X
,
// final int dbg_X,
dbg_Y
);
// final int dbg_Y)
subs_grown
[
nSub
]
=
growClusterOnce
(
// used from inside threads
subs_pure
[
nSub
],
// final boolean [][] cluster_local_pure,
window
,
// final int [] window,
img_tile_indices
,
// final int [] bbox_indices,
surf_tile_indices
,
// final int [] surf_indices,
0
,
// dl, // debugLevel, // final int debugLevel,
dbg_X
,
// final int dbg_X,
dbg_Y
);
// final int dbg_Y)
}
// now build sets of clusters
// a) connected to current (and having higher number)
// b) conflicting with current (and having higher number)
ArrayList
<
HashSet
<
Integer
>>
touching_list
=
new
ArrayList
<
HashSet
<
Integer
>>();
ArrayList
<
HashSet
<
Integer
>>
conflict_list
=
new
ArrayList
<
HashSet
<
Integer
>>();
for
(
int
nSub
=
0
;
nSub
<
num_subs
;
nSub
++){
touching_list
.
add
(
new
HashSet
<
Integer
>());
conflict_list
.
add
(
new
HashSet
<
Integer
>());
}
for
(
int
neTile
=
0
;
neTile
<
num_tiles
;
neTile
++){
for
(
int
nSub
=
0
;
nSub
<
num_subs
;
nSub
++){
if
(
subs_grown
[
nSub
][
neTile
]
!=
null
){
for
(
int
nl1
=
0
;
nl1
<
subs_grown
[
nSub
][
neTile
].
length
;
nl1
++){
if
(
subs_grown
[
nSub
][
neTile
][
nl1
]){
clust_sizes
[
nSub
]
++;
for
(
int
nSub2
=
nSub
+
1
;
nSub2
<
num_subs
;
nSub2
++)
if
(
subs_grown
[
nSub2
][
neTile
]
!=
null
){
for
(
int
nl2
=
0
;
nl2
<
subs_grown
[
nSub2
][
neTile
].
length
;
nl2
++){
//*
if
(
subs_grown
[
nSub2
][
neTile
][
nl2
]){
if
(
nl1
==
nl2
)
{
touching_list
.
get
(
nSub
).
add
(
new
Integer
(
nSub2
));
touching_list
.
get
(
nSub2
).
add
(
new
Integer
(
nSub
));
// both ways
}
else
{
conflict_list
.
get
(
nSub
).
add
(
new
Integer
(
nSub2
));
conflict_list
.
get
(
nSub2
).
add
(
new
Integer
(
nSub
));
// both ways
}
}
}
}
}
}
}
}
}
ArrayList
<
HashSet
<
Integer
>>
groups_list
=
new
ArrayList
<
HashSet
<
Integer
>>();
HashSet
<
Integer
>
clusters_left
=
new
HashSet
<
Integer
>();
for
(
int
nSub
=
0
;
nSub
<
num_subs
;
nSub
++){
clusters_left
.
add
(
new
Integer
(
nSub
));
}
while
(!
clusters_left
.
isEmpty
()){
HashSet
<
Integer
>
new_group
=
new
HashSet
<
Integer
>();
HashSet
<
Integer
>
candidates
=
new
HashSet
<
Integer
>();
HashSet
<
Integer
>
conflicts
=
new
HashSet
<
Integer
>();
// HashSet<Integer> touching = new HashSet<Integer>();
// start with the largest of the remaining clusters
Integer
best_sub
=
-
1
;
for
(
Integer
sc:
clusters_left
){
if
((
best_sub
<
0
)
||
(
clust_sizes
[
best_sub
]
<
clust_sizes
[
sc
])){
best_sub
=
sc
;
}
}
if
(
!
clusters_left
.
remove
(
best_sub
))
{
System
.
out
.
println
(
"mergeNoConflict() bug: can not remove"
+
best_sub
);
}
new_group
.
add
(
best_sub
);
conflicts
.
addAll
(
conflict_list
.
get
(
best_sub
));
// keep track of all accumulated conflicts
candidates
.
addAll
(
touching_list
.
get
(
best_sub
));
// add all clusters that are touching the current selection
candidates
.
removeAll
(
conflicts
);
// remove all conflicts
while
(!
candidates
.
isEmpty
())
{
// add more clusters if possible
// Find the largest one
best_sub
=
-
1
;
for
(
Integer
sc:
candidates
){
if
((
best_sub
<
0
)
||
(
clust_sizes
[
best_sub
]
<
clust_sizes
[
sc
])){
best_sub
=
sc
;
}
}
// add that new cluster
clusters_left
.
remove
(
best_sub
);
new_group
.
add
(
best_sub
);
conflicts
.
addAll
(
conflict_list
.
get
(
best_sub
));
// keep track of all accumulated conflicts
candidates
.
addAll
(
touching_list
.
get
(
best_sub
));
// add all clusters that are touching the current selection
candidates
.
removeAll
(
conflicts
);
// remove all conflicts
candidates
.
removeAll
(
new_group
);
// remove all what is already included
}
groups_list
.
add
(
new_group
);
}
pre_merged_subs
[
iGClust
]
=
new
int
[
groups_list
.
size
()][];
for
(
int
ng
=
0
;
ng
<
groups_list
.
size
();
ng
++)
{
pre_merged_subs
[
iGClust
][
ng
]
=
new
int
[
groups_list
.
get
(
ng
).
size
()];
int
nc
=
0
;
for
(
Integer
cl:
groups_list
.
get
(
ng
)){
pre_merged_subs
[
iGClust
][
ng
][
nc
++]
=
cl
;
}
}
}
}
}
};
}
ImageDtt
.
startAndJoin
(
threads
);
// "flatten" pre_merged_subs
int
num_new_groups
=
0
;
for
(
int
ng
=
0
;
ng
<
pre_merged_subs
.
length
;
ng
++)
{
num_new_groups
+=
pre_merged_subs
[
ng
].
length
;
}
if
(
debugLevel
>-
1
){
System
.
out
.
println
(
"mergeNoConflict(): groups: "
+
num_new_groups
+
" (was : "
+
matchedGrown
.
length
+
")"
);
}
final
int
[][]
merged_subs
=
new
int
[
num_new_groups
][];
int
indx
=
0
;
for
(
int
ng
=
0
;
ng
<
pre_merged_subs
.
length
;
ng
++)
{
for
(
int
nc
=
0
;
nc
<
pre_merged_subs
[
ng
].
length
;
nc
++){
merged_subs
[
indx
++]
=
pre_merged_subs
[
ng
][
nc
];
}
}
if
(
debugLevel
>
-
1
)
{
for
(
int
i
=
0
;
i
<
num_new_groups
;
i
++){
System
.
out
.
print
(
"mergeNoConflict(): "
+
(
i
+
1
)+
" ["
);
for
(
int
j
=
0
;
j
<
merged_subs
[
i
].
length
;
j
++)
{
if
(
j
>
0
)
System
.
out
.
print
(
", "
);
System
.
out
.
print
(
merged_subs
[
i
][
j
]);
}
System
.
out
.
println
(
" ]"
);
}
}
return
merged_subs
;
}
/**
* Grow each of the clusters (encoded as positive cluster numbers per tile per layer) by 1 in each
* of 8 directions. As they now may overlap they are encoded in boolean array [cluster][tile][layer]
* @param clusters_pure cluster numbers (>0) for each tile, each layer
* @param debugLevel
* @param dbg_X
* @param dbg_Y
* @return boolean array [cluster-1][tile][layer]
*/
public
boolean
[][][]
growEachCluster
(
final
int
[][]
clusters_pure
,
final
int
debugLevel
,
final
int
dbg_X
,
final
int
dbg_Y
)
{
final
int
num_clusters
=
getNumClusters
(
clusters_pure
);
final
boolean
[][][]
clusters
=
new
boolean
[
num_clusters
][
clusters_pure
.
length
][];
final
TileNeibs
tnImage
=
new
TileNeibs
(
imageTilesX
,
imageTilesY
);
final
Thread
[]
threads
=
ImageDtt
.
newThreadArray
(
threadsMax
);
final
AtomicInteger
ai
=
new
AtomicInteger
(
0
);
for
(
int
ithread
=
0
;
ithread
<
threads
.
length
;
ithread
++)
{
threads
[
ithread
]
=
new
Thread
()
{
public
void
run
()
{
for
(
int
iClust
=
ai
.
getAndIncrement
();
iClust
<
num_clusters
;
iClust
=
ai
.
getAndIncrement
())
{
int
nClust
=
iClust
+
1
;
// 1-based
for
(
int
nTile0
=
0
;
nTile0
<
clusters_pure
.
length
;
nTile0
++)
if
(
clusters_pure
[
nTile0
]
!=
null
){
for
(
int
nl0
=
0
;
nl0
<
clusters_pure
[
nTile0
].
length
;
nl0
++){
if
(
clusters_pure
[
nTile0
][
nl0
]
==
nClust
){
// source should be single-layer, but ...
if
(
clusters
[
iClust
][
nTile0
]==
null
)
{
clusters
[
iClust
][
nTile0
]
=
new
boolean
[
clusters_pure
[
nTile0
].
length
];
}
clusters
[
iClust
][
nTile0
][
nl0
]
=
true
;
int
nSurfTile0
=
getSurfaceTileIndex
(
nTile0
);
int
[]
neibs
=
tileData
[
nSurfTile0
][
nl0
].
getNeighbors
();
for
(
int
dir
=
0
;
dir
<
tnImage
.
dirs
;
dir
++)
{
int
nl1
=
neibs
[
dir
];
if
(
nl1
<
0
){
if
(
debugLevel
>-
1
)
{
System
.
out
.
println
(
"growEachCluster(): Expected 8 neighbors for tile nSurfTile0="
+
nSurfTile0
+
" neibs["
+
dir
+
"] = "
+
nl1
);
}
int
nTile1
=
tnImage
.
getNeibIndex
(
nTile0
,
dir
);
if
(
nTile1
>=
0
)
{
if
(
((
clusters_pure
[
nTile1
]
==
null
)
||
(
clusters_pure
[
nTile1
][
nl1
]
!=
nClust
))
&&
((
clusters
[
iClust
][
nTile1
]
==
null
)
||
!
clusters
[
iClust
][
nTile1
][
nl1
]))
{
if
(
clusters
[
iClust
][
nTile1
]
==
null
)
{
int
nSurfTile1
=
getSurfaceTileIndex
(
nTile1
);
clusters
[
iClust
][
nTile1
]
=
new
boolean
[
tileData
[
nSurfTile1
].
length
];
}
clusters
[
iClust
][
nTile1
][
nl1
]
=
true
;
}
}
}
}
}
}
}
}
}
};
}
ImageDtt
.
startAndJoin
(
threads
);
return
clusters
;
}
/**
* Match grown/merged clusters to the ones they were made of
* @param clusters_grown per-tile, per layer array of positive merged cluster numbers (0 - empty)
* @param clusters_pure per-tile, per layer array of positive pure cluster numbers (0 - empty)
* @param debugLevel
* @param dbg_X
* @param dbg_Y
* @return array for each of the merged clusters array of pure cluster numbers it is made of
*/
public
int
[][]
matchPureGrown
(
final
int
[][]
clusters_grown
,
final
int
[][]
clusters_pure
,
...
...
@@ -2915,7 +3266,7 @@ public class TileSurface {
System
.
out
.
println
(
" ]"
);
}
}
return
null
;
return
grown_subs
;
}
...
...
@@ -3414,6 +3765,14 @@ public class TileSurface {
dbg_X
,
dbg_Y
);
int
[][]
merged_no_conflict
=
mergeNoConflict
(
grown_sub_clusters
,
clusters_grown
,
clusters_pure
,
debugLevel
,
dbg_X
,
dbg_Y
);
System
.
out
.
println
(
"mergeAndGrow() done"
);
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment