annotate Matcher.cpp @ 26:9f60d097f0b2

Pull out DistanceMetric into its own class
author Chris Cannam
date Fri, 31 Oct 2014 11:31:08 +0000
parents 64c4c0cf80c9
children cd0ffe6b1e68
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
cannam@0 19 #include <iostream>
cannam@0 20
cannam@4 21 #include <cstdlib>
Chris@16 22 #include <cassert>
cannam@4 23
cannam@0 24 bool Matcher::silent = true;
cannam@0 25
Chris@10 26 //#define DEBUG_MATCHER 1
Chris@10 27
Chris@15 28 Matcher::Matcher(Parameters parameters, Matcher *p) :
Chris@26 29 params(parameters),
Chris@26 30 metric(parameters.distanceNorm)
cannam@0 31 {
Chris@10 32 #ifdef DEBUG_MATCHER
Chris@15 33 cerr << "Matcher::Matcher(" << params.sampleRate << ", " << p << ")" << endl;
Chris@10 34 #endif
cannam@0 35
cannam@0 36 otherMatcher = p; // the first matcher will need this to be set later
cannam@0 37 firstPM = (!p);
cannam@0 38 ltAverage = 0;
cannam@0 39 frameCount = 0;
cannam@0 40 runCount = 0;
Chris@23 41 freqMapSize = 0;
Chris@23 42 externalFeatureSize = 0;
Chris@23 43 featureSize = 0;
Chris@23 44 blockSize = 0;
Chris@23 45 scale = 90;
Chris@23 46
Chris@23 47 blockSize = lrint(params.blockTime / params.hopTime);
Chris@23 48 #ifdef DEBUG_MATCHER
Chris@23 49 cerr << "Matcher: blockSize = " << blockSize << endl;
Chris@23 50 #endif
Chris@23 51
Chris@23 52 distance = 0;
Chris@23 53 bestPathCost = 0;
Chris@23 54 distYSizes = 0;
Chris@23 55 distXSize = 0;
Chris@23 56
Chris@23 57 initialised = false;
Chris@23 58 }
Chris@23 59
Chris@23 60 Matcher::Matcher(Parameters parameters, Matcher *p, int featureSize) :
Chris@23 61 params(parameters),
Chris@26 62 externalFeatureSize(featureSize),
Chris@26 63 metric(parameters.distanceNorm)
Chris@23 64 {
Chris@23 65 #ifdef DEBUG_MATCHER
Chris@23 66 cerr << "Matcher::Matcher(" << params.sampleRate << ", " << p << ", " << featureSize << ")" << endl;
Chris@23 67 #endif
Chris@23 68
Chris@23 69 otherMatcher = p; // the first matcher will need this to be set later
Chris@23 70 firstPM = (!p);
Chris@23 71 ltAverage = 0;
Chris@23 72 frameCount = 0;
Chris@23 73 runCount = 0;
Chris@23 74 freqMapSize = 0;
Chris@23 75 featureSize = 0;
cannam@0 76 blockSize = 0;
cannam@0 77 scale = 90;
cannam@0 78
Chris@15 79 blockSize = lrint(params.blockTime / params.hopTime);
Chris@15 80 #ifdef DEBUG_MATCHER
Chris@15 81 cerr << "Matcher: blockSize = " << blockSize << endl;
Chris@15 82 #endif
cannam@0 83
cannam@0 84 distance = 0;
cannam@0 85 bestPathCost = 0;
cannam@0 86 distYSizes = 0;
cannam@0 87 distXSize = 0;
cannam@0 88
cannam@0 89 initialised = false;
cannam@0 90
Chris@23 91 }
cannam@0 92
cannam@0 93 Matcher::~Matcher()
cannam@0 94 {
Chris@10 95 #ifdef DEBUG_MATCHER
Chris@15 96 cerr << "Matcher(" << this << ")::~Matcher()" << endl;
Chris@10 97 #endif
cannam@0 98
cannam@0 99 if (initialised) {
cannam@0 100
cannam@0 101 for (int i = 0; i < distXSize; ++i) {
cannam@0 102 if (distance[i]) {
cannam@0 103 free(distance[i]);
cannam@0 104 free(bestPathCost[i]);
cannam@0 105 }
cannam@0 106 }
cannam@0 107 free(distance);
cannam@0 108 free(bestPathCost);
cannam@0 109
cannam@0 110 free(first);
cannam@0 111 free(last);
cannam@0 112
cannam@0 113 free(distYSizes);
cannam@0 114 }
cannam@0 115 }
cannam@0 116
cannam@0 117 void
cannam@0 118 Matcher::init()
cannam@0 119 {
cannam@0 120 if (initialised) return;
cannam@0 121
cannam@0 122 initialised = true;
cannam@0 123
Chris@23 124 if (externalFeatureSize == 0) {
Chris@23 125 freqMapSize = getFeatureSizeFor(params);
Chris@23 126 featureSize = freqMapSize;
Chris@23 127 makeFreqMap();
Chris@23 128 } else {
Chris@23 129 featureSize = externalFeatureSize;
Chris@23 130 }
Chris@16 131
Chris@23 132 initVector<double>(prevFrame, featureSize);
Chris@23 133 initVector<double>(newFrame, featureSize);
Chris@23 134 initMatrix<double>(frames, blockSize, featureSize);
Chris@13 135 initVector<double>(totalEnergies, blockSize);
cannam@0 136
Chris@15 137 int distSize = (params.maxRunCount + 1) * blockSize;
cannam@0 138
cannam@0 139 distXSize = blockSize * 2;
cannam@0 140
cannam@0 141 distance = (unsigned char **)malloc(distXSize * sizeof(unsigned char *));
cannam@0 142 bestPathCost = (int **)malloc(distXSize * sizeof(int *));
cannam@0 143 distYSizes = (int *)malloc(distXSize * sizeof(int));
cannam@0 144
cannam@0 145 for (int i = 0; i < blockSize; ++i) {
cannam@0 146 distance[i] = (unsigned char *)malloc(distSize * sizeof(unsigned char));
cannam@0 147 bestPathCost[i] = (int *)malloc(distSize * sizeof(int));
cannam@0 148 distYSizes[i] = distSize;
cannam@0 149 }
cannam@0 150 for (int i = blockSize; i < distXSize; ++i) {
cannam@0 151 distance[i] = 0;
cannam@0 152 }
cannam@0 153
cannam@0 154 first = (int *)malloc(distXSize * sizeof(int));
cannam@0 155 last = (int *)malloc(distXSize * sizeof(int));
cannam@0 156
cannam@0 157 frameCount = 0;
cannam@0 158 runCount = 0;
cannam@0 159 ltAverage = 0;
cannam@0 160
cannam@0 161 } // init
cannam@0 162
cannam@0 163 void
Chris@15 164 Matcher::makeFreqMap()
cannam@0 165 {
Chris@15 166 initVector<int>(freqMap, params.fftSize/2 + 1);
Chris@23 167
Chris@15 168 if (params.useChromaFrequencyMap) {
Chris@15 169 #ifdef DEBUG_MATCHER
Chris@15 170 cerr << "makeFreqMap: calling makeChromaFrequencyMap" << endl;
Chris@15 171 #endif
Chris@15 172 makeChromaFrequencyMap();
Chris@15 173 } else {
Chris@15 174 #ifdef DEBUG_MATCHER
Chris@15 175 cerr << "makeFreqMap: calling makeStandardFrequencyMap" << endl;
Chris@15 176 #endif
Chris@15 177 makeStandardFrequencyMap();
Chris@15 178 }
cannam@0 179 } // makeFreqMap()
cannam@0 180
Chris@16 181 int
Chris@23 182 Matcher::getFeatureSizeFor(Parameters params)
Chris@16 183 {
Chris@16 184 if (params.useChromaFrequencyMap) {
Chris@16 185 return 13;
Chris@16 186 } else {
Chris@16 187 return 84;
Chris@16 188 }
Chris@16 189 }
Chris@16 190
cannam@0 191 void
Chris@15 192 Matcher::makeStandardFrequencyMap()
cannam@0 193 {
Chris@15 194 double binWidth = params.sampleRate / params.fftSize;
cannam@0 195 int crossoverBin = (int)(2 / (pow(2, 1/12.0) - 1));
cannam@7 196 int crossoverMidi = lrint(log(crossoverBin*binWidth/440.0)/
cannam@7 197 log(2.0) * 12 + 69);
cannam@0 198 // freq = 440 * Math.pow(2, (midi-69)/12.0) / binWidth;
cannam@0 199 int i = 0;
cannam@0 200 while (i <= crossoverBin) {
cannam@0 201 freqMap[i] = i;
cannam@0 202 ++i;
cannam@0 203 }
Chris@15 204 while (i <= params.fftSize/2) {
cannam@7 205 double midi = log(i*binWidth/440.0) / log(2.0) * 12 + 69;
Chris@16 206 if (midi > 127) midi = 127;
cannam@0 207 freqMap[i++] = crossoverBin + lrint(midi) - crossoverMidi;
cannam@0 208 }
Chris@16 209 assert(freqMapSize == freqMap[i-1] + 1);
cannam@0 210 if (!silent) {
cannam@0 211 cerr << "Standard map size: " << freqMapSize
cannam@0 212 << "; Crossover at: " << crossoverBin << endl;
Chris@15 213 for (i = 0; i < params.fftSize / 2; i++)
Chris@15 214 cerr << "freqMap[" << i << "] = " << freqMap[i] << endl;
cannam@0 215 }
cannam@0 216 } // makeStandardFrequencyMap()
cannam@0 217
cannam@0 218 void
Chris@15 219 Matcher::makeChromaFrequencyMap()
cannam@0 220 {
Chris@15 221 double binWidth = params.sampleRate / params.fftSize;
cannam@0 222 int crossoverBin = (int)(1 / (pow(2, 1/12.0) - 1));
cannam@0 223 // freq = 440 * Math.pow(2, (midi-69)/12.0) / binWidth;
cannam@0 224 int i = 0;
cannam@0 225 while (i <= crossoverBin)
cannam@0 226 freqMap[i++] = 0;
Chris@15 227 while (i <= params.fftSize/2) {
cannam@7 228 double midi = log(i*binWidth/440.0) / log(2.0) * 12 + 69;
cannam@0 229 freqMap[i++] = (lrint(midi)) % 12 + 1;
cannam@0 230 }
cannam@0 231 if (!silent) {
cannam@0 232 cerr << "Chroma map size: " << freqMapSize
cannam@0 233 << "; Crossover at: " << crossoverBin << endl;
Chris@15 234 for (i = 0; i < params.fftSize / 2; i++)
cannam@0 235 cerr << "freqMap[" << i << "] = " << freqMap[i] << endl;
cannam@0 236 }
cannam@0 237 } // makeChromaFrequencyMap()
cannam@0 238
Chris@14 239 vector<double>
Chris@21 240 Matcher::consumeFrame(double *reBuffer, double *imBuffer)
cannam@0 241 {
cannam@0 242 if (!initialised) init();
cannam@0 243
Chris@21 244 vector<double> processedFrame =
Chris@21 245 processFrameFromFreqData(reBuffer, imBuffer);
Chris@21 246
Chris@21 247 calcAdvance();
Chris@21 248
Chris@23 249 return processedFrame;
Chris@23 250 }
Chris@21 251
Chris@23 252 void
Chris@23 253 Matcher::consumeFeatureVector(std::vector<double> feature)
Chris@23 254 {
Chris@23 255 if (!initialised) init();
Chris@23 256 int frameIndex = frameCount % blockSize;
Chris@23 257 frames[frameIndex] = feature;
Chris@23 258 calcAdvance();
Chris@21 259 }
Chris@21 260
Chris@21 261 vector<double>
Chris@21 262 Matcher::processFrameFromFreqData(double *reBuffer, double *imBuffer)
Chris@21 263 {
cannam@0 264 for (int i = 0; i < (int)newFrame.size(); ++i) {
cannam@0 265 newFrame[i] = 0;
cannam@0 266 }
cannam@0 267 double rms = 0;
Chris@15 268 for (int i = 0; i <= params.fftSize/2; i++) {
cannam@0 269 double mag = reBuffer[i] * reBuffer[i] +
cannam@0 270 imBuffer[i] * imBuffer[i];
cannam@0 271 rms += mag;
cannam@0 272 newFrame[freqMap[i]] += mag;
cannam@0 273 }
Chris@15 274 rms = sqrt(rms / (params.fftSize/2));
cannam@0 275
cannam@0 276 int frameIndex = frameCount % blockSize;
cannam@0 277
Chris@21 278 vector<double> processedFrame(freqMapSize, 0.0);
Chris@21 279
Chris@21 280 double totalEnergy = 0;
Chris@21 281 if (params.useSpectralDifference) {
Chris@21 282 for (int i = 0; i < freqMapSize; i++) {
Chris@21 283 totalEnergy += newFrame[i];
Chris@21 284 if (newFrame[i] > prevFrame[i]) {
Chris@21 285 processedFrame[i] = newFrame[i] - prevFrame[i];
Chris@21 286 } else {
Chris@21 287 processedFrame[i] = 0;
Chris@21 288 }
Chris@21 289 }
Chris@21 290 } else {
Chris@21 291 for (int i = 0; i < freqMapSize; i++) {
Chris@21 292 processedFrame[i] = newFrame[i];
Chris@21 293 totalEnergy += processedFrame[i];
Chris@21 294 }
Chris@21 295 }
Chris@21 296 totalEnergies[frameIndex] = totalEnergy;
Chris@21 297
Chris@21 298 double decay = frameCount >= 200 ? 0.99:
Chris@21 299 (frameCount < 100? 0: (frameCount - 100) / 100.0);
Chris@21 300
Chris@21 301 if (ltAverage == 0)
Chris@21 302 ltAverage = totalEnergy;
Chris@21 303 else
Chris@21 304 ltAverage = ltAverage * decay + totalEnergy * (1.0 - decay);
Chris@21 305
Chris@21 306 if (rms <= params.silenceThreshold)
Chris@21 307 for (int i = 0; i < freqMapSize; i++)
Chris@21 308 processedFrame[i] = 0;
Chris@21 309 else if (params.frameNorm == NormaliseFrameToSum1)
Chris@21 310 for (int i = 0; i < freqMapSize; i++)
Chris@21 311 processedFrame[i] /= totalEnergy;
Chris@21 312 else if (params.frameNorm == NormaliseFrameToLTAverage)
Chris@21 313 for (int i = 0; i < freqMapSize; i++)
Chris@21 314 processedFrame[i] /= ltAverage;
Chris@21 315
Chris@21 316 vector<double> tmp = prevFrame;
Chris@21 317 prevFrame = newFrame;
Chris@21 318 newFrame = tmp;
Chris@21 319
Chris@21 320 frames[frameIndex] = processedFrame;
Chris@21 321
Chris@23 322 if ((frameCount % 100) == 0) {
Chris@23 323 if (!silent) {
Chris@23 324 cerr << "Progress:" << frameCount << " " << ltAverage << endl;
Chris@23 325 }
Chris@23 326 }
Chris@23 327
Chris@21 328 return processedFrame;
Chris@21 329 }
Chris@21 330
Chris@21 331 void
Chris@21 332 Matcher::calcAdvance()
Chris@21 333 {
Chris@21 334 int frameIndex = frameCount % blockSize;
Chris@21 335
cannam@0 336 if (frameCount >= distXSize) {
cannam@0 337 // std::cerr << "Resizing " << distXSize << " -> " << distXSize * 2 << std::endl;
cannam@0 338 distXSize *= 2;
cannam@0 339 distance = (unsigned char **)realloc(distance, distXSize * sizeof(unsigned char *));
cannam@0 340 bestPathCost = (int **)realloc(bestPathCost, distXSize * sizeof(int *));
cannam@0 341 distYSizes = (int *)realloc(distYSizes, distXSize * sizeof(int));
cannam@0 342 first = (int *)realloc(first, distXSize * sizeof(int));
cannam@0 343 last = (int *)realloc(last, distXSize * sizeof(int));
cannam@0 344
cannam@0 345 for (int i = distXSize/2; i < distXSize; ++i) {
cannam@0 346 distance[i] = 0;
cannam@0 347 }
cannam@0 348 }
cannam@0 349
cannam@0 350 if (firstPM && (frameCount >= blockSize)) {
cannam@0 351
cannam@0 352 int len = last[frameCount - blockSize] -
cannam@0 353 first[frameCount - blockSize];
cannam@0 354
cannam@0 355 // We need to copy distance[frameCount-blockSize] to
cannam@0 356 // distance[frameCount], and then truncate
cannam@0 357 // distance[frameCount-blockSize] to its first len elements.
cannam@0 358 // Same for bestPathCost.
cannam@0 359 /*
cannam@4 360 std::cerr << "Matcher(" << this << "): moving " << distYSizes[frameCount - blockSize] << " from " << frameCount - blockSize << " to "
cannam@0 361 << frameCount << ", allocating " << len << " for "
cannam@0 362 << frameCount - blockSize << std::endl;
cannam@0 363 */
cannam@0 364 distance[frameCount] = distance[frameCount - blockSize];
cannam@0 365
cannam@0 366 distance[frameCount - blockSize] = (unsigned char *)
cannam@0 367 malloc(len * sizeof(unsigned char));
cannam@0 368 for (int i = 0; i < len; ++i) {
cannam@0 369 distance[frameCount - blockSize][i] =
cannam@0 370 distance[frameCount][i];
cannam@0 371 }
cannam@0 372
cannam@0 373 bestPathCost[frameCount] = bestPathCost[frameCount - blockSize];
cannam@0 374
cannam@0 375 bestPathCost[frameCount - blockSize] = (int *)
cannam@0 376 malloc(len * sizeof(int));
cannam@0 377 for (int i = 0; i < len; ++i) {
cannam@0 378 bestPathCost[frameCount - blockSize][i] =
cannam@0 379 bestPathCost[frameCount][i];
cannam@0 380 }
cannam@0 381
cannam@0 382 distYSizes[frameCount] = distYSizes[frameCount - blockSize];
cannam@0 383 distYSizes[frameCount - blockSize] = len;
cannam@0 384 }
cannam@0 385
cannam@0 386 int stop = otherMatcher->frameCount;
cannam@0 387 int index = stop - blockSize;
cannam@0 388 if (index < 0)
cannam@0 389 index = 0;
cannam@0 390 first[frameCount] = index;
cannam@0 391 last[frameCount] = stop;
cannam@0 392
cannam@0 393 bool overflow = false;
cannam@0 394 int mn= -1;
cannam@0 395 int mx= -1;
cannam@0 396 for ( ; index < stop; index++) {
Chris@26 397
Chris@26 398 int dMN = metric.calcDistanceScaled
Chris@26 399 (frames[frameIndex],
Chris@26 400 otherMatcher->frames[index % blockSize],
Chris@26 401 scale);
Chris@26 402
cannam@0 403 if (mx<0)
cannam@0 404 mx = mn = dMN;
cannam@0 405 else if (dMN > mx)
cannam@0 406 mx = dMN;
cannam@0 407 else if (dMN < mn)
cannam@0 408 mn = dMN;
cannam@0 409 if (dMN >= 255) {
cannam@0 410 overflow = true;
cannam@0 411 dMN = 255;
cannam@0 412 }
Chris@26 413
cannam@0 414 if ((frameCount == 0) && (index == 0)) // first element
cannam@0 415 setValue(0, 0, 0, 0, dMN);
cannam@0 416 else if (frameCount == 0) // first row
cannam@0 417 setValue(0, index, ADVANCE_OTHER,
cannam@0 418 getValue(0, index-1, true), dMN);
cannam@0 419 else if (index == 0) // first column
cannam@0 420 setValue(frameCount, index, ADVANCE_THIS,
cannam@0 421 getValue(frameCount - 1, 0, true), dMN);
cannam@0 422 else if (index == otherMatcher->frameCount - blockSize) {
cannam@0 423 // missing value(s) due to cutoff
cannam@0 424 // - no previous value in current row (resp. column)
cannam@0 425 // - no diagonal value if prev. dir. == curr. dirn
cannam@0 426 int min2 = getValue(frameCount - 1, index, true);
cannam@0 427 // if ((firstPM && (first[frameCount - 1] == index)) ||
cannam@0 428 // (!firstPM && (last[index-1] < frameCount)))
cannam@0 429 if (first[frameCount - 1] == index)
cannam@0 430 setValue(frameCount, index, ADVANCE_THIS, min2, dMN);
cannam@0 431 else {
cannam@0 432 int min1 = getValue(frameCount - 1, index - 1, true);
cannam@0 433 if (min1 + dMN <= min2)
cannam@0 434 setValue(frameCount, index, ADVANCE_BOTH, min1,dMN);
cannam@0 435 else
cannam@0 436 setValue(frameCount, index, ADVANCE_THIS, min2,dMN);
cannam@0 437 }
cannam@0 438 } else {
cannam@0 439 int min1 = getValue(frameCount, index-1, true);
cannam@0 440 int min2 = getValue(frameCount - 1, index, true);
cannam@0 441 int min3 = getValue(frameCount - 1, index-1, true);
cannam@0 442 if (min1 <= min2) {
cannam@0 443 if (min3 + dMN <= min1)
cannam@0 444 setValue(frameCount, index, ADVANCE_BOTH, min3,dMN);
cannam@0 445 else
cannam@0 446 setValue(frameCount, index, ADVANCE_OTHER,min1,dMN);
cannam@0 447 } else {
cannam@0 448 if (min3 + dMN <= min2)
cannam@0 449 setValue(frameCount, index, ADVANCE_BOTH, min3,dMN);
cannam@0 450 else
cannam@0 451 setValue(frameCount, index, ADVANCE_THIS, min2,dMN);
cannam@0 452 }
cannam@0 453 }
cannam@0 454 otherMatcher->last[index]++;
cannam@0 455 } // loop for row (resp. column)
cannam@0 456
cannam@0 457 frameCount++;
cannam@0 458 runCount++;
cannam@0 459
cannam@0 460 otherMatcher->runCount = 0;
cannam@0 461
cannam@0 462 if (overflow && !silent)
cannam@0 463 cerr << "WARNING: overflow in distance metric: "
cannam@0 464 << "frame " << frameCount << ", val = " << mx << endl;
Chris@21 465
cannam@0 466 if (!silent)
cannam@0 467 std::cerr << "Frame " << frameCount << ", d = " << (mx-mn) << std::endl;
Chris@21 468 }
cannam@0 469
cannam@0 470 int
cannam@0 471 Matcher::getValue(int i, int j, bool firstAttempt)
cannam@0 472 {
cannam@0 473 if (firstPM)
cannam@0 474 return bestPathCost[i][j - first[i]];
cannam@0 475 else
cannam@0 476 return otherMatcher->bestPathCost[j][i - otherMatcher->first[j]];
cannam@0 477 } // getValue()
cannam@0 478
cannam@0 479 void
cannam@0 480 Matcher::setValue(int i, int j, int dir, int value, int dMN)
cannam@0 481 {
cannam@0 482 if (firstPM) {
cannam@0 483 distance[i][j - first[i]] = (unsigned char)((dMN & MASK) | dir);
cannam@0 484 bestPathCost[i][j - first[i]] =
cannam@0 485 (value + (dir==ADVANCE_BOTH? dMN*2: dMN));
cannam@0 486 } else {
cannam@0 487 if (dir == ADVANCE_THIS)
cannam@0 488 dir = ADVANCE_OTHER;
cannam@0 489 else if (dir == ADVANCE_OTHER)
cannam@0 490 dir = ADVANCE_THIS;
cannam@0 491 int idx = i - otherMatcher->first[j];
cannam@0 492 if (idx == (int)otherMatcher->distYSizes[j]) {
cannam@0 493 // This should never happen, but if we allow arbitrary
cannam@0 494 // pauses in either direction, and arbitrary lengths at
cannam@0 495 // end, it is better than a segmentation fault.
cannam@0 496 std::cerr << "Emergency resize: " << idx << " -> " << idx * 2 << std::endl;
cannam@0 497 otherMatcher->distYSizes[j] = idx * 2;
cannam@0 498 otherMatcher->bestPathCost[j] =
cannam@0 499 (int *)realloc(otherMatcher->bestPathCost[j],
cannam@0 500 idx * 2 * sizeof(int));
cannam@0 501 otherMatcher->distance[j] =
cannam@0 502 (unsigned char *)realloc(otherMatcher->distance[j],
cannam@0 503 idx * 2 * sizeof(unsigned char));
cannam@0 504 }
cannam@0 505 otherMatcher->distance[j][idx] = (unsigned char)((dMN & MASK) | dir);
cannam@0 506 otherMatcher->bestPathCost[j][idx] =
cannam@0 507 (value + (dir==ADVANCE_BOTH? dMN*2: dMN));
cannam@0 508 }
cannam@0 509 } // setValue()
cannam@0 510