![]() |
Mesh Oriented datABase
(version 5.4.1)
Array-based unstructured mesh datastructure
|
00001 /**
00002 * \class ReadTemplate
00003 * \brief Template for writing a new reader in MOAB
00004 *
00005 */
00006
00007 #include "ReadTemplate.hpp"
00008 #include "moab/Interface.hpp"
00009 #include "moab/ReadUtilIface.hpp"
00010 #include "moab/Range.hpp"
00011 #include "moab/FileOptions.hpp"
00012
00013 #include
00014 #include
00015
00016 namespace moab
00017 {
00018
00019 ReaderIface* ReadTemplate::factory( Interface* iface )
00020 {
00021 return new ReadTemplate( iface );
00022 }
00023
00024 ReadTemplate::ReadTemplate( Interface* impl ) : mbImpl( impl ), fileName( NULL )
00025 {
00026 mbImpl->query_interface( readMeshIface );
00027 }
00028
00029 ReadTemplate::~ReadTemplate()
00030 {
00031 if( readMeshIface )
00032 {
00033 mbImpl->release_interface( readMeshIface );
00034 readMeshIface = 0;
00035 }
00036 }
00037
00038 ErrorCode ReadTemplate::read_tag_values( const char* /* file_name */,
00039 const char* /* tag_name */,
00040 const FileOptions& /* opts */,
00041 std::vector< int >& /* tag_values_out */,
00042 const SubsetList* /* subset_list */ )
00043 {
00044 return MB_NOT_IMPLEMENTED;
00045 }
00046
00047 ErrorCode ReadTemplate::load_file( const char* filename,
00048 const EntityHandle* file_set,
00049 const FileOptions& opts,
00050 const ReaderIface::SubsetList* subset_list,
00051 const Tag* /*file_id_tag*/ )
00052 {
00053 if( subset_list )
00054 {
00055 // See src/moab/ReaderIface.hpp, definition of SubsetList struct; this basically specifies
00056 // an integer tag and tag values for sets to read on this proc, or a part number and total #
00057 // parts for reading a trivial partition of entities
00058 }
00059
00060 // Save filename to member variable so we don't need to pass as an argument
00061 // to called functions
00062 fileName = filename;
00063
00064 // Process options; see src/FileOptions.hpp for API for FileOptions class, and
00065 // doc/metadata_info.doc for a description of various options used by some of the readers in
00066 // MOAB
00067 ErrorCode result = process_options( opts );MB_CHK_SET_ERR( result, fileName << ": problem reading options" );
00068
00069 // Open file; filePtr is member of ReadTemplate, change to whatever mechanism is used to
00070 // identify file
00071 FILE* filePtr = fopen( fileName, "r" );
00072 if( !filePtr )
00073 {
00074 MB_SET_ERR( MB_FILE_DOES_NOT_EXIST, fileName << ": fopen returned error" );
00075 }
00076
00077 // Read number of verts, elements, sets
00078 long num_verts = 0, num_elems = 0, num_sets = 0;
00079
00080 // read_ents keeps a running set of entities read from this file, including vertices, elements,
00081 // and sets; these will get added to file_set (if input) at the end of the read
00082 Range read_ents;
00083
00084 // start_vertex is passed back so we know how to convert indices from the file into vertex
00085 // handles; most of the time this is done by adding start_vertex to the (0-based) index; if the
00086 // index is 1-based, you also need to subtract one; see read_elements for details
00087 EntityHandle start_vertex;
00088 result = read_vertices( num_verts, start_vertex, read_ents );
00089 if( MB_SUCCESS != result )
00090 {
00091 fclose( filePtr );
00092 return result;
00093 }
00094
00095 // Create/read elements; this template assumes that all elements are the same type, so can be
00096 // read in a single call to read_elements, and kept track of with a single start_elem handle. If
00097 // there are more entity types, might have to keep these start handles in an array/vector.
00098 // start_elem is only really needed if you're reading sets later, and need to convert some
00099 // file-based index to an entity handle
00100 EntityHandle start_elem;
00101 result = read_elements( num_elems, start_vertex, start_elem, read_ents );
00102 if( MB_SUCCESS != result )
00103 {
00104 fclose( filePtr );
00105 return result;
00106 }
00107
00108 // Read/create entity sets; typically these sets have some tag identifying what they're for, see
00109 // doc/metadata_info.doc for examples of different kinds of sets and how they're marked
00110 result = create_sets( num_sets, start_vertex, num_verts, start_elem, num_elems, read_ents );
00111 if( MB_SUCCESS != result )
00112 {
00113 fclose( filePtr );
00114 return result;
00115 }
00116
00117 // Finally, add all read_ents into the file set, if one was input
00118 if( file_set && *file_set )
00119 {
00120 result = mbImpl->add_entities( *file_set, read_ents );
00121 if( MB_SUCCESS != result )
00122 {
00123 fclose( filePtr );
00124 return result;
00125 }
00126 }
00127
00128 fclose( filePtr );
00129
00130 return result;
00131 }
00132
00133 ErrorCode ReadTemplate::read_vertices( int num_verts, EntityHandle& start_vertex, Range& read_ents )
00134 {
00135 // Allocate nodes; these are allocated in one shot, get contiguous handles starting with
00136 // start_handle, and the reader is passed back double*'s pointing to MOAB's native storage for
00137 // vertex coordinates for those verts
00138 std::vector< double* > coord_arrays;
00139 ErrorCode result = readMeshIface->get_node_coords( 3, num_verts, 1, start_vertex, coord_arrays );MB_CHK_SET_ERR( result, fileName << ": Trouble reading vertices" );
00140
00141 // Fill in vertex coordinate arrays
00142 double *x = coord_arrays[0], *y = coord_arrays[1], *z = coord_arrays[2];
00143 for( long i = 0; i < num_verts; ++i )
00144 {
00145 // Read x/y/z; do something with them for now to avoid warning
00146 if( x || y || z )
00147 {
00148 }
00149 }
00150
00151 if( num_verts ) read_ents.insert( start_vertex, start_vertex + num_verts - 1 );
00152
00153 return result;
00154 }
00155
00156 //! Read/create elements
00157 ErrorCode ReadTemplate::read_elements( int num_elems,
00158 EntityHandle start_vertex,
00159 EntityHandle& start_elem,
00160 Range& read_ents )
00161 {
00162 // Get the entity type being read
00163 EntityType ent_type = MBHEX;
00164
00165 // Get the number of vertices per entity
00166 int verts_per_elem = 8;
00167
00168 // Create the element sequence; passes back a pointer to the internal storage for connectivity
00169 // and the starting entity handle
00170 EntityHandle* conn_array;
00171 ErrorCode result =
00172 readMeshIface->get_element_connect( num_elems, verts_per_elem, ent_type, 1, start_elem, conn_array );MB_CHK_SET_ERR( result, fileName << ": Trouble reading elements" );
00173
00174 // Read connectivity into conn_array directly
00175 for( long i = 0; i < num_elems; i++ )
00176 {
00177 // Read connectivity
00178 }
00179
00180 // Convert file-based connectivity indices to vertex handles in-place; be careful, if indices
00181 // are smaller than handles, need to do from the end of the list so we don't overwrite data
00182 //
00183 // Here, we assume indices are smaller than handles, just for demonstration; create an
00184 // integer-type pointer to connectivity initialized to same start of connectivity array
00185 int* ind_array = reinterpret_cast< int* >( conn_array );
00186 // OFFSET is value of first vertex index in file; most files are 1-based, but some might be
00187 // 0-based
00188 int OFFSET = 1;
00189 for( long i = num_elems * verts_per_elem - 1; i >= 0; i-- )
00190 {
00191 conn_array[i] = ind_array[i] + start_vertex + OFFSET;
00192
00193 // This assert assumes last handle in read_ents is highest vertex handle in this file
00194 assert( conn_array[i] >= start_vertex && conn_array[i] <= *read_ents.rbegin() );
00195 }
00196
00197 // Notify MOAB of the new elements
00198 result = readMeshIface->update_adjacencies( start_elem, num_elems, verts_per_elem, conn_array );
00199 if( MB_SUCCESS != result ) return result;
00200
00201 // Add elements to read_ents
00202 if( num_elems ) read_ents.insert( start_elem, start_elem + num_elems - 1 );
00203
00204 return MB_SUCCESS;
00205 }
00206
00207 //! Read/create sets
00208 ErrorCode ReadTemplate::create_sets( int num_sets,
00209 EntityHandle /*start_vertex*/,
00210 int /*num_verts*/,
00211 EntityHandle /*start_elem*/,
00212 int /*num_elems*/,
00213 Range& read_ents )
00214 {
00215 ErrorCode result = MB_SUCCESS;
00216 EntityHandle this_set;
00217
00218 for( int i = 0; i < num_sets; i++ )
00219 {
00220 // Create set
00221 result = mbImpl->create_meshset( MESHSET_SET, this_set );MB_CHK_SET_ERR( result, fileName << ": Trouble creating set" );
00222
00223 Range set_ents;
00224 // Read/compute what's in this set; REMEMBER TO CONVERT THESE TO MOAB HANDLES
00225
00226 // Add them to the set
00227 result = mbImpl->add_entities( this_set, set_ents );MB_CHK_SET_ERR( result, fileName << ": Trouble putting entities in set" );
00228
00229 // Add the new set to read_ents
00230 read_ents.insert( this_set );
00231 }
00232
00233 return MB_SUCCESS;
00234 }
00235
00236 ErrorCode ReadTemplate::process_options( const FileOptions& opts )
00237 {
00238 // Mark all options seen, to avoid compile warning on unused variable
00239 opts.mark_all_seen();
00240 return MB_SUCCESS;
00241 }
00242
00243 } // namespace moab