Page Menu
Home
c4science
Search
Configure Global Search
Log In
Files
F100889821
input_file_parser.hh
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
Mon, Feb 3, 16:05
Size
8 KB
Mime Type
text/x-c++
Expires
Wed, Feb 5, 16:05 (2 d)
Engine
blob
Format
Raw Data
Handle
24047997
Attached To
rCADDMESH CADD_mesher
input_file_parser.hh
View Options
/**
* @file input_file_parser.hh
*
* @author Nicolas Richart <nicolas.richart@epfl.ch>
*
* @date creation: Wed Nov 13 2013
* @date last modification: Tue May 19 2015
*
* @brief Grammar definition for the input files
*
* @section LICENSE
*
* Copyright (©) 2014, 2015 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/>.
*
*/
/* -------------------------------------------------------------------------- */
// Boost
/* -------------------------------------------------------------------------- */
#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>
#include <boost/fusion/include/adapt_struct.hpp>
#include <boost/variant/recursive_variant.hpp>
//#include <boost/foreach.hpp>
#ifndef __AKANTU_INPUT_FILE_PARSER_HH__
#define __AKANTU_INPUT_FILE_PARSER_HH__
namespace
spirit
=
boost
::
spirit
;
namespace
qi
=
boost
::
spirit
::
qi
;
namespace
lbs
=
boost
::
spirit
::
qi
::
labels
;
namespace
ascii
=
boost
::
spirit
::
ascii
;
namespace
phx
=
boost
::
phoenix
;
__BEGIN_AKANTU__
namespace
parser
{
struct
error_handler_
{
template
<
typename
,
typename
,
typename
,
typename
>
struct
result
{
typedef
void
type
;
};
template
<
typename
Iterator
>
void
operator
()(
qi
::
info
const
&
what
,
Iterator
err_pos
,
__attribute__
((
unused
))
Iterator
first
,
__attribute__
((
unused
))
Iterator
last
)
const
{
spirit
::
classic
::
file_position
pos
=
err_pos
.
get_position
();
AKANTU_EXCEPTION
(
"Parse error [ "
<<
"Expecting "
<<
what
<<
" instead of
\"
"
<<
*
err_pos
<<
"
\"
]"
<<
" in file "
<<
pos
.
file
<<
" line "
<<
pos
.
line
<<
" column "
<<
pos
.
column
<<
std
::
endl
<<
"'"
<<
err_pos
.
get_currentline
()
<<
"'"
<<
std
::
endl
<<
std
::
setw
(
pos
.
column
)
<<
" "
<<
"^- here"
);
}
private
:
};
struct
lazy_create_subsection_
{
template
<
typename
,
typename
,
typename
,
typename
>
struct
result
{
typedef
ParserSection
&
type
;
};
template
<
typename
T
>
ParserSection
&
operator
()(
const
SectionType
&
type
,
const
std
::
string
&
name
,
const
T
&
option
,
ParserSection
&
sect
)
const
{
std
::
string
opt
;
if
(
option
)
opt
=
*
option
;
ParserSection
sect_tmp
(
name
,
type
,
opt
,
sect
);
return
sect
.
addSubSection
(
sect_tmp
);
}
};
template
<
typename
Iterator
>
struct
lazy_create_parameter_
{
lazy_create_parameter_
(
std
::
string
&
error_message
)
:
error_message
(
error_message
)
{}
template
<
typename
,
typename
,
typename
>
struct
result
{
typedef
bool
type
;
};
template
<
typename
Range
>
bool
operator
()(
const
Range
&
rng
,
const
std
::
string
&
value
,
ParserSection
&
sect
)
const
{
try
{
std
::
string
name
(
rng
.
begin
(),
rng
.
end
());
name
=
trim
(
name
);
spirit
::
classic
::
file_position
pos
=
rng
.
begin
().
get_position
();
ParserParameter
param_tmp
(
name
,
value
,
sect
);
param_tmp
.
setDebugInfo
(
pos
.
file
,
pos
.
line
,
pos
.
column
);
sect
.
addParameter
(
param_tmp
);
}
catch
(
debug
::
Exception
&
e
)
{
error_message
=
e
.
info
();
return
false
;
}
return
true
;
}
private
:
std
::
string
&
error_message
;
};
struct
lazy_concatenate_
{
template
<
class
T1
,
class
T2
>
struct
result
{
typedef
T1
type
;
};
template
<
class
T1
,
class
T2
>
T1
operator
()(
const
T1
&
t1
,
const
T2
&
t2
)
const
{
return
(
t1
+
t2
);
}
};
/* ---------------------------------------------------------------------- */
/* Grammars definitions */
/* ---------------------------------------------------------------------- */
template
<
class
Iterator
>
struct
InputFileGrammar
:
qi
::
grammar
<
Iterator
,
void
(),
typename
Skipper
<
Iterator
>::
type
>
{
InputFileGrammar
(
ParserSection
*
sect
)
:
InputFileGrammar
::
base_type
(
start
,
"input_file_grammar"
),
parent_section
(
sect
)
{
phx
::
function
<
error_handler_
>
const
error_handler
=
error_handler_
();
phx
::
function
<
lazy_create_parameter_
<
Iterator
>
>
lazy_create_parameter
=
lazy_create_parameter_
<
Iterator
>
(
error_message
);
phx
::
function
<
lazy_create_subsection_
>
lazy_create_subsection
;
phx
::
function
<
lazy_concatenate_
>
lazy_concatenate
;
/* clang-format off */
start
=
mini_section
(
parent_section
)
;
mini_section
=
*
(
entry
(
lbs
::
_r1
)
|
section
(
lbs
::
_r1
)
)
;
entry
=
(
qi
::
raw
[
key
]
>>
'='
>
value
)
[
lbs
::
_pass
=
lazy_create_parameter
(
lbs
::
_1
,
lbs
::
_2
,
*
lbs
::
_r1
)
]
;
section
=
(
qi
::
no_case
[
section_type
]
>
qi
::
lexeme
[
section_name
>
-
section_option
]
)
[
lbs
::
_a
=
&
lazy_create_subsection
(
lbs
::
_1
,
phx
::
at_c
<
0
>
(
lbs
::
_2
),
phx
::
at_c
<
1
>
(
lbs
::
_2
),
*
lbs
::
_r1
)
]
>
'['
>
mini_section
(
lbs
::
_a
)
>
']'
;
section_name
=
qi
::
char_
(
"a-zA-Z_"
)
>>
*
qi
::
char_
(
"a-zA-Z_0-9"
)
;
section_option
=
(
+
ascii
::
space
>>
section_name
)
[
lbs
::
_val
=
lbs
::
_2
]
;
key
=
qi
::
char_
(
"a-zA-Z_"
)
>>
*
qi
::
char_
(
"a-zA-Z_0-9"
)
;
value
=
(
mono_line_value
[
lbs
::
_a
=
lazy_concatenate
(
lbs
::
_a
,
lbs
::
_1
)
]
>
*
(
'\\'
>
mono_line_value
[
lbs
::
_a
=
lazy_concatenate
(
lbs
::
_a
,
lbs
::
_1
)
]
)
)
[
lbs
::
_val
=
lbs
::
_a
]
;
mono_line_value
=
qi
::
lexeme
[
+
(
qi
::
char_
-
(
qi
::
char_
(
'='
)
|
spirit
::
eol
|
'#'
|
';'
|
'\\'
))
]
;
skipper
=
ascii
::
space
|
"#"
>>
*
(
qi
::
char_
-
spirit
::
eol
)
;
/* clang-format on */
#define AKANTU_SECTION_TYPE_ADD(r, data, elem) \
(BOOST_PP_STRINGIZE(elem), AKANTU_SECTION_TYPES_PREFIX(elem))
section_type
.
add
BOOST_PP_SEQ_FOR_EACH
(
AKANTU_SECTION_TYPE_ADD
,
_
,
AKANTU_SECTION_TYPES
);
#undef AKANTU_SECTION_TYPE_ADD
qi
::
on_error
<
qi
::
fail
>
(
start
,
error_handler
(
lbs
::
_4
,
lbs
::
_3
,
lbs
::
_1
,
lbs
::
_2
));
section
.
name
(
"section"
);
section_name
.
name
(
"section-name"
);
section_option
.
name
(
"section-option"
);
mini_section
.
name
(
"section-content"
);
entry
.
name
(
"parameter"
);
key
.
name
(
"parameter-name"
);
value
.
name
(
"parameter-value"
);
section_type
.
name
(
"section-types-list"
);
mono_line_value
.
name
(
"mono-line-value"
);
#if !defined AKANTU_NDEBUG
if
(
AKANTU_DEBUG_TEST
(
dblDebug
))
{
// qi::debug(section);
qi
::
debug
(
section_name
);
qi
::
debug
(
section_option
);
// qi::debug(mini_section);
// qi::debug(entry);
qi
::
debug
(
key
);
qi
::
debug
(
value
);
qi
::
debug
(
mono_line_value
);
}
#endif
}
const
std
::
string
&
getErrorMessage
()
const
{
return
error_message
;
};
typedef
typename
Skipper
<
Iterator
>::
type
skipper_type
;
skipper_type
skipper
;
private
:
std
::
string
error_message
;
qi
::
rule
<
Iterator
,
void
(
ParserSection
*
),
skipper_type
>
mini_section
;
qi
::
rule
<
Iterator
,
void
(
ParserSection
*
),
qi
::
locals
<
ParserSection
*>
,
skipper_type
>
section
;
qi
::
rule
<
Iterator
,
void
(),
skipper_type
>
start
;
qi
::
rule
<
Iterator
,
std
::
string
()
>
section_name
;
qi
::
rule
<
Iterator
,
std
::
string
()
>
section_option
;
qi
::
rule
<
Iterator
,
void
(
ParserSection
*
),
skipper_type
>
entry
;
qi
::
rule
<
Iterator
,
std
::
string
(),
skipper_type
>
key
;
qi
::
rule
<
Iterator
,
std
::
string
(),
qi
::
locals
<
std
::
string
>
,
skipper_type
>
value
;
qi
::
rule
<
Iterator
,
std
::
string
(),
skipper_type
>
mono_line_value
;
qi
::
symbols
<
char
,
SectionType
>
section_type
;
ParserSection
*
parent_section
;
};
}
__END_AKANTU__
#endif
/* __AKANTU_INPUT_FILE_PARSER_HH__ */
Event Timeline
Log In to Comment