⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 cholsplit.c

📁 斯坦福大学Grant和Boyd教授等开发的凸优化matlab工具箱
💻 C
字号:
/* L.split = cholsplit(L, cachesiz)    This file is part of SeDuMi 1.04a    Copyright (C) 2000 Jos F. Sturm    Dept. Quantitative Economics, Maastricht University, the Netherlands.    Affiliations up to SeDuMi 1.02 (AUG1998):      CRL, McMaster University, Canada.      Supported by the Netherlands Organization for Scientific Research (NWO).      This program is free software; you can redistribute it and/or modify    it under the terms of the GNU General Public License as published by    the Free Software Foundation; either version 2 of the License, or    (at your option) any later version.      This program is distributed in the hope that it will be useful,    but WITHOUT ANY WARRANTY; without even the implied warranty of    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the    GNU General Public License for more details.      You should have received a copy of the GNU General Public License    along with this program; if not, write to the Free Software    Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.*/#define SPLIT_OUT plhs[0]#define NPAROUT 1#define L_IN prhs[0]       /* symbolic Cholesky structure: {L.L, L.xsuper} */#define CACHESIZ_IN prhs[1]       /* number of KBs cache, e.g. 256 */#define NPARIN 2#include "mex.h"/* ************************************************************   PROCEDURE getsplit - Compute splitting of supernodes, such that     the columns of the splitted supernode fit entirely into the     computer-cache (whenever possible). Will reduce memory-retrieval     time in BLKCHOL2.   INPUT     ljc,lir - sparsity structure of m x m matrix L (not compressed).     xsuper,nsuper - supernodal partition of nodes 1:m.     cachesiz - should be approx. 90% of cache-size in *DOUBLES* (NOT KBs !).   OUTPUT     split - length m integer array. For the start of each splitted       supernode j. Thus, 1 <= split[j] < xsuper[snode[j] + 1].       For intermediate nodes, split[j] = 0.   ************************************************************ */void getsplit(int *split, const int *ljc,const int *lir,const int *xsuper,              const int nsuper, const int cachesiz){  int j,k,ksup,mk, used,nextk;/* ------------------------------------------------------------   For each supernode ksup = 1:nsuper, column k=1:m.   ------------------------------------------------------------ */  k = 0;  for(ksup = 0; ksup < nsuper; ksup++){    mk = ljc[k+1] - ljc[k];                /* length of column k */    used = 2 * mk;                         /* 1st col counts twice */    nextk = xsuper[ksup + 1];    j = k;                                 /* j = start cache-group *//* ------------------------------------------------------------   If 1st column in ksup is too big to fit into cache, then pick   columns together until we arrive at a column k that fits.   ------------------------------------------------------------ */    if(used > cachesiz){      k = j + (used - cachesiz) / 2;       /* k is 1st col that fits in cache*/      if(k >= nextk)        k = nextk;                         /* all cols in ksup too long */      else{        mk -= k-j;        used = 2*mk;                       /* start new cache-group at k */      }      split[j] = k-j;      j = k;    }    else{      j = k;      k++;      mk--;    }/* ------------------------------------------------------------   Split remainder of supernode into cache-groups that fit into cache.   ------------------------------------------------------------ */    for(;k < nextk; k++, mk--)      if(used + mk < cachesiz)        used += mk;                    /* add into current cache-group */      else{        split[j] = k-j;                /* close cache group at previous col. */        j = k;                         /* start new cache group */        used = 2 * mk;                 /* insert its first column */      }/* ------------------------------------------------------------   Close last cache-group in this supernode.   ------------------------------------------------------------ */    if(j < nextk){      split[j] = nextk-j;              /* last cache group in this supernode */    }  }}/* ============================================================   MEXFUNCTION   ============================================================ *//* ************************************************************   PROCEDURE mexFunction - Entry for Matlab   ************************************************************ */void mexFunction(const int nlhs, mxArray *plhs[],                 const int nrhs, const mxArray *prhs[]){  const mxArray *L_FIELD;  int i,j, nsuper,m, cachesiz;  const int *ljc,*lir;  int *xsuper, *split;  const double *xsuperPr;  double *splitPr;/* ------------------------------------------------------------   Check for proper number of arguments   ------------------------------------------------------------ */  if(nrhs < NPARIN)    mexErrMsgTxt("cholsplit requires more input arguments.");  if(nlhs > NPAROUT)    mexErrMsgTxt("cholsplit produces less output arguments.");/* ------------------------------------------------------------   Get cachesiz, and transform from KBs into 90% of FLOATS.   ------------------------------------------------------------ */  cachesiz = 0.9 * (1024 / sizeof(double)) * mxGetScalar(CACHESIZ_IN);/* ------------------------------------------------------------   Disassemble block Cholesky structure L   ------------------------------------------------------------ */  if(!mxIsStruct(L_IN))    mexErrMsgTxt("Parameter `L' should be a structure.");  if( (L_FIELD = mxGetField(L_IN,0,"L")) == NULL)           /* L.L */    mexErrMsgTxt("Missing field L.L.");  m = mxGetM(L_FIELD);  if(m != mxGetN(L_FIELD) )    mexErrMsgTxt("L.L must be square.");  if(!mxIsSparse(L_FIELD))    mexErrMsgTxt("L.L should be sparse.");  ljc = mxGetJc(L_FIELD);  lir = mxGetIr(L_FIELD);  if( (L_FIELD = mxGetField(L_IN,0,"xsuper")) == NULL)      /* L.xsuper */    mexErrMsgTxt("Missing field L.xsuper.");  nsuper = mxGetM(L_FIELD) * mxGetN(L_FIELD) - 1;  if( nsuper > m )    mexErrMsgTxt("Size L.xsuper mismatch.");  xsuperPr = mxGetPr(L_FIELD);/* ------------------------------------------------------------   Allocate working arrays:   ------------------------------------------------------------ */  xsuper    = (int *) mxCalloc(nsuper+1,sizeof(int));  split     = (int *) mxCalloc(m,sizeof(int));/* ------------------------------------------------------------   Convert XSUPER to integer and C-Style   ------------------------------------------------------------ */  for(i = 0; i <= nsuper; i++){    j =  xsuperPr[i];    xsuper[i] = --j;  }/* ------------------------------------------------------------   The main job: compute (upper bound on) blkchol-split.   ------------------------------------------------------------ */  getsplit(split, ljc,lir,xsuper,nsuper, cachesiz);/* ------------------------------------------------------------   create OUTPUT variable SPLIT(m)   ------------------------------------------------------------ */  SPLIT_OUT = mxCreateDoubleMatrix(m, 1, mxREAL);          /* L.split */  splitPr = mxGetPr(SPLIT_OUT);  for(i = 0; i < m; i += j){    j = split[i];    splitPr[i] = j;  }/* ------------------------------------------------------------   Release working arrays.   ------------------------------------------------------------ */  mxFree(split);  mxFree(xsuper);}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -