// // dim.cc // // Copyright (C) 1996 Limit Point Systems, Inc. // // Author: Curtis Janssen // Maintainer: LPS // // This file is part of the SC Toolkit. // // The SC Toolkit is free software; you can redistribute it and/or modify // it under the terms of the GNU Library General Public License as published by // the Free Software Foundation; either version 2, or (at your option) // any later version. // // The SC Toolkit 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 Library General Public License for more details. // // You should have received a copy of the GNU Library General Public License // along with the SC Toolkit; see the file COPYING.LIB. If not, write to // the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. // // The U.S. Government is granted a limited license as per AL 91-7. // #ifdef __GNUC__ #pragma implementation #endif #include #include #include #include #include #include using namespace std; using namespace sc; static void fail(const char *s) { ExEnv::errn() << indent << "math/scmat/dim.cc: " << s << endl; abort(); } ///////////////////////////////////////////////////////////////////////////// // SCBlockInfo member functions static ClassDesc SCBlockInfo_cd( typeid(SCBlockInfo),"SCBlockInfo",1,"public SavableState", 0, create, create); SCBlockInfo::SCBlockInfo(int n, int nblocks, const int *blocksizes): subdims_(0) { n_ = n; nblocks_ = nblocks; size_ = 0; if (n_ == 0) nblocks_ = 0; if (n_ != 0 && nblocks_ == 0) { nblocks_ = 1; size_ = new int[1]; size_[0] = n; } else if (nblocks_ == 0) { size_ = 0; } else { int i; size_ = new int[nblocks_]; if (blocksizes) { for (i=0; i&keyval): subdims_(0) { nblocks_ = keyval->count("sizes"); n_ = 0; size_ = new int[nblocks_]; for (int i=0; iintvalue("sizes",i); n_ += size_[i]; } int nsubdims = keyval->count("subdims"); if (nsubdims) { if (nblocks_ != 0 && nsubdims != nblocks_) { fail("SCBlockInfo(const Ref&): nsubdims != nblocks"); } subdims_ = new RefSCDimension[nsubdims]; for (int i=0; idescribedclassvalue("subdims",i); } if (nblocks_ == 0) { delete[] size_; size_ = new int[nsubdims]; for (int i=0; in_) return 0; if (nblocks_ != bi->nblocks_) return 0; int i; for (i=0; istart_[i]) return 0; } if (subdims_) { if (bi->subdims_) { for (i=0; isubdims_[i].nonnull()) { if (!subdims_[i]->equiv(bi->subdims_[i])) return 0; } else return 0; } else if (bi->subdims_[i].nonnull()) return 0; } } else { return 0; } } else if (bi->subdims_) { return 0; } return 1; } void SCBlockInfo::elem_to_block(int elem, int &block, int &offset) { for (int i=nblocks_-1; i>=0; i--) { if (start_[i] <= elem) { block = i; offset = elem-start_[i]; return; } } fail("SCBlockInfo::elem_to_block: couldn't find block"); } SCBlockInfo::~SCBlockInfo() { delete[] start_; delete[] size_; delete[] subdims_; } RefSCDimension SCBlockInfo::subdim(int i) { if (i >= nblocks_ || i < 0) { fail("SCBlockInfo::subdim: bad block index"); } if (!subdims_) return 0; return subdims_[i]; } void SCBlockInfo::set_subdim(int i, const RefSCDimension &dim) { if (!subdims_) { subdims_ = new RefSCDimension[nblocks_]; } if (i >= nblocks_ || i < 0) { fail("SCBlockInfo::set_subdim: bad block index"); } if (size(i) != dim.n()) { fail("SCBlockInfo::set_subdim: size mismatch"); } subdims_[i] = dim; } void SCBlockInfo::print(ostream&o) const { indent(o); o << "nblocks = " << nblocks_ << endl; indent(o); o << "sizes ="; for (int i=0; i, create); SCDimension::SCDimension(const char* name): n_(0) { if (name) name_ = strcpy(new char[strlen(name)+1], name); else name_ = 0; } SCDimension::SCDimension(int n, const char* name): n_(n) { if (name) name_ = strcpy(new char[strlen(name)+1], name); else name_ = 0; blocks_ = new SCBlockInfo(n, 1); } SCDimension::SCDimension(const Ref& b, const char* name): n_(b->nelem()), blocks_(b) { if (name) name_ = strcpy(new char[strlen(name)+1], name); else name_ = 0; } SCDimension::SCDimension(int n, int nblocks, const int *blocksizes, const char* name): n_(n) { if (name) name_ = strcpy(new char[strlen(name)+1], name); else name_ = 0; blocks_ = new SCBlockInfo(n, nblocks, blocksizes); } SCDimension::SCDimension(const Ref& keyval) { blocks_ << keyval->describedclassvalue("blocks"); n_ = keyval->intvalue("n"); if (blocks_.null()) { if (keyval->error() != KeyVal::OK) { fail("SCDimension(const Ref&): missing input"); } blocks_ = new SCBlockInfo(n_); } else { if (n_ != 0 && n_ != blocks_->nelem()) { fail("SCDimension(const Ref&): inconsistent sizes"); } n_ = blocks_->nelem(); } name_ = keyval->pcharvalue("name"); } SCDimension::SCDimension(StateIn&s): SavableState(s) { s.getstring(name_); s.get(n_); blocks_ << SavableState::restore_state(s); } void SCDimension::save_data_state(StateOut&s) { s.putstring(name_); s.put(n_); SavableState::save_state(blocks_.pointer(), s); } SCDimension::~SCDimension() { if (name_) delete[] name_; } int SCDimension::equiv(const SCDimension *a) const { if (n_ != a->n_) return 0; if (!blocks_->equiv(a->blocks_)) return 0; return 1; } void SCDimension::print(ostream&o) const { indent(o); o << "n = " << n_; if (name_) { o << ", name = " << name_; } o << endl; incindent(o); if (blocks_.nonnull()) blocks_->print(o); decindent(o); } ///////////////////////////////////////////////////////////////////////////// // RefSCDimension member functions void RefSCDimension::print(ostream&o) const { if (null()) { indent(o); o << "n = 0" << endl; } else pointer()->print(o); } ///////////////////////////////////////////////////////////////////////////// // Local Variables: // mode: c++ // c-file-style: "CLJ" // End: