build_tree.sh
8.64 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
#!/bin/bash
# Copyright 2012-2015 Johns Hopkins University (Author: Daniel Povey).
# Apache 2.0.
# This script builds a tree for use in the 'chain' systems (although the script
# itself is pretty generic and doesn't use any 'chain' binaries). This is just
# like the first stages of a standard system, like 'train_sat.sh', except it
# does 'convert-ali' to convert alignments to a monophone topology just created
# from the 'lang' directory (in case the topology is different from where you
# got the system's alignments from), and it stops after the tree-building and
# model-initialization stage, without re-estimating the Gaussians or training
# the transitions.
# Begin configuration section.
stage=-5
exit_stage=-100 # you can use this to require it to exit at the
# beginning of a specific stage. Not all values are
# supported.
cmd=run.pl
context_opts= # e.g. set this to "--context-width 5 --central-position 2" for quinphone.
cluster_thresh=-1 # for build-tree control final bottom-up clustering of leaves
frame_subsampling_factor=1
alignment_subsampling_factor=
leftmost_questions_truncate=-1 # note: this option is deprecated and has no effect
tree_stats_opts=
cluster_phones_opts=
repeat_frames=false
# End configuration section.
echo "$0 $@" # Print the command line for logging
[ -f path.sh ] && . ./path.sh
. parse_options.sh || exit 1;
if [ $# != 5 ]; then
echo "Usage: $0 <#leaves> <data> <lang> <ali-dir> <exp-dir>"
echo " e.g.: $0 --frame-subsampling-factor 3 \\"
echo " --context-opts '--context-width=2 --central-position=1' \\"
echo " 3500 data/train_si84 data/lang_chain exp/tri3b_ali_si284_sp exp/chain/tree_a_sp"
echo "Main options (for others, see top of script file)"
echo " --cmd (utils/run.pl|utils/queue.pl <queue opts>) # how to run jobs."
echo " --config <config-file> # config containing options"
echo " --stage <stage> # stage to do partial re-run from."
echo " --repeat-frames <true|false> # Only affects alignment conversion at"
echo " # the end. If true, generate an "
echo " # alignment using the frame-subsampled "
echo " # topology that is repeated "
echo " # --frame-subsampling-factor times "
echo " # and interleaved, to be the same "
echo " # length as the original alignment "
echo " # (useful for cross-entropy training "
echo " # of reduced frame rate systems)."
echo " --context-opts <option-string> # Options controlling phonetic context;"
echo " # we suggest '--context-width=2 --central-position=1',"
echo " # which is left bigram."
echo " --frame-subsampling-factor <factor> # Factor (e.g. 3) controlling frame subsampling"
echo " # at the neural net output, so the frame rate at"
echo " # the output is less than at the input."
exit 1;
fi
numleaves=$1
data=$2
lang=$3
alidir=$4
dir=$5
for f in $data/feats.scp $lang/phones.txt $alidir/final.mdl $alidir/ali.1.gz; do
[ ! -f $f ] && echo "train_sat.sh: no such file $f" && exit 1;
done
oov=`cat $lang/oov.int`
nj=`cat $alidir/num_jobs` || exit 1;
silphonelist=`cat $lang/phones/silence.csl`
ciphonelist=`cat $lang/phones/context_indep.csl` || exit 1;
splice_opts=`cat $alidir/splice_opts 2>/dev/null` # frame-splicing options.
cmvn_opts=`cat $alidir/cmvn_opts 2>/dev/null`
delta_opts=`cat $alidir/delta_opts 2>/dev/null`
mkdir -p $dir/log
cp $alidir/splice_opts $dir 2>/dev/null # frame-splicing options.
cp $alidir/cmvn_opts $dir 2>/dev/null # cmn/cmvn option.
cp $alidir/delta_opts $dir 2>/dev/null # delta option.
utils/lang/check_phones_compatible.sh $lang/phones.txt $alidir/phones.txt || exit 1;
cp $lang/phones.txt $dir || exit 1;
echo $nj >$dir/num_jobs
if [ -f $alidir/per_utt ]; then
sdata=$data/split${nj}utt
utils/split_data.sh --per-utt $data $nj
else
sdata=$data/split$nj
utils/split_data.sh $data $nj
fi
# Set up features.
if [ -f $alidir/final.mat ]; then feat_type=lda; else feat_type=delta; fi
echo "$0: feature type is $feat_type"
## Set up speaker-independent features.
case $feat_type in
delta) feats="ark,s,cs:apply-cmvn $cmvn_opts --utt2spk=ark:$sdata/JOB/utt2spk scp:$sdata/JOB/cmvn.scp scp:$sdata/JOB/feats.scp ark:- | add-deltas $delta_opts ark:- ark:- |";;
lda) feats="ark,s,cs:apply-cmvn $cmvn_opts --utt2spk=ark:$sdata/JOB/utt2spk scp:$sdata/JOB/cmvn.scp scp:$sdata/JOB/feats.scp ark:- | splice-feats $splice_opts ark:- ark:- | transform-feats $alidir/final.mat ark:- ark:- |"
cp $alidir/final.mat $dir
cp $alidir/full.mat $dir 2>/dev/null
;;
*) echo "$0: invalid feature type $feat_type" && exit 1;
esac
# Add fMLLR transforms if available
if [ -f $alidir/trans.1 ]; then
echo "$0: Using transforms from $alidir"
feats="$feats transform-feats --utt2spk=ark:$sdata/JOB/utt2spk ark,s,cs:$alidir/trans.JOB ark:- ark:- |"
fi
# Do subsampling of feats, if needed
if [ $frame_subsampling_factor -gt 1 ]; then
feats="$feats subsample-feats --n=$frame_subsampling_factor ark:- ark:- |"
fi
if [ -z $alignment_subsampling_factor ]; then
alignment_subsampling_factor=$frame_subsampling_factor
fi
if [ $stage -le -5 ]; then
echo "$0: Initializing monophone model (for alignment conversion, in case topology changed)"
[ ! -f $lang/phones/sets.int ] && exit 1;
shared_phones_opt="--shared-phones=$lang/phones/sets.int"
# get feature dimension
example_feats="`echo $feats | sed s/JOB/1/g`";
if ! feat_dim=$(feat-to-dim "$example_feats" - 2>/dev/null) || [ -z $feat_dim ]; then
feat-to-dim "$example_feats" - # to see the error message.
echo "error getting feature dimension"
exit 1;
fi
$cmd JOB=1 $dir/log/init_mono.log \
gmm-init-mono $shared_phones_opt "--train-feats=$feats subset-feats --n=10 ark:- ark:-|" $lang/topo $feat_dim \
$dir/mono.mdl $dir/mono.tree || exit 1;
fi
if [ $stage -le -4 ]; then
# Get tree stats.
echo "$0: Accumulating tree stats"
$cmd JOB=1:$nj $dir/log/acc_tree.JOB.log \
convert-ali --frame-subsampling-factor=$alignment_subsampling_factor \
$alidir/final.mdl $dir/mono.mdl $dir/mono.tree "ark:gunzip -c $alidir/ali.JOB.gz|" ark:- \| \
acc-tree-stats $context_opts $tree_stats_opts --ci-phones=$ciphonelist $dir/mono.mdl \
"$feats" ark:- $dir/JOB.treeacc || exit 1;
[ "`ls $dir/*.treeacc | wc -w`" -ne "$nj" ] && echo "$0: Wrong #tree-accs" && exit 1;
$cmd $dir/log/sum_tree_acc.log \
sum-tree-stats $dir/treeacc $dir/*.treeacc || exit 1;
rm $dir/*.treeacc
fi
if [ $stage -le -3 ] && $train_tree; then
echo "$0: Getting questions for tree clustering."
# preparing questions, roots file...
$cmd $dir/log/questions.log \
cluster-phones $cluster_phones_opts $context_opts $dir/treeacc \
$lang/phones/sets.int $dir/questions.int || exit 1;
cat $lang/phones/extra_questions.int >> $dir/questions.int
$cmd $dir/log/compile_questions.log \
compile-questions $context_opts $lang/topo \
$dir/questions.int $dir/questions.qst || exit 1;
echo "$0: Building the tree"
$cmd $dir/log/build_tree.log \
build-tree $context_opts --verbose=1 --max-leaves=$numleaves \
--cluster-thresh=$cluster_thresh $dir/treeacc $lang/phones/roots.int \
$dir/questions.qst $lang/topo $dir/tree || exit 1;
fi
if [ $stage -le -2 ]; then
echo "$0: Initializing the model"
gmm-init-model --write-occs=$dir/1.occs \
$dir/tree $dir/treeacc $lang/topo $dir/1.mdl 2> $dir/log/init_model.log || exit 1;
grep 'no stats' $dir/log/init_model.log && echo "This is a bad warning.";
rm $dir/treeacc
fi
if [ $stage -le -1 ]; then
# Convert the alignments to the new tree. Note: we likely will not use these
# converted alignments in the chain system directly, but they could be useful
# for other purposes.
echo "$0: Converting alignments from $alidir to use current tree"
$cmd JOB=1:$nj $dir/log/convert.JOB.log \
convert-ali --repeat-frames=$repeat_frames \
--frame-subsampling-factor=$alignment_subsampling_factor \
$alidir/final.mdl $dir/1.mdl $dir/tree \
"ark:gunzip -c $alidir/ali.JOB.gz|" "ark:|gzip -c >$dir/ali.JOB.gz" || exit 1;
fi
cp $dir/1.mdl $dir/final.mdl
echo $0: Done building tree