On Sat, Mar 10, 2007 at 04:23:55PM +0100, Martin Sandve Alnæs wrote:
2007/3/9, Garth N. Wells <g.n.wells@xxxxxxxxxx>:
Anders Logg wrote:
On Fri, Mar 09, 2007 at 07:10:31PM +0100, Garth N. Wells wrote:
Anders Logg wrote:
On Fri, Mar 09, 2007 at 03:07:43PM +0100, Garth N. Wells wrote:
Why do we need a class AssemblyMatrix? GenericMatrix/GenericVector
should provide the necessary virtual functions for assembly.
Garth
Yes, definitely. The assembler assumes it gets a GenericTensor that
has a function
void add(real* block, uint* size, uint** entries)
(entries is probably a bad choice of name here, let's change it)
This makes it possible to implement just one assembly function and
keep the Assembler class small.
Then GenericVector and GenericMatrix should then derive from GenericTensor?
Yes. So far it is just two functions that are needed, init() and add().
Anyway, the class AssemblyMatrix is just a temporary implementation of
something that implements the GenericTensor interface. My suggestion
is that we overload the add() function above in both GenericMatrix and
GenericVector. The overloaded functions will map to the correct add()
function in the respective classes. For GenericMatrix, we'll have
void add(real* block, uint* size, uint** entries)
{
add(block, size[0], entries[0], size[1], entries[1]);
}
So, even if AssemblyMatrix could be of some use (it seems to be a fast
option for assembly, the STL vector of maps beat PETSc in at least one
benchmark I made)
Don't draw conclusions too quickly. STL vector of maps is fast to
assemble, but is close to useless for linear algebra. The current
implementation for the assembly uBLAS matrices use something like STL
vector of maps (fast to assemble but slow for linear algebra) and
converts it to compressed row storage (fast for linear algebra). The
conversion can take some time. A STL vector of integer maps should be
created and then used to initialise a particular sparse storage format
(before first use only). Then we wouldn't have to convert matrix formats.
Do you mean this should be done in DofMap?
Yes. The STL vector of maps is cheap to construct, and can be used to
initialise the non-zero layout of a sparse matrix.
You should use vector<set<int>>, not vector<map<int>>, if the goal is
only to build the sparsity pattern, as this gives lower memory
overhead since the set is a map with no value.
Yes, we use vector<set<int>> for the sparsity pattern, but in the
class AssemblyMatrix (which is something I use temporarily until we
make GenericMatrix and GenericTensor subclasses of GenericTensor) I
have used vector<map<int>> to represent the sparse matrix.
Anders: I have an implementation of this in
pycc/MatSparse/CRSGraph.cpp and pycc/FEMAssembler/CRSAssembler.cpp,
feel free to use any suitable parts of it.
We (Garth) has an implementation of this in the class OldDofMap in
DOLFIN that should be merged with the "new" class DofMap for UFC-based
assembly. It looks very similar to what you have in
CRSAssembler/CRSGraph:
for (CellIterator cell(*mesh); !cell.end(); ++cell)
{
element[0]->nodemap(row_dof, *cell, *mesh);
for (uint i = 0; i < element[0]->spacedim(); ++i)
{
element[1]->nodemap(column_dof, *cell, *mesh);
for (uint j = 0; j < element[1]->spacedim(); ++j)
matrix_sparsity_pattern[ row_dof[i] ].insert( column_dof[j] );
}
}
Here, matrix_sparsity_pattern is a vector<set<int>> and the nodemap
function is the old (non-UFC) name of the tabulate_dofs() function.
Is this the same as your implementation? (Other than splitting the
insertion into a separate call insert_block()?)