annotate ffmpeg/libavutil/lls.c @ 11:f445c3017523

new files
author Yading Song <yading.song@eecs.qmul.ac.uk>
date Sun, 21 Apr 2013 11:16:23 +0200
parents
children
rev   line source
yading@11 1 /*
yading@11 2 * linear least squares model
yading@11 3 *
yading@11 4 * Copyright (c) 2006 Michael Niedermayer <michaelni@gmx.at>
yading@11 5 *
yading@11 6 * This file is part of FFmpeg.
yading@11 7 *
yading@11 8 * FFmpeg is free software; you can redistribute it and/or
yading@11 9 * modify it under the terms of the GNU Lesser General Public
yading@11 10 * License as published by the Free Software Foundation; either
yading@11 11 * version 2.1 of the License, or (at your option) any later version.
yading@11 12 *
yading@11 13 * FFmpeg is distributed in the hope that it will be useful,
yading@11 14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
yading@11 15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
yading@11 16 * Lesser General Public License for more details.
yading@11 17 *
yading@11 18 * You should have received a copy of the GNU Lesser General Public
yading@11 19 * License along with FFmpeg; if not, write to the Free Software
yading@11 20 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
yading@11 21 */
yading@11 22
yading@11 23 /**
yading@11 24 * @file
yading@11 25 * linear least squares model
yading@11 26 */
yading@11 27
yading@11 28 #include <math.h>
yading@11 29 #include <string.h>
yading@11 30
yading@11 31 #include "version.h"
yading@11 32 #include "lls.h"
yading@11 33
yading@11 34 void avpriv_init_lls(LLSModel *m, int indep_count)
yading@11 35 {
yading@11 36 memset(m, 0, sizeof(LLSModel));
yading@11 37 m->indep_count = indep_count;
yading@11 38 }
yading@11 39
yading@11 40 void avpriv_update_lls(LLSModel *m, double *var, double decay)
yading@11 41 {
yading@11 42 int i, j;
yading@11 43
yading@11 44 for (i = 0; i <= m->indep_count; i++) {
yading@11 45 for (j = i; j <= m->indep_count; j++) {
yading@11 46 m->covariance[i][j] *= decay;
yading@11 47 m->covariance[i][j] += var[i] * var[j];
yading@11 48 }
yading@11 49 }
yading@11 50 }
yading@11 51
yading@11 52 void avpriv_solve_lls(LLSModel *m, double threshold, unsigned short min_order)
yading@11 53 {
yading@11 54 int i, j, k;
yading@11 55 double (*factor)[MAX_VARS + 1] = (void *) &m->covariance[1][0];
yading@11 56 double (*covar) [MAX_VARS + 1] = (void *) &m->covariance[1][1];
yading@11 57 double *covar_y = m->covariance[0];
yading@11 58 int count = m->indep_count;
yading@11 59
yading@11 60 for (i = 0; i < count; i++) {
yading@11 61 for (j = i; j < count; j++) {
yading@11 62 double sum = covar[i][j];
yading@11 63
yading@11 64 for (k = i - 1; k >= 0; k--)
yading@11 65 sum -= factor[i][k] * factor[j][k];
yading@11 66
yading@11 67 if (i == j) {
yading@11 68 if (sum < threshold)
yading@11 69 sum = 1.0;
yading@11 70 factor[i][i] = sqrt(sum);
yading@11 71 } else {
yading@11 72 factor[j][i] = sum / factor[i][i];
yading@11 73 }
yading@11 74 }
yading@11 75 }
yading@11 76
yading@11 77 for (i = 0; i < count; i++) {
yading@11 78 double sum = covar_y[i + 1];
yading@11 79
yading@11 80 for (k = i - 1; k >= 0; k--)
yading@11 81 sum -= factor[i][k] * m->coeff[0][k];
yading@11 82
yading@11 83 m->coeff[0][i] = sum / factor[i][i];
yading@11 84 }
yading@11 85
yading@11 86 for (j = count - 1; j >= min_order; j--) {
yading@11 87 for (i = j; i >= 0; i--) {
yading@11 88 double sum = m->coeff[0][i];
yading@11 89
yading@11 90 for (k = i + 1; k <= j; k++)
yading@11 91 sum -= factor[k][i] * m->coeff[j][k];
yading@11 92
yading@11 93 m->coeff[j][i] = sum / factor[i][i];
yading@11 94 }
yading@11 95
yading@11 96 m->variance[j] = covar_y[0];
yading@11 97
yading@11 98 for (i = 0; i <= j; i++) {
yading@11 99 double sum = m->coeff[j][i] * covar[i][i] - 2 * covar_y[i + 1];
yading@11 100
yading@11 101 for (k = 0; k < i; k++)
yading@11 102 sum += 2 * m->coeff[j][k] * covar[k][i];
yading@11 103
yading@11 104 m->variance[j] += m->coeff[j][i] * sum;
yading@11 105 }
yading@11 106 }
yading@11 107 }
yading@11 108
yading@11 109 double avpriv_evaluate_lls(LLSModel *m, double *param, int order)
yading@11 110 {
yading@11 111 int i;
yading@11 112 double out = 0;
yading@11 113
yading@11 114 for (i = 0; i <= order; i++)
yading@11 115 out += param[i] * m->coeff[order][i];
yading@11 116
yading@11 117 return out;
yading@11 118 }
yading@11 119
yading@11 120 #if FF_API_LLS_PRIVATE
yading@11 121 void av_init_lls(LLSModel *m, int indep_count)
yading@11 122 {
yading@11 123 avpriv_init_lls(m, indep_count);
yading@11 124 }
yading@11 125 void av_update_lls(LLSModel *m, double *param, double decay)
yading@11 126 {
yading@11 127 avpriv_update_lls(m, param, decay);
yading@11 128 }
yading@11 129 void av_solve_lls(LLSModel *m, double threshold, int min_order)
yading@11 130 {
yading@11 131 avpriv_solve_lls(m, threshold, min_order);
yading@11 132 }
yading@11 133 double av_evaluate_lls(LLSModel *m, double *param, int order)
yading@11 134 {
yading@11 135 return avpriv_evaluate_lls(m, param, order);
yading@11 136 }
yading@11 137 #endif /* FF_API_LLS_PRIVATE */
yading@11 138
yading@11 139 #ifdef TEST
yading@11 140
yading@11 141 #include <stdio.h>
yading@11 142 #include <limits.h>
yading@11 143 #include "lfg.h"
yading@11 144
yading@11 145 int main(void)
yading@11 146 {
yading@11 147 LLSModel m;
yading@11 148 int i, order;
yading@11 149 AVLFG lfg;
yading@11 150
yading@11 151 av_lfg_init(&lfg, 1);
yading@11 152 avpriv_init_lls(&m, 3);
yading@11 153
yading@11 154 for (i = 0; i < 100; i++) {
yading@11 155 double var[4];
yading@11 156 double eval;
yading@11 157
yading@11 158 var[0] = (av_lfg_get(&lfg) / (double) UINT_MAX - 0.5) * 2;
yading@11 159 var[1] = var[0] + av_lfg_get(&lfg) / (double) UINT_MAX - 0.5;
yading@11 160 var[2] = var[1] + av_lfg_get(&lfg) / (double) UINT_MAX - 0.5;
yading@11 161 var[3] = var[2] + av_lfg_get(&lfg) / (double) UINT_MAX - 0.5;
yading@11 162 avpriv_update_lls(&m, var, 0.99);
yading@11 163 avpriv_solve_lls(&m, 0.001, 0);
yading@11 164 for (order = 0; order < 3; order++) {
yading@11 165 eval = avpriv_evaluate_lls(&m, var + 1, order);
yading@11 166 printf("real:%9f order:%d pred:%9f var:%f coeffs:%f %9f %9f\n",
yading@11 167 var[0], order, eval, sqrt(m.variance[order] / (i + 1)),
yading@11 168 m.coeff[order][0], m.coeff[order][1],
yading@11 169 m.coeff[order][2]);
yading@11 170 }
yading@11 171 }
yading@11 172 return 0;
yading@11 173 }
yading@11 174
yading@11 175 #endif