annotate Matcher.cpp @ 9:fec395fcdc7c

ming32 build
author Chris Cannam
date Tue, 05 Aug 2014 11:06:04 +0100
parents b3ec48d9cd2e
children 6ea008aa8817
rev   line source
cannam@0 1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
cannam@0 2
cannam@0 3 /*
cannam@0 4 Vamp feature extraction plugin using the MATCH audio alignment
cannam@0 5 algorithm.
cannam@0 6
cannam@0 7 Centre for Digital Music, Queen Mary, University of London.
cannam@0 8 This file copyright 2007 Simon Dixon, Chris Cannam and QMUL.
cannam@0 9
cannam@0 10 This program is free software; you can redistribute it and/or
cannam@0 11 modify it under the terms of the GNU General Public License as
cannam@0 12 published by the Free Software Foundation; either version 2 of the
cannam@0 13 License, or (at your option) any later version. See the file
cannam@0 14 COPYING included with this distribution for more information.
cannam@0 15 */
cannam@0 16
cannam@0 17 #include "Matcher.h"
cannam@0 18 #include "Finder.h"
cannam@0 19
cannam@0 20 #include <iostream>
cannam@0 21
cannam@4 22 #include <cstdlib>
cannam@4 23
cannam@0 24 bool Matcher::silent = true;
cannam@0 25
cannam@7 26 const double Matcher::decay = 0.99;
cannam@7 27 const double Matcher::silenceThreshold = 0.0004;
cannam@7 28 const int Matcher::MAX_RUN_COUNT = 3;
cannam@7 29
cannam@0 30 Matcher::Matcher(float rate, Matcher *p)
cannam@0 31 {
cannam@0 32 std::cerr << "Matcher::Matcher(" << rate << ", " << p << ")" << std::endl;
cannam@0 33
cannam@0 34 sampleRate = rate;
cannam@0 35 otherMatcher = p; // the first matcher will need this to be set later
cannam@0 36 firstPM = (!p);
cannam@0 37 matchFileOffset = 0;
cannam@0 38 ltAverage = 0;
cannam@0 39 frameCount = 0;
cannam@0 40 runCount = 0;
cannam@0 41 paused = false;
cannam@0 42 hopSize = 0;
cannam@0 43 fftSize = 0;
cannam@0 44 blockSize = 0;
cannam@0 45 hopTime = 0.020; // DEFAULT, overridden with -h //!!!
cannam@0 46 fftTime = 0.04644; // DEFAULT, overridden with -f
cannam@0 47 blockTime = 10.0; // DEFAULT, overridden with -c
cannam@0 48 normalise1 = true;
cannam@0 49 normalise2 = false;
cannam@0 50 normalise3 = false;
cannam@0 51 normalise4 = true;
cannam@0 52 useSpectralDifference = true;
cannam@0 53 useChromaFrequencyMap = false;
cannam@0 54 scale = 90;
cannam@0 55 maxFrames = 0; // stop at EOF
cannam@0 56
cannam@0 57 hopSize = lrint(sampleRate * hopTime);
cannam@7 58 fftSize = lrint(pow(2.0, (int)lrint(log(fftTime * sampleRate) / log(2.0))));
cannam@0 59 blockSize = lrint(blockTime / hopTime);
cannam@0 60
cannam@0 61 distance = 0;
cannam@0 62 bestPathCost = 0;
cannam@0 63 distYSizes = 0;
cannam@0 64 distXSize = 0;
cannam@0 65
cannam@0 66 initialised = false;
cannam@0 67
cannam@0 68 } // default constructor
cannam@0 69
cannam@1 70 void
cannam@1 71 Matcher::setHopSize(int sz)
cannam@1 72 {
cannam@1 73 if (initialised) {
cannam@1 74 std::cerr << "Matcher::setHopSize: Can't set after use" << std::endl;
cannam@1 75 return;
cannam@1 76 }
cannam@1 77
cannam@1 78 hopSize = sz;
cannam@1 79 hopTime = float(hopSize) / sampleRate;
cannam@1 80 blockTime = blockSize * hopTime;
cannam@1 81 }
cannam@1 82
cannam@0 83 Matcher::~Matcher()
cannam@0 84 {
cannam@0 85 std::cerr << "Matcher(" << this << ")::~Matcher()" << std::endl;
cannam@0 86
cannam@0 87 if (initialised) {
cannam@0 88
cannam@0 89 for (int i = 0; i < distXSize; ++i) {
cannam@0 90 if (distance[i]) {
cannam@0 91 free(distance[i]);
cannam@0 92 free(bestPathCost[i]);
cannam@0 93 }
cannam@0 94 }
cannam@0 95 free(distance);
cannam@0 96 free(bestPathCost);
cannam@0 97
cannam@0 98 free(first);
cannam@0 99 free(last);
cannam@0 100
cannam@0 101 free(distYSizes);
cannam@0 102 }
cannam@0 103 }
cannam@0 104
cannam@0 105 void
cannam@0 106 Matcher::print()
cannam@0 107 {
cannam@0 108 cerr << toString() << endl;
cannam@0 109 } // print()
cannam@0 110
cannam@0 111 string
cannam@0 112 Matcher::toString()
cannam@0 113 {
cannam@0 114 std::stringstream os;
cannam@0 115 os << "Matcher " << this << ": (" << sampleRate
cannam@0 116 << "kHz)"
cannam@0 117 << "\n\tHop size: " << hopSize
cannam@0 118 << "\n\tFFT size: " << fftSize
cannam@0 119 << "\n\tBlock size: " << blockSize;
cannam@0 120 return os.str();
cannam@0 121 } // toString()
cannam@0 122
cannam@0 123 void
cannam@0 124 Matcher::init()
cannam@0 125 {
cannam@0 126 if (initialised) return;
cannam@0 127
cannam@0 128 initialised = true;
cannam@0 129
cannam@0 130 makeFreqMap(fftSize, sampleRate);
cannam@0 131
cannam@0 132 initVector<double>(prevFrame, freqMapSize);
cannam@0 133 initVector<double>(newFrame, freqMapSize);
cannam@0 134 initMatrix<double>(frames, blockSize, freqMapSize + 1);
cannam@0 135
cannam@0 136 int distSize = (MAX_RUN_COUNT + 1) * blockSize;
cannam@0 137
cannam@0 138 distXSize = blockSize * 2;
cannam@0 139
cannam@0 140 // std::cerr << "Matcher::init: distXSize = " << distXSize << std::endl;
cannam@0 141
cannam@0 142 distance = (unsigned char **)malloc(distXSize * sizeof(unsigned char *));
cannam@0 143 bestPathCost = (int **)malloc(distXSize * sizeof(int *));
cannam@0 144 distYSizes = (int *)malloc(distXSize * sizeof(int));
cannam@0 145
cannam@0 146 for (int i = 0; i < blockSize; ++i) {
cannam@0 147 distance[i] = (unsigned char *)malloc(distSize * sizeof(unsigned char));
cannam@0 148 bestPathCost[i] = (int *)malloc(distSize * sizeof(int));
cannam@0 149 distYSizes[i] = distSize;
cannam@0 150 }
cannam@0 151 for (int i = blockSize; i < distXSize; ++i) {
cannam@0 152 distance[i] = 0;
cannam@0 153 }
cannam@0 154
cannam@0 155 first = (int *)malloc(distXSize * sizeof(int));
cannam@0 156 last = (int *)malloc(distXSize * sizeof(int));
cannam@0 157
cannam@0 158 frameCount = 0;
cannam@0 159 runCount = 0;
cannam@0 160 // frameRMS = 0;
cannam@0 161 ltAverage = 0;
cannam@0 162
cannam@0 163 if (!silent) print();
cannam@0 164 } // init
cannam@0 165
cannam@0 166 void
cannam@0 167 Matcher::makeFreqMap(int fftSize, float sampleRate)
cannam@0 168 {
cannam@0 169 initVector<int>(freqMap, fftSize/2 + 1);
cannam@0 170 if (useChromaFrequencyMap)
cannam@0 171 makeChromaFrequencyMap(fftSize, sampleRate);
cannam@0 172 else
cannam@0 173 makeStandardFrequencyMap(fftSize, sampleRate);
cannam@0 174 } // makeFreqMap()
cannam@0 175
cannam@0 176 void
cannam@0 177 Matcher::makeStandardFrequencyMap(int fftSize, float sampleRate)
cannam@0 178 {
cannam@0 179 double binWidth = sampleRate / fftSize;
cannam@0 180 int crossoverBin = (int)(2 / (pow(2, 1/12.0) - 1));
cannam@7 181 int crossoverMidi = lrint(log(crossoverBin*binWidth/440.0)/
cannam@7 182 log(2.0) * 12 + 69);
cannam@0 183 // freq = 440 * Math.pow(2, (midi-69)/12.0) / binWidth;
cannam@0 184 int i = 0;
cannam@0 185 while (i <= crossoverBin) {
cannam@0 186 freqMap[i] = i;
cannam@0 187 ++i;
cannam@0 188 }
cannam@0 189 while (i <= fftSize/2) {
cannam@7 190 double midi = log(i*binWidth/440.0) / log(2.0) * 12 + 69;
cannam@0 191 if (midi > 127)
cannam@0 192 midi = 127;
cannam@0 193 freqMap[i++] = crossoverBin + lrint(midi) - crossoverMidi;
cannam@0 194 }
cannam@0 195 freqMapSize = freqMap[i-1] + 1;
cannam@0 196 if (!silent) {
cannam@0 197 cerr << "Standard map size: " << freqMapSize
cannam@0 198 << "; Crossover at: " << crossoverBin << endl;
cannam@0 199 //!!! for (i = 0; i < fftSize / 2; i++)
cannam@0 200 // cerr << "freqMap[" << i << "] = " << freqMap[i] << endl;
cannam@0 201 }
cannam@0 202 } // makeStandardFrequencyMap()
cannam@0 203
cannam@0 204 void
cannam@0 205 Matcher::makeChromaFrequencyMap(int fftSize, float sampleRate)
cannam@0 206 {
cannam@0 207 double binWidth = sampleRate / fftSize;
cannam@0 208 int crossoverBin = (int)(1 / (pow(2, 1/12.0) - 1));
cannam@0 209 // freq = 440 * Math.pow(2, (midi-69)/12.0) / binWidth;
cannam@0 210 int i = 0;
cannam@0 211 while (i <= crossoverBin)
cannam@0 212 freqMap[i++] = 0;
cannam@0 213 while (i <= fftSize/2) {
cannam@7 214 double midi = log(i*binWidth/440.0) / log(2.0) * 12 + 69;
cannam@0 215 freqMap[i++] = (lrint(midi)) % 12 + 1;
cannam@0 216 }
cannam@0 217 freqMapSize = 13;
cannam@0 218 if (!silent) {
cannam@0 219 cerr << "Chroma map size: " << freqMapSize
cannam@0 220 << "; Crossover at: " << crossoverBin << endl;
cannam@0 221 for (i = 0; i < fftSize / 2; i++)
cannam@0 222 cerr << "freqMap[" << i << "] = " << freqMap[i] << endl;
cannam@0 223 }
cannam@0 224 } // makeChromaFrequencyMap()
cannam@0 225
cannam@0 226 void
cannam@0 227 Matcher::processFrame(double *reBuffer, double *imBuffer)
cannam@0 228 {
cannam@0 229 if (!initialised) init();
cannam@0 230
cannam@0 231 for (int i = 0; i < (int)newFrame.size(); ++i) {
cannam@0 232 newFrame[i] = 0;
cannam@0 233 }
cannam@0 234 double rms = 0;
cannam@0 235 for (int i = 0; i <= fftSize/2; i++) {
cannam@0 236 double mag = reBuffer[i] * reBuffer[i] +
cannam@0 237 imBuffer[i] * imBuffer[i];
cannam@0 238 rms += mag;
cannam@0 239 newFrame[freqMap[i]] += mag;
cannam@0 240 }
cannam@0 241 rms = sqrt(rms / (fftSize/2));
cannam@0 242
cannam@0 243 int frameIndex = frameCount % blockSize;
cannam@0 244
cannam@0 245 if (frameCount >= distXSize) {
cannam@0 246 // std::cerr << "Resizing " << distXSize << " -> " << distXSize * 2 << std::endl;
cannam@0 247 distXSize *= 2;
cannam@0 248 distance = (unsigned char **)realloc(distance, distXSize * sizeof(unsigned char *));
cannam@0 249 bestPathCost = (int **)realloc(bestPathCost, distXSize * sizeof(int *));
cannam@0 250 distYSizes = (int *)realloc(distYSizes, distXSize * sizeof(int));
cannam@0 251 first = (int *)realloc(first, distXSize * sizeof(int));
cannam@0 252 last = (int *)realloc(last, distXSize * sizeof(int));
cannam@0 253
cannam@0 254 for (int i = distXSize/2; i < distXSize; ++i) {
cannam@0 255 distance[i] = 0;
cannam@0 256 }
cannam@0 257 }
cannam@0 258
cannam@0 259 if (firstPM && (frameCount >= blockSize)) {
cannam@0 260
cannam@0 261 int len = last[frameCount - blockSize] -
cannam@0 262 first[frameCount - blockSize];
cannam@0 263
cannam@0 264 // We need to copy distance[frameCount-blockSize] to
cannam@0 265 // distance[frameCount], and then truncate
cannam@0 266 // distance[frameCount-blockSize] to its first len elements.
cannam@0 267 // Same for bestPathCost.
cannam@0 268 /*
cannam@4 269 std::cerr << "Matcher(" << this << "): moving " << distYSizes[frameCount - blockSize] << " from " << frameCount - blockSize << " to "
cannam@0 270 << frameCount << ", allocating " << len << " for "
cannam@0 271 << frameCount - blockSize << std::endl;
cannam@0 272 */
cannam@0 273 distance[frameCount] = distance[frameCount - blockSize];
cannam@0 274
cannam@0 275 distance[frameCount - blockSize] = (unsigned char *)
cannam@0 276 malloc(len * sizeof(unsigned char));
cannam@0 277 for (int i = 0; i < len; ++i) {
cannam@0 278 distance[frameCount - blockSize][i] =
cannam@0 279 distance[frameCount][i];
cannam@0 280 }
cannam@0 281
cannam@0 282 bestPathCost[frameCount] = bestPathCost[frameCount - blockSize];
cannam@0 283
cannam@0 284 bestPathCost[frameCount - blockSize] = (int *)
cannam@0 285 malloc(len * sizeof(int));
cannam@0 286 for (int i = 0; i < len; ++i) {
cannam@0 287 bestPathCost[frameCount - blockSize][i] =
cannam@0 288 bestPathCost[frameCount][i];
cannam@0 289 }
cannam@0 290
cannam@0 291 distYSizes[frameCount] = distYSizes[frameCount - blockSize];
cannam@0 292 distYSizes[frameCount - blockSize] = len;
cannam@0 293 }
cannam@0 294
cannam@0 295 double totalEnergy = 0;
cannam@0 296 if (useSpectralDifference) {
cannam@0 297 for (int i = 0; i < freqMapSize; i++) {
cannam@0 298 totalEnergy += newFrame[i];
cannam@0 299 if (newFrame[i] > prevFrame[i]) {
cannam@0 300 frames[frameIndex][i] = newFrame[i] - prevFrame[i];
cannam@0 301 } else {
cannam@0 302 frames[frameIndex][i] = 0;
cannam@0 303 }
cannam@0 304 }
cannam@0 305 } else {
cannam@0 306 for (int i = 0; i < freqMapSize; i++) {
cannam@0 307 frames[frameIndex][i] = newFrame[i];
cannam@0 308 totalEnergy += frames[frameIndex][i];
cannam@0 309 }
cannam@0 310 }
cannam@0 311 frames[frameIndex][freqMapSize] = totalEnergy;
cannam@0 312
cannam@0 313 double decay = frameCount >= 200 ? 0.99:
cannam@0 314 (frameCount < 100? 0: (frameCount - 100) / 100.0);
cannam@0 315
cannam@0 316 if (ltAverage == 0)
cannam@0 317 ltAverage = totalEnergy;
cannam@0 318 else
cannam@0 319 ltAverage = ltAverage * decay + totalEnergy * (1.0 - decay);
cannam@0 320
cannam@0 321 // System.err.println(Format.d(ltAverage,4) + " " +
cannam@0 322 // Format.d(totalEnergy) + " " +
cannam@0 323 // Format.d(frameRMS));
cannam@0 324
cannam@0 325 // std::cerr << "ltAverage: " << ltAverage << ", totalEnergy: " << totalEnergy << ", frameRMS: " << rms << std::endl;
cannam@0 326
cannam@0 327 if (rms <= 0.01) //!!! silenceThreshold)
cannam@0 328 for (int i = 0; i < freqMapSize; i++)
cannam@0 329 frames[frameIndex][i] = 0;
cannam@0 330 else if (normalise1)
cannam@0 331 for (int i = 0; i < freqMapSize; i++)
cannam@0 332 frames[frameIndex][i] /= totalEnergy;
cannam@0 333 else if (normalise3)
cannam@0 334 for (int i = 0; i < freqMapSize; i++)
cannam@0 335 frames[frameIndex][i] /= ltAverage;
cannam@0 336
cannam@0 337 int stop = otherMatcher->frameCount;
cannam@0 338 int index = stop - blockSize;
cannam@0 339 if (index < 0)
cannam@0 340 index = 0;
cannam@0 341 first[frameCount] = index;
cannam@0 342 last[frameCount] = stop;
cannam@0 343
cannam@0 344 bool overflow = false;
cannam@0 345 int mn= -1;
cannam@0 346 int mx= -1;
cannam@0 347 for ( ; index < stop; index++) {
cannam@0 348 int dMN = calcDistance(frames[frameIndex],
cannam@0 349 otherMatcher->frames[index % blockSize]);
cannam@0 350 if (mx<0)
cannam@0 351 mx = mn = dMN;
cannam@0 352 else if (dMN > mx)
cannam@0 353 mx = dMN;
cannam@0 354 else if (dMN < mn)
cannam@0 355 mn = dMN;
cannam@0 356 if (dMN >= 255) {
cannam@0 357 overflow = true;
cannam@0 358 dMN = 255;
cannam@0 359 }
cannam@0 360 if ((frameCount == 0) && (index == 0)) // first element
cannam@0 361 setValue(0, 0, 0, 0, dMN);
cannam@0 362 else if (frameCount == 0) // first row
cannam@0 363 setValue(0, index, ADVANCE_OTHER,
cannam@0 364 getValue(0, index-1, true), dMN);
cannam@0 365 else if (index == 0) // first column
cannam@0 366 setValue(frameCount, index, ADVANCE_THIS,
cannam@0 367 getValue(frameCount - 1, 0, true), dMN);
cannam@0 368 else if (index == otherMatcher->frameCount - blockSize) {
cannam@0 369 // missing value(s) due to cutoff
cannam@0 370 // - no previous value in current row (resp. column)
cannam@0 371 // - no diagonal value if prev. dir. == curr. dirn
cannam@0 372 int min2 = getValue(frameCount - 1, index, true);
cannam@0 373 // if ((firstPM && (first[frameCount - 1] == index)) ||
cannam@0 374 // (!firstPM && (last[index-1] < frameCount)))
cannam@0 375 if (first[frameCount - 1] == index)
cannam@0 376 setValue(frameCount, index, ADVANCE_THIS, min2, dMN);
cannam@0 377 else {
cannam@0 378 int min1 = getValue(frameCount - 1, index - 1, true);
cannam@0 379 if (min1 + dMN <= min2)
cannam@0 380 setValue(frameCount, index, ADVANCE_BOTH, min1,dMN);
cannam@0 381 else
cannam@0 382 setValue(frameCount, index, ADVANCE_THIS, min2,dMN);
cannam@0 383 }
cannam@0 384 } else {
cannam@0 385 int min1 = getValue(frameCount, index-1, true);
cannam@0 386 int min2 = getValue(frameCount - 1, index, true);
cannam@0 387 int min3 = getValue(frameCount - 1, index-1, true);
cannam@0 388 if (min1 <= min2) {
cannam@0 389 if (min3 + dMN <= min1)
cannam@0 390 setValue(frameCount, index, ADVANCE_BOTH, min3,dMN);
cannam@0 391 else
cannam@0 392 setValue(frameCount, index, ADVANCE_OTHER,min1,dMN);
cannam@0 393 } else {
cannam@0 394 if (min3 + dMN <= min2)
cannam@0 395 setValue(frameCount, index, ADVANCE_BOTH, min3,dMN);
cannam@0 396 else
cannam@0 397 setValue(frameCount, index, ADVANCE_THIS, min2,dMN);
cannam@0 398 }
cannam@0 399 }
cannam@0 400 otherMatcher->last[index]++;
cannam@0 401 } // loop for row (resp. column)
cannam@0 402
cannam@0 403 vector<double> tmp = prevFrame;
cannam@0 404 prevFrame = newFrame;
cannam@0 405 newFrame = tmp;
cannam@0 406
cannam@0 407 frameCount++;
cannam@0 408 runCount++;
cannam@0 409
cannam@0 410 otherMatcher->runCount = 0;
cannam@0 411
cannam@0 412 if (overflow && !silent)
cannam@0 413 cerr << "WARNING: overflow in distance metric: "
cannam@0 414 << "frame " << frameCount << ", val = " << mx << endl;
cannam@0 415
cannam@0 416 if (!silent)
cannam@0 417 std::cerr << "Frame " << frameCount << ", d = " << (mx-mn) << std::endl;
cannam@0 418
cannam@0 419 if ((frameCount % 100) == 0) {
cannam@0 420 if (!silent) {
cannam@0 421 cerr << "Progress:" << frameCount << " " << ltAverage << endl;
cannam@0 422 // Profile.report();
cannam@0 423 }
cannam@0 424 }
cannam@0 425 //!!! if (frameCount == maxFrames)
cannam@0 426 // closeStreams();
cannam@0 427 // }
cannam@0 428 } // processFrame()
cannam@0 429
cannam@0 430 int
cannam@0 431 Matcher::calcDistance(const vector<double> &f1, const vector<double> &f2)
cannam@0 432 {
cannam@0 433 double d = 0;
cannam@0 434 double sum = 0;
cannam@0 435 for (int i = 0; i < freqMapSize; i++) {
cannam@0 436 d += fabs(f1[i] - f2[i]);
cannam@0 437 sum += f1[i] + f2[i];
cannam@0 438 }
cannam@0 439 // System.err.print(" " + Format.d(d,3));
cannam@0 440 if (sum == 0)
cannam@0 441 return 0;
cannam@0 442 if (normalise2)
cannam@0 443 return (int)(scale * d / sum); // 0 <= d/sum <= 2
cannam@0 444 if (!normalise4)
cannam@0 445 return (int)(scale * d);
cannam@0 446 // double weight = (5 + Math.log(f1[freqMapSize] + f2[freqMapSize]))/10.0;
cannam@0 447 double weight = (8 + log(sum)) / 10.0;
cannam@0 448 // if (weight < mins) {
cannam@0 449 // mins = weight;
cannam@0 450 // System.err.println(Format.d(mins,3) + " " + Format.d(maxs));
cannam@0 451 // }
cannam@0 452 // if (weight > maxs) {
cannam@0 453 // maxs = weight;
cannam@0 454 // System.err.println(Format.d(mins,3) + " " + Format.d(maxs));
cannam@0 455 // }
cannam@0 456 if (weight < 0)
cannam@0 457 weight = 0;
cannam@0 458 else if (weight > 1)
cannam@0 459 weight = 1;
cannam@0 460 return (int)(scale * d / sum * weight);
cannam@0 461 } // calcDistance()
cannam@0 462
cannam@0 463 int
cannam@0 464 Matcher::getValue(int i, int j, bool firstAttempt)
cannam@0 465 {
cannam@0 466 if (firstPM)
cannam@0 467 return bestPathCost[i][j - first[i]];
cannam@0 468 else
cannam@0 469 return otherMatcher->bestPathCost[j][i - otherMatcher->first[j]];
cannam@0 470 } // getValue()
cannam@0 471
cannam@0 472 void
cannam@0 473 Matcher::setValue(int i, int j, int dir, int value, int dMN)
cannam@0 474 {
cannam@0 475 if (firstPM) {
cannam@0 476 distance[i][j - first[i]] = (unsigned char)((dMN & MASK) | dir);
cannam@0 477 bestPathCost[i][j - first[i]] =
cannam@0 478 (value + (dir==ADVANCE_BOTH? dMN*2: dMN));
cannam@0 479 } else {
cannam@0 480 if (dir == ADVANCE_THIS)
cannam@0 481 dir = ADVANCE_OTHER;
cannam@0 482 else if (dir == ADVANCE_OTHER)
cannam@0 483 dir = ADVANCE_THIS;
cannam@0 484 int idx = i - otherMatcher->first[j];
cannam@0 485 if (idx == (int)otherMatcher->distYSizes[j]) {
cannam@0 486 // This should never happen, but if we allow arbitrary
cannam@0 487 // pauses in either direction, and arbitrary lengths at
cannam@0 488 // end, it is better than a segmentation fault.
cannam@0 489 std::cerr << "Emergency resize: " << idx << " -> " << idx * 2 << std::endl;
cannam@0 490 otherMatcher->distYSizes[j] = idx * 2;
cannam@0 491 otherMatcher->bestPathCost[j] =
cannam@0 492 (int *)realloc(otherMatcher->bestPathCost[j],
cannam@0 493 idx * 2 * sizeof(int));
cannam@0 494 otherMatcher->distance[j] =
cannam@0 495 (unsigned char *)realloc(otherMatcher->distance[j],
cannam@0 496 idx * 2 * sizeof(unsigned char));
cannam@0 497 }
cannam@0 498 otherMatcher->distance[j][idx] = (unsigned char)((dMN & MASK) | dir);
cannam@0 499 otherMatcher->bestPathCost[j][idx] =
cannam@0 500 (value + (dir==ADVANCE_BOTH? dMN*2: dMN));
cannam@0 501 }
cannam@0 502 } // setValue()
cannam@0 503