Page Menu
Home
c4science
Search
Configure Global Search
Log In
Files
F99186076
mesh_io_abaqus.cc
No One
Temporary
Actions
Download File
Edit File
Delete File
View Transforms
Subscribe
Mute Notifications
Award Token
Subscribers
None
File Metadata
Details
File Info
Storage
Attached
Created
Wed, Jan 22, 03:44
Size
17 KB
Mime Type
text/x-c++
Expires
Fri, Jan 24, 03:44 (2 d)
Engine
blob
Format
Raw Data
Handle
23715219
Attached To
rAKA akantu
mesh_io_abaqus.cc
View Options
/**
* @file mesh_io_abaqus.cc
*
* @author Nicolas Richart <nicolas.richart@epfl.ch>
*
* @date creation: Wed Jan 16 2013
* @date last modification: Fri Sep 19 2014
*
* @brief read a mesh from an abaqus input file
*
* @section LICENSE
*
* Copyright (©) 2014 EPFL (Ecole Polytechnique Fédérale de Lausanne)
* Laboratory (LSMS - Laboratoire de Simulation en Mécanique des Solides)
*
* Akantu is free software: you can redistribute it and/or modify it under the
* terms of the GNU Lesser General Public License as published by the Free
* Software Foundation, either version 3 of the License, or (at your option) any
* later version.
*
* Akantu 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 Lesser General Public License for more
* details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with Akantu. If not, see <http://www.gnu.org/licenses/>.
*
*/
/* -------------------------------------------------------------------------- */
// std library header files
#include <fstream>
// akantu header files
#include "mesh_io_abaqus.hh"
#include "mesh.hh"
#include "element_group.hh"
#include "node_group.hh"
/* -------------------------------------------------------------------------- */
#include <boost/config/warning_disable.hpp>
#include <boost/spirit/include/qi.hpp>
#include <boost/spirit/include/phoenix_core.hpp>
#include <boost/spirit/include/phoenix_fusion.hpp>
#include <boost/spirit/include/phoenix_object.hpp>
#include <boost/spirit/include/phoenix_container.hpp>
#include <boost/spirit/include/phoenix_operator.hpp>
#include <boost/spirit/include/phoenix_bind.hpp>
#include <boost/spirit/include/phoenix_stl.hpp>
/* -------------------------------------------------------------------------- */
__BEGIN_AKANTU__
/* -------------------------------------------------------------------------- */
MeshIOAbaqus
::
MeshIOAbaqus
()
{
}
/* -------------------------------------------------------------------------- */
MeshIOAbaqus
::~
MeshIOAbaqus
()
{
}
/* -------------------------------------------------------------------------- */
namespace
spirit
=
boost
::
spirit
;
namespace
qi
=
boost
::
spirit
::
qi
;
namespace
ascii
=
boost
::
spirit
::
ascii
;
namespace
lbs
=
boost
::
spirit
::
qi
::
labels
;
namespace
phx
=
boost
::
phoenix
;
namespace
mesh_io_abaqus_lazy_eval
{
struct
mesh_abaqus_error_handler_
{
template
<
typename
,
typename
,
typename
>
struct
result
{
typedef
void
type
;
};
template
<
typename
Iterator
>
void
operator
()(
qi
::
info
const
&
what
,
Iterator
err_pos
,
Iterator
last
)
const
{
AKANTU_EXCEPTION
(
"Error! Expecting "
<<
what
// what failed?
<<
" here:
\"
"
<<
std
::
string
(
err_pos
,
last
)
// iterators to error-pos, end
<<
"
\"
"
);
}
};
struct
lazy_element_read_
{
template
<
class
Mesh
,
class
ET
,
class
ID
,
class
V
,
class
NodeMap
,
class
ElemMap
>
struct
result
{
typedef
void
type
;
};
template
<
class
Mesh
,
class
ET
,
class
ID
,
class
V
,
class
NodeMap
,
class
ElemMap
>
void
operator
()(
Mesh
&
mesh
,
const
ET
&
type
,
const
ID
&
id
,
const
V
&
conn
,
const
NodeMap
&
nodes_mapping
,
ElemMap
&
elements_mapping
)
const
{
Vector
<
UInt
>
tmp_conn
(
Mesh
::
getNbNodesPerElement
(
type
));
AKANTU_DEBUG_ASSERT
(
conn
.
size
()
==
tmp_conn
.
size
(),
"The nodes in the Abaqus file have too many coordinates"
<<
" for the mesh you try to fill."
);
mesh
.
addConnectivityType
(
type
);
Array
<
UInt
>
&
connectivity
=
mesh
.
getConnectivity
(
type
);
UInt
i
=
0
;
for
(
typename
V
::
const_iterator
it
=
conn
.
begin
();
it
!=
conn
.
end
();
++
it
)
{
typename
NodeMap
::
const_iterator
nit
=
nodes_mapping
.
find
(
*
it
);
AKANTU_DEBUG_ASSERT
(
nit
!=
nodes_mapping
.
end
(),
"There is an unknown node in the connectivity."
);
tmp_conn
[
i
++
]
=
nit
->
second
;
}
Element
el
(
type
,
connectivity
.
getSize
());
elements_mapping
[
id
]
=
el
;
connectivity
.
push_back
(
tmp_conn
);
}
};
struct
lazy_node_read_
{
template
<
class
Mesh
,
class
ID
,
class
V
,
class
Map
>
struct
result
{
typedef
void
type
;
};
template
<
class
Mesh
,
class
ID
,
class
V
,
class
Map
>
void
operator
()(
Mesh
&
mesh
,
const
ID
&
id
,
const
V
&
pos
,
Map
&
nodes_mapping
)
const
{
Vector
<
Real
>
tmp_pos
(
mesh
.
getSpatialDimension
());
UInt
i
=
0
;
for
(
typename
V
::
const_iterator
it
=
pos
.
begin
();
it
!=
pos
.
end
()
||
i
<
mesh
.
getSpatialDimension
();
++
it
)
tmp_pos
[
i
++
]
=
*
it
;
nodes_mapping
[
id
]
=
mesh
.
getNbNodes
();
mesh
.
getNodes
().
push_back
(
tmp_pos
);
}
};
/* ------------------------------------------------------------------------ */
struct
lazy_element_group_create_
{
template
<
class
Mesh
,
class
S
>
struct
result
{
typedef
ElementGroup
&
type
;
};
template
<
class
Mesh
,
class
S
>
ElementGroup
&
operator
()(
Mesh
&
mesh
,
const
S
&
name
)
const
{
typename
Mesh
::
element_group_iterator
eg_it
=
mesh
.
element_group_find
(
name
);
if
(
eg_it
!=
mesh
.
element_group_end
())
{
return
*
eg_it
->
second
;
}
else
{
return
mesh
.
createElementGroup
(
name
,
_all_dimensions
);
}
}
};
struct
lazy_add_element_to_group_
{
template
<
class
EG
,
class
ID
,
class
Map
>
struct
result
{
typedef
void
type
;
};
template
<
class
EG
,
class
ID
,
class
Map
>
void
operator
()(
EG
*
el_grp
,
const
ID
&
element
,
const
Map
&
elements_mapping
)
const
{
typename
Map
::
const_iterator
eit
=
elements_mapping
.
find
(
element
);
AKANTU_DEBUG_ASSERT
(
eit
!=
elements_mapping
.
end
(),
"There is an unknown element ("
<<
element
<<
") in the in the ELSET "
<<
el_grp
->
getName
()
<<
"."
);
el_grp
->
add
(
eit
->
second
,
true
,
false
);
}
};
/* ------------------------------------------------------------------------ */
struct
lazy_node_group_create_
{
template
<
class
Mesh
,
class
S
>
struct
result
{
typedef
NodeGroup
&
type
;
};
template
<
class
Mesh
,
class
S
>
NodeGroup
&
operator
()(
Mesh
&
mesh
,
const
S
&
name
)
const
{
typename
Mesh
::
node_group_iterator
ng_it
=
mesh
.
node_group_find
(
name
);
if
(
ng_it
!=
mesh
.
node_group_end
())
{
return
*
ng_it
->
second
;
}
else
{
return
mesh
.
createNodeGroup
(
name
,
mesh
.
getSpatialDimension
());
}
}
};
struct
lazy_add_node_to_group_
{
template
<
class
NG
,
class
ID
,
class
Map
>
struct
result
{
typedef
void
type
;
};
template
<
class
NG
,
class
ID
,
class
Map
>
void
operator
()(
NG
*
node_grp
,
const
ID
&
node
,
const
Map
&
nodes_mapping
)
const
{
typename
Map
::
const_iterator
nit
=
nodes_mapping
.
find
(
node
);
AKANTU_DEBUG_ASSERT
(
nit
!=
nodes_mapping
.
end
(),
"There is an unknown node in the in the NSET "
<<
node_grp
->
getName
()
<<
"."
);
node_grp
->
add
(
nit
->
second
,
false
);
}
};
struct
lazy_optimize_group_
{
template
<
class
G
>
struct
result
{
typedef
void
type
;
};
template
<
class
G
>
void
operator
()(
G
*
grp
)
const
{
grp
->
optimize
();
}
};
}
/* -------------------------------------------------------------------------- */
template
<
class
Iterator
>
struct
AbaqusSkipper
:
qi
::
grammar
<
Iterator
>
{
AbaqusSkipper
()
:
AbaqusSkipper
::
base_type
(
skip
,
"abaqus_skipper"
)
{
skip
=
(
ascii
::
space
-
spirit
::
eol
)
|
"**"
>>
*
(
qi
::
char_
-
spirit
::
eol
)
>>
spirit
::
eol
;
}
qi
::
rule
<
Iterator
>
skip
;
};
/* -------------------------------------------------------------------------- */
template
<
class
Iterator
,
typename
Skipper
=
AbaqusSkipper
<
Iterator
>
>
struct
AbaqusMeshGrammar
:
qi
::
grammar
<
Iterator
,
void
(),
Skipper
>
{
public
:
AbaqusMeshGrammar
(
Mesh
&
mesh
)
:
AbaqusMeshGrammar
::
base_type
(
start
,
"abaqus_mesh_reader"
),
mesh
(
mesh
)
{
phx
::
function
<
mesh_io_abaqus_lazy_eval
::
mesh_abaqus_error_handler_
>
const
error_handler
=
mesh_io_abaqus_lazy_eval
::
mesh_abaqus_error_handler_
();
phx
::
function
<
mesh_io_abaqus_lazy_eval
::
lazy_element_read_
>
lazy_element_read
;
phx
::
function
<
mesh_io_abaqus_lazy_eval
::
lazy_node_read_
>
lazy_node_read
;
phx
::
function
<
mesh_io_abaqus_lazy_eval
::
lazy_element_group_create_
>
lazy_element_group_create
;
phx
::
function
<
mesh_io_abaqus_lazy_eval
::
lazy_add_element_to_group_
>
lazy_add_element_to_group
;
phx
::
function
<
mesh_io_abaqus_lazy_eval
::
lazy_node_group_create_
>
lazy_node_group_create
;
phx
::
function
<
mesh_io_abaqus_lazy_eval
::
lazy_add_node_to_group_
>
lazy_add_node_to_group
;
phx
::
function
<
mesh_io_abaqus_lazy_eval
::
lazy_optimize_group_
>
lazy_optimize_group
;
start
=
*
(
(
qi
::
char_
(
'*'
)
>
(
(
qi
::
no_case
[
qi
::
lit
(
"node output"
)
]
>
any_section
)
|
(
qi
::
no_case
[
qi
::
lit
(
"element output"
)
]
>
any_section
)
|
(
qi
::
no_case
[
qi
::
lit
(
"node"
)
]
>
nodes
)
|
(
qi
::
no_case
[
qi
::
lit
(
"element"
)
]
>
elements
)
|
(
qi
::
no_case
[
qi
::
lit
(
"heading"
)
]
>
header
)
|
(
qi
::
no_case
[
qi
::
lit
(
"elset"
)
]
>
elements_set
)
|
(
qi
::
no_case
[
qi
::
lit
(
"nset"
)
]
>
nodes_set
)
|
(
qi
::
no_case
[
qi
::
lit
(
"material"
)
]
>
material
)
|
(
keyword
>
any_section
)
)
)
|
spirit
::
eol
)
;
header
=
spirit
::
eol
>
*
any_line
;
nodes
=
*
(
qi
::
char_
(
','
)
>>
option
)
>>
spirit
::
eol
>>
*
(
(
qi
::
int_
>
node_position
)
[
lazy_node_read
(
phx
::
ref
(
mesh
),
lbs
::
_1
,
lbs
::
_2
,
phx
::
ref
(
abaqus_nodes_to_akantu
))
]
>>
spirit
::
eol
)
;
elements
=
(
(
qi
::
char_
(
','
)
>>
qi
::
no_case
[
qi
::
lit
(
"type"
)]
>>
'='
>>
abaqus_element_type
[
lbs
::
_a
=
lbs
::
_1
]
)
^
*
(
qi
::
char_
(
','
)
>>
option
)
)
>>
spirit
::
eol
>>
*
(
(
qi
::
int_
>
connectivity
)
[
lazy_element_read
(
phx
::
ref
(
mesh
),
lbs
::
_a
,
lbs
::
_1
,
lbs
::
_2
,
phx
::
cref
(
abaqus_nodes_to_akantu
),
phx
::
ref
(
abaqus_elements_to_akantu
))
]
>>
spirit
::
eol
)
;
elements_set
=
(
(
(
qi
::
char_
(
','
)
>>
qi
::
no_case
[
qi
::
lit
(
"elset"
)
]
>>
'='
>>
value
[
lbs
::
_a
=
&
lazy_element_group_create
(
phx
::
ref
(
mesh
),
lbs
::
_1
)
]
)
^
*
(
qi
::
char_
(
','
)
>>
option
)
)
>>
spirit
::
eol
>>
qi
::
skip
(
qi
::
char_
(
','
)
|
qi
::
space
)
[
+
(
qi
::
int_
[
lazy_add_element_to_group
(
lbs
::
_a
,
lbs
::
_1
,
phx
::
cref
(
abaqus_elements_to_akantu
)
)
]
)
]
)
[
lazy_optimize_group
(
lbs
::
_a
)
]
;
nodes_set
=
(
(
(
qi
::
char_
(
','
)
>>
qi
::
no_case
[
qi
::
lit
(
"nset"
)
]
>>
'='
>>
value
[
lbs
::
_a
=
&
lazy_node_group_create
(
phx
::
ref
(
mesh
),
lbs
::
_1
)
]
)
^
*
(
qi
::
char_
(
','
)
>>
option
)
)
>>
spirit
::
eol
>>
qi
::
skip
(
qi
::
char_
(
','
)
|
qi
::
space
)
[
+
(
qi
::
int_
[
lazy_add_node_to_group
(
lbs
::
_a
,
lbs
::
_1
,
phx
::
cref
(
abaqus_nodes_to_akantu
)
)
]
)
]
)
[
lazy_optimize_group
(
lbs
::
_a
)
]
;
material
=
(
(
qi
::
char_
(
','
)
>>
qi
::
no_case
[
qi
::
lit
(
"name"
)
]
>>
'='
>>
value
[
phx
::
push_back
(
phx
::
ref
(
material_names
),
lbs
::
_1
)
]
)
^
*
(
qi
::
char_
(
','
)
>>
option
)
)
>>
spirit
::
eol
;
;
node_position
=
+
(
qi
::
char_
(
','
)
>
real
[
phx
::
push_back
(
lbs
::
_val
,
lbs
::
_1
)
])
;
connectivity
=
+
(
qi
::
char_
(
','
)
>
qi
::
int_
[
phx
::
push_back
(
lbs
::
_val
,
lbs
::
_1
)
])
;
any_section
=
*
(
qi
::
char_
(
','
)
>>
option
)
>
spirit
::
eol
>
*
any_line
;
any_line
=
*
(
qi
::
char_
-
spirit
::
eol
-
qi
::
char_
(
'*'
))
>>
spirit
::
eol
;
keyword
=
qi
::
lexeme
[
+
(
qi
::
char_
-
(
qi
::
char_
(
'*'
)
|
spirit
::
eol
))
]
;
option
=
key
>
-
(
'='
>>
value
);
key
=
qi
::
char_
(
"a-zA-Z_"
)
>>
*
(
qi
::
char_
(
"a-zA-Z_0-9"
)
|
qi
::
char_
(
'-'
))
;
value
=
key
.
alias
()
;
BOOST_SPIRIT_DEBUG_NODE
(
start
);
abaqus_element_type
.
add
#if defined(AKANTU_STRUCTURAL_MECHANICS)
(
"BE21"
,
_bernoulli_beam_2
)
(
"BE31"
,
_bernoulli_beam_3
)
#endif
(
"T3D2"
,
_segment_2
)
// Gmsh generates this elements
(
"T3D3"
,
_segment_3
)
// Gmsh generates this elements
(
"CPE3"
,
_triangle_3
)
(
"CPS3"
,
_triangle_3
)
(
"DC2D3"
,
_triangle_3
)
(
"CPE6"
,
_triangle_6
)
(
"CPS6"
,
_triangle_6
)
(
"DC2D6"
,
_triangle_6
)
(
"CPE4"
,
_quadrangle_4
)
(
"CPS4"
,
_quadrangle_4
)
(
"DC2D4"
,
_quadrangle_4
)
(
"CPE8"
,
_quadrangle_8
)
(
"CPS8"
,
_quadrangle_8
)
(
"DC2D8"
,
_quadrangle_8
)
(
"C3D4"
,
_tetrahedron_4
)
(
"DC3D4"
,
_tetrahedron_4
)
(
"C3D8"
,
_hexahedron_8
)
(
"C3D8R"
,
_hexahedron_8
)
(
"DC3D8"
,
_hexahedron_8
)
(
"C3D10"
,
_tetrahedron_10
)
(
"DC3D10"
,
_tetrahedron_10
);
qi
::
on_error
<
qi
::
fail
>
(
start
,
error_handler
(
lbs
::
_4
,
lbs
::
_3
,
lbs
::
_2
));
start
.
name
(
"abaqus-start-rule"
);
connectivity
.
name
(
"abaqus-connectivity"
);
node_position
.
name
(
"abaqus-nodes-position"
);
nodes
.
name
(
"abaqus-nodes"
);
any_section
.
name
(
"abaqus-any_section"
);
header
.
name
(
"abaqus-header"
);
material
.
name
(
"abaqus-material"
);
elements
.
name
(
"abaqus-elements"
);
elements_set
.
name
(
"abaqus-elements-set"
);
nodes_set
.
name
(
"abaqus-nodes-set"
);
key
.
name
(
"abaqus-key"
);
value
.
name
(
"abaqus-value"
);
option
.
name
(
"abaqus-option"
);
keyword
.
name
(
"abaqus-keyword"
);
any_line
.
name
(
"abaqus-any-line"
);
abaqus_element_type
.
name
(
"abaqus-element-type"
);
}
public
:
AKANTU_GET_MACRO
(
MaterialNames
,
material_names
,
const
std
::
vector
<
std
::
string
>
&
);
/* ------------------------------------------------------------------------ */
/* Rules */
/* ------------------------------------------------------------------------ */
private
:
qi
::
rule
<
Iterator
,
void
(),
Skipper
>
start
;
qi
::
rule
<
Iterator
,
std
::
vector
<
int
>
(),
Skipper
>
connectivity
;
qi
::
rule
<
Iterator
,
std
::
vector
<
Real
>
(),
Skipper
>
node_position
;
qi
::
rule
<
Iterator
,
void
(),
Skipper
>
nodes
,
any_section
,
header
,
material
;
qi
::
rule
<
Iterator
,
void
(),
qi
::
locals
<
ElementType
>
,
Skipper
>
elements
;
qi
::
rule
<
Iterator
,
void
(),
qi
::
locals
<
ElementGroup
*>
,
Skipper
>
elements_set
;
qi
::
rule
<
Iterator
,
void
(),
qi
::
locals
<
NodeGroup
*>
,
Skipper
>
nodes_set
;
qi
::
rule
<
Iterator
,
std
::
string
(),
Skipper
>
key
,
value
,
option
,
keyword
,
any_line
;
qi
::
real_parser
<
Real
,
qi
::
real_policies
<
Real
>
>
real
;
qi
::
symbols
<
char
,
ElementType
>
abaqus_element_type
;
/* ------------------------------------------------------------------------ */
/* Mambers */
/* ------------------------------------------------------------------------ */
private
:
/// reference to the mesh to read
Mesh
&
mesh
;
/// correspondance between the numbering of nodes in the abaqus file and in
/// the akantu mesh
std
::
map
<
UInt
,
UInt
>
abaqus_nodes_to_akantu
;
/// correspondance between the element number in the abaqus file and the
/// Element in the akantu mesh
std
::
map
<
UInt
,
Element
>
abaqus_elements_to_akantu
;
/// list of the material names
std
::
vector
<
std
::
string
>
material_names
;
};
/* -------------------------------------------------------------------------- */
/* -------------------------------------------------------------------------- */
void
MeshIOAbaqus
::
read
(
const
std
::
string
&
filename
,
Mesh
&
mesh
)
{
namespace
spirit
=
boost
::
spirit
;
namespace
qi
=
boost
::
spirit
::
qi
;
namespace
lbs
=
boost
::
spirit
::
qi
::
labels
;
namespace
ascii
=
boost
::
spirit
::
ascii
;
namespace
phx
=
boost
::
phoenix
;
std
::
ifstream
infile
;
infile
.
open
(
filename
.
c_str
());
if
(
!
infile
.
good
())
{
AKANTU_DEBUG_ERROR
(
"Cannot open file "
<<
filename
);
}
std
::
string
storage
;
// We will read the contents here.
infile
.
unsetf
(
std
::
ios
::
skipws
);
// No white space skipping!
std
::
copy
(
std
::
istream_iterator
<
char
>
(
infile
),
std
::
istream_iterator
<
char
>
(),
std
::
back_inserter
(
storage
));
typedef
std
::
string
::
const_iterator
iterator_t
;
typedef
AbaqusSkipper
<
iterator_t
>
skipper
;
typedef
AbaqusMeshGrammar
<
iterator_t
,
skipper
>
grammar
;
grammar
g
(
mesh
);
skipper
ws
;
iterator_t
iter
=
storage
.
begin
();
iterator_t
end
=
storage
.
end
();
qi
::
phrase_parse
(
iter
,
end
,
g
,
ws
);
std
::
vector
<
std
::
string
>::
const_iterator
mnit
=
g
.
getMaterialNames
().
begin
();
std
::
vector
<
std
::
string
>::
const_iterator
mnend
=
g
.
getMaterialNames
().
end
();
for
(;
mnit
!=
mnend
;
++
mnit
)
{
Mesh
::
element_group_iterator
eg_it
=
mesh
.
element_group_find
(
*
mnit
);
ElementGroup
&
eg
=
*
eg_it
->
second
;
if
(
eg_it
!=
mesh
.
element_group_end
())
{
ElementGroup
::
type_iterator
tit
=
eg
.
firstType
();
ElementGroup
::
type_iterator
tend
=
eg
.
lastType
();
for
(;
tit
!=
tend
;
++
tit
)
{
Array
<
std
::
string
>
&
abaqus_material
=
*
mesh
.
getDataPointer
<
std
::
string
>
(
"abaqus_material"
,
*
tit
);
ElementGroup
::
const_element_iterator
eit
=
eg
.
element_begin
(
*
tit
);
ElementGroup
::
const_element_iterator
eend
=
eg
.
element_end
(
*
tit
);
for
(;
eit
!=
eend
;
++
eit
)
{
abaqus_material
(
*
eit
)
=
*
mnit
;
}
}
}
}
}
__END_AKANTU__
Event Timeline
Log In to Comment