2015-05-19 03:48:29 +00:00
|
|
|
/*
|
|
|
|
Open Asset Import Library (assimp)
|
|
|
|
----------------------------------------------------------------------
|
|
|
|
|
2020-01-20 13:53:12 +00:00
|
|
|
Copyright (c) 2006-2020, assimp team
|
2018-01-28 18:42:05 +00:00
|
|
|
|
2017-05-09 17:57:36 +00:00
|
|
|
|
2015-05-19 03:48:29 +00:00
|
|
|
All rights reserved.
|
|
|
|
|
2015-05-19 03:52:10 +00:00
|
|
|
Redistribution and use of this software in source and binary forms,
|
|
|
|
with or without modification, are permitted provided that the
|
2015-05-19 03:48:29 +00:00
|
|
|
following conditions are met:
|
|
|
|
|
|
|
|
* Redistributions of source code must retain the above
|
|
|
|
copyright notice, this list of conditions and the
|
|
|
|
following disclaimer.
|
|
|
|
|
|
|
|
* Redistributions in binary form must reproduce the above
|
|
|
|
copyright notice, this list of conditions and the
|
|
|
|
following disclaimer in the documentation and/or other
|
|
|
|
materials provided with the distribution.
|
|
|
|
|
|
|
|
* Neither the name of the assimp team, nor the names of its
|
|
|
|
contributors may be used to endorse or promote products
|
|
|
|
derived from this software without specific prior
|
|
|
|
written permission of the assimp team.
|
|
|
|
|
2015-05-19 03:52:10 +00:00
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
|
"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
2015-05-19 03:48:29 +00:00
|
|
|
LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
2015-05-19 03:52:10 +00:00
|
|
|
A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
2015-05-19 03:48:29 +00:00
|
|
|
OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
2015-05-19 03:52:10 +00:00
|
|
|
SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
2015-05-19 03:48:29 +00:00
|
|
|
LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
2015-05-19 03:52:10 +00:00
|
|
|
DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
2015-05-19 03:48:29 +00:00
|
|
|
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
|
|
|
----------------------------------------------------------------------
|
|
|
|
*/
|
|
|
|
|
|
|
|
/** @file Definition of the base class for all importer worker classes. */
|
2019-10-11 11:27:36 +00:00
|
|
|
#pragma once
|
2015-05-19 03:48:29 +00:00
|
|
|
#ifndef INCLUDED_AI_BASEIMPORTER_H
|
|
|
|
#define INCLUDED_AI_BASEIMPORTER_H
|
|
|
|
|
2019-10-11 11:27:36 +00:00
|
|
|
#ifdef __GNUC__
|
|
|
|
# pragma GCC system_header
|
|
|
|
#endif
|
|
|
|
|
2015-05-19 03:48:29 +00:00
|
|
|
#include "Exceptional.h"
|
|
|
|
|
|
|
|
#include <vector>
|
|
|
|
#include <set>
|
2019-08-21 19:29:46 +00:00
|
|
|
#include <map>
|
2016-06-06 20:04:29 +00:00
|
|
|
#include <assimp/types.h>
|
|
|
|
#include <assimp/ProgressHandler.hpp>
|
2019-08-21 19:29:46 +00:00
|
|
|
#include <assimp/ai_assert.h>
|
2015-05-19 03:48:29 +00:00
|
|
|
|
|
|
|
struct aiScene;
|
2017-02-22 16:20:26 +00:00
|
|
|
struct aiImporterDesc;
|
2015-05-19 03:48:29 +00:00
|
|
|
|
2015-05-19 03:57:13 +00:00
|
|
|
namespace Assimp {
|
2015-05-19 03:48:29 +00:00
|
|
|
|
|
|
|
class Importer;
|
|
|
|
class IOSystem;
|
|
|
|
class BaseProcess;
|
|
|
|
class SharedPostProcessInfo;
|
|
|
|
class IOStream;
|
|
|
|
|
|
|
|
// utility to do char4 to uint32 in a portable manner
|
|
|
|
#define AI_MAKE_MAGIC(string) ((uint32_t)((string[0] << 24) + \
|
2015-05-19 03:57:13 +00:00
|
|
|
(string[1] << 16) + (string[2] << 8) + string[3]))
|
2015-05-19 03:48:29 +00:00
|
|
|
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------------------
|
2015-05-19 03:52:10 +00:00
|
|
|
/** FOR IMPORTER PLUGINS ONLY: The BaseImporter defines a common interface
|
2015-05-19 03:48:29 +00:00
|
|
|
* for all importer worker classes.
|
|
|
|
*
|
2015-05-19 03:52:10 +00:00
|
|
|
* The interface defines two functions: CanRead() is used to check if the
|
|
|
|
* importer can handle the format of the given file. If an implementation of
|
|
|
|
* this function returns true, the importer then calls ReadFile() which
|
|
|
|
* imports the given file. ReadFile is not overridable, it just calls
|
2015-05-19 03:48:29 +00:00
|
|
|
* InternReadFile() and catches any ImportErrorException that might occur.
|
|
|
|
*/
|
2018-09-21 14:07:09 +00:00
|
|
|
class ASSIMP_API BaseImporter {
|
2015-05-19 03:57:13 +00:00
|
|
|
friend class Importer;
|
2015-05-19 03:48:29 +00:00
|
|
|
|
2019-08-27 14:50:50 +00:00
|
|
|
private:
|
|
|
|
/* Pushes state into importer for the importer scale */
|
|
|
|
virtual void UpdateImporterScale( Importer* pImp );
|
|
|
|
|
2015-05-19 03:48:29 +00:00
|
|
|
public:
|
|
|
|
|
2015-05-19 03:57:13 +00:00
|
|
|
/** Constructor to be privately used by #Importer */
|
2018-09-21 14:25:27 +00:00
|
|
|
BaseImporter() AI_NO_EXCEPT;
|
2015-05-19 03:48:29 +00:00
|
|
|
|
2015-05-19 03:57:13 +00:00
|
|
|
/** Destructor, private as well */
|
|
|
|
virtual ~BaseImporter();
|
2015-05-19 03:48:29 +00:00
|
|
|
|
2015-05-19 03:57:13 +00:00
|
|
|
// -------------------------------------------------------------------
|
|
|
|
/** Returns whether the class can handle the format of the given file.
|
|
|
|
*
|
|
|
|
* The implementation should be as quick as possible. A check for
|
|
|
|
* the file extension is enough. If no suitable loader is found with
|
|
|
|
* this strategy, CanRead() is called again, the 'checkSig' parameter
|
|
|
|
* set to true this time. Now the implementation is expected to
|
|
|
|
* perform a full check of the file structure, possibly searching the
|
|
|
|
* first bytes of the file for magic identifiers or keywords.
|
|
|
|
*
|
|
|
|
* @param pFile Path and file name of the file to be examined.
|
|
|
|
* @param pIOHandler The IO handler to use for accessing any file.
|
|
|
|
* @param checkSig Set to true if this method is called a second time.
|
|
|
|
* This time, the implementation may take more time to examine the
|
|
|
|
* contents of the file to be loaded for magic bytes, keywords, etc
|
|
|
|
* to be able to load files with unknown/not existent file extensions.
|
|
|
|
* @return true if the class can read this file, false if not.
|
|
|
|
*/
|
|
|
|
virtual bool CanRead(
|
|
|
|
const std::string& pFile,
|
|
|
|
IOSystem* pIOHandler,
|
|
|
|
bool checkSig
|
|
|
|
) const = 0;
|
|
|
|
|
|
|
|
// -------------------------------------------------------------------
|
|
|
|
/** Imports the given file and returns the imported data.
|
|
|
|
* If the import succeeds, ownership of the data is transferred to
|
|
|
|
* the caller. If the import fails, NULL is returned. The function
|
|
|
|
* takes care that any partially constructed data is destroyed
|
|
|
|
* beforehand.
|
|
|
|
*
|
|
|
|
* @param pImp #Importer object hosting this loader.
|
|
|
|
* @param pFile Path of the file to be imported.
|
|
|
|
* @param pIOHandler IO-Handler used to open this and possible other files.
|
|
|
|
* @return The imported data or NULL if failed. If it failed a
|
|
|
|
* human-readable error description can be retrieved by calling
|
|
|
|
* GetErrorText()
|
|
|
|
*
|
|
|
|
* @note This function is not intended to be overridden. Implement
|
|
|
|
* InternReadFile() to do the import. If an exception is thrown somewhere
|
|
|
|
* in InternReadFile(), this function will catch it and transform it into
|
|
|
|
* a suitable response to the caller.
|
|
|
|
*/
|
|
|
|
aiScene* ReadFile(
|
2019-08-27 14:50:50 +00:00
|
|
|
Importer* pImp,
|
2015-05-19 03:57:13 +00:00
|
|
|
const std::string& pFile,
|
|
|
|
IOSystem* pIOHandler
|
|
|
|
);
|
|
|
|
|
|
|
|
// -------------------------------------------------------------------
|
2016-04-03 00:38:00 +00:00
|
|
|
/** Returns the error description of the last error that occurred.
|
|
|
|
* @return A description of the last error that occurred. An empty
|
2015-05-19 03:57:13 +00:00
|
|
|
* string if there was no error.
|
|
|
|
*/
|
|
|
|
const std::string& GetErrorText() const {
|
2015-09-23 22:57:47 +00:00
|
|
|
return m_ErrorText;
|
2015-05-19 03:57:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// -------------------------------------------------------------------
|
|
|
|
/** Called prior to ReadFile().
|
|
|
|
* The function is a request to the importer to update its configuration
|
|
|
|
* basing on the Importer's configuration property list.
|
|
|
|
* @param pImp Importer instance
|
|
|
|
*/
|
|
|
|
virtual void SetupProperties(
|
|
|
|
const Importer* pImp
|
|
|
|
);
|
|
|
|
|
|
|
|
// -------------------------------------------------------------------
|
|
|
|
/** Called by #Importer::GetImporterInfo to get a description of
|
|
|
|
* some loader features. Importers must provide this information. */
|
|
|
|
virtual const aiImporterDesc* GetInfo() const = 0;
|
|
|
|
|
2019-08-21 19:29:46 +00:00
|
|
|
/**
|
|
|
|
* Will be called only by scale process when scaling is requested.
|
|
|
|
*/
|
|
|
|
virtual void SetFileScale(double scale)
|
|
|
|
{
|
|
|
|
fileScale = scale;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual double GetFileScale() const
|
|
|
|
{
|
|
|
|
return fileScale;
|
|
|
|
}
|
|
|
|
|
|
|
|
enum ImporterUnits {
|
|
|
|
M,
|
|
|
|
MM,
|
|
|
|
CM,
|
|
|
|
INCHES,
|
|
|
|
FEET
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Assimp Importer
|
|
|
|
* unit conversions available
|
2019-10-16 08:06:57 +00:00
|
|
|
* NOTE: Valid options are initialised in the
|
|
|
|
* constructor in the implementation file to
|
|
|
|
* work around a VS2013 compiler bug if support
|
|
|
|
* for that compiler is dropped in the future
|
|
|
|
* initialisation can be moved back here
|
2019-08-21 19:29:46 +00:00
|
|
|
* */
|
2019-10-16 08:06:57 +00:00
|
|
|
std::map<ImporterUnits, double> importerUnits;
|
2019-08-21 19:29:46 +00:00
|
|
|
|
|
|
|
virtual void SetApplicationUnits( const ImporterUnits& unit )
|
|
|
|
{
|
|
|
|
importerScale = importerUnits[unit];
|
|
|
|
applicationUnits = unit;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual const ImporterUnits& GetApplicationUnits()
|
|
|
|
{
|
|
|
|
return applicationUnits;
|
|
|
|
}
|
|
|
|
|
2015-05-19 03:57:13 +00:00
|
|
|
// -------------------------------------------------------------------
|
|
|
|
/** Called by #Importer::GetExtensionList for each loaded importer.
|
|
|
|
* Take the extension list contained in the structure returned by
|
|
|
|
* #GetInfo and insert all file extensions into the given set.
|
|
|
|
* @param extension set to collect file extensions in*/
|
|
|
|
void GetExtensionList(std::set<std::string>& extensions);
|
2019-08-21 19:29:46 +00:00
|
|
|
|
|
|
|
protected:
|
|
|
|
ImporterUnits applicationUnits = ImporterUnits::M;
|
|
|
|
double importerScale = 1.0;
|
|
|
|
double fileScale = 1.0;
|
2015-05-19 03:48:29 +00:00
|
|
|
|
|
|
|
|
2019-08-27 14:50:50 +00:00
|
|
|
|
2015-05-19 03:57:13 +00:00
|
|
|
// -------------------------------------------------------------------
|
|
|
|
/** Imports the given file into the given scene structure. The
|
|
|
|
* function is expected to throw an ImportErrorException if there is
|
|
|
|
* an error. If it terminates normally, the data in aiScene is
|
|
|
|
* expected to be correct. Override this function to implement the
|
|
|
|
* actual importing.
|
|
|
|
* <br>
|
|
|
|
* The output scene must meet the following requirements:<br>
|
|
|
|
* <ul>
|
|
|
|
* <li>At least a root node must be there, even if its only purpose
|
|
|
|
* is to reference one mesh.</li>
|
|
|
|
* <li>aiMesh::mPrimitiveTypes may be 0. The types of primitives
|
|
|
|
* in the mesh are determined automatically in this case.</li>
|
|
|
|
* <li>the vertex data is stored in a pseudo-indexed "verbose" format.
|
|
|
|
* In fact this means that every vertex that is referenced by
|
|
|
|
* a face is unique. Or the other way round: a vertex index may
|
|
|
|
* not occur twice in a single aiMesh.</li>
|
|
|
|
* <li>aiAnimation::mDuration may be -1. Assimp determines the length
|
|
|
|
* of the animation automatically in this case as the length of
|
|
|
|
* the longest animation channel.</li>
|
|
|
|
* <li>aiMesh::mBitangents may be NULL if tangents and normals are
|
|
|
|
* given. In this case bitangents are computed as the cross product
|
|
|
|
* between normal and tangent.</li>
|
|
|
|
* <li>There needn't be a material. If none is there a default material
|
|
|
|
* is generated. However, it is recommended practice for loaders
|
|
|
|
* to generate a default material for yourself that matches the
|
|
|
|
* default material setting for the file format better than Assimp's
|
|
|
|
* generic default material. Note that default materials *should*
|
|
|
|
* be named AI_DEFAULT_MATERIAL_NAME if they're just color-shaded
|
|
|
|
* or AI_DEFAULT_TEXTURED_MATERIAL_NAME if they define a (dummy)
|
|
|
|
* texture. </li>
|
|
|
|
* </ul>
|
|
|
|
* If the AI_SCENE_FLAGS_INCOMPLETE-Flag is <b>not</b> set:<ul>
|
|
|
|
* <li> at least one mesh must be there</li>
|
|
|
|
* <li> there may be no meshes with 0 vertices or faces</li>
|
|
|
|
* </ul>
|
|
|
|
* This won't be checked (except by the validation step): Assimp will
|
|
|
|
* crash if one of the conditions is not met!
|
|
|
|
*
|
|
|
|
* @param pFile Path of the file to be imported.
|
|
|
|
* @param pScene The scene object to hold the imported data.
|
|
|
|
* NULL is not a valid parameter.
|
|
|
|
* @param pIOHandler The IO handler to use for any file access.
|
|
|
|
* NULL is not a valid parameter. */
|
|
|
|
virtual void InternReadFile(
|
|
|
|
const std::string& pFile,
|
|
|
|
aiScene* pScene,
|
|
|
|
IOSystem* pIOHandler
|
|
|
|
) = 0;
|
2015-05-19 03:48:29 +00:00
|
|
|
|
|
|
|
public: // static utilities
|
|
|
|
|
2015-05-19 03:57:13 +00:00
|
|
|
// -------------------------------------------------------------------
|
|
|
|
/** A utility for CanRead().
|
|
|
|
*
|
|
|
|
* The function searches the header of a file for a specific token
|
|
|
|
* and returns true if this token is found. This works for text
|
|
|
|
* files only. There is a rudimentary handling of UNICODE files.
|
|
|
|
* The comparison is case independent.
|
|
|
|
*
|
|
|
|
* @param pIOSystem IO System to work with
|
|
|
|
* @param file File name of the file
|
|
|
|
* @param tokens List of tokens to search for
|
|
|
|
* @param numTokens Size of the token array
|
|
|
|
* @param searchBytes Number of bytes to be searched for the tokens.
|
|
|
|
*/
|
|
|
|
static bool SearchFileHeaderForToken(
|
|
|
|
IOSystem* pIOSystem,
|
|
|
|
const std::string& file,
|
|
|
|
const char** tokens,
|
|
|
|
unsigned int numTokens,
|
|
|
|
unsigned int searchBytes = 200,
|
2018-06-25 10:05:37 +00:00
|
|
|
bool tokensSol = false,
|
|
|
|
bool noAlphaBeforeTokens = false);
|
2015-05-19 03:57:13 +00:00
|
|
|
|
|
|
|
// -------------------------------------------------------------------
|
|
|
|
/** @brief Check whether a file has a specific file extension
|
|
|
|
* @param pFile Input file
|
|
|
|
* @param ext0 Extension to check for. Lowercase characters only, no dot!
|
|
|
|
* @param ext1 Optional second extension
|
|
|
|
* @param ext2 Optional third extension
|
|
|
|
* @note Case-insensitive
|
|
|
|
*/
|
|
|
|
static bool SimpleExtensionCheck (
|
|
|
|
const std::string& pFile,
|
|
|
|
const char* ext0,
|
|
|
|
const char* ext1 = NULL,
|
|
|
|
const char* ext2 = NULL);
|
|
|
|
|
|
|
|
// -------------------------------------------------------------------
|
|
|
|
/** @brief Extract file extension from a string
|
|
|
|
* @param pFile Input file
|
|
|
|
* @return Extension without trailing dot, all lowercase
|
|
|
|
*/
|
|
|
|
static std::string GetExtension (
|
|
|
|
const std::string& pFile);
|
|
|
|
|
|
|
|
// -------------------------------------------------------------------
|
|
|
|
/** @brief Check whether a file starts with one or more magic tokens
|
|
|
|
* @param pFile Input file
|
|
|
|
* @param pIOHandler IO system to be used
|
|
|
|
* @param magic n magic tokens
|
|
|
|
* @params num Size of magic
|
|
|
|
* @param offset Offset from file start where tokens are located
|
|
|
|
* @param Size of one token, in bytes. Maximally 16 bytes.
|
|
|
|
* @return true if one of the given tokens was found
|
|
|
|
*
|
2017-06-17 17:38:47 +00:00
|
|
|
* @note For convenience, the check is also performed for the
|
2015-05-19 03:57:13 +00:00
|
|
|
* byte-swapped variant of all tokens (big endian). Only for
|
|
|
|
* tokens of size 2,4.
|
|
|
|
*/
|
|
|
|
static bool CheckMagicToken(
|
|
|
|
IOSystem* pIOHandler,
|
|
|
|
const std::string& pFile,
|
|
|
|
const void* magic,
|
|
|
|
unsigned int num,
|
|
|
|
unsigned int offset = 0,
|
|
|
|
unsigned int size = 4);
|
|
|
|
|
|
|
|
// -------------------------------------------------------------------
|
|
|
|
/** An utility for all text file loaders. It converts a file to our
|
|
|
|
* UTF8 character set. Errors are reported, but ignored.
|
|
|
|
*
|
|
|
|
* @param data File buffer to be converted to UTF8 data. The buffer
|
|
|
|
* is resized as appropriate. */
|
|
|
|
static void ConvertToUTF8(
|
|
|
|
std::vector<char>& data);
|
|
|
|
|
|
|
|
// -------------------------------------------------------------------
|
|
|
|
/** An utility for all text file loaders. It converts a file from our
|
|
|
|
* UTF8 character set back to ISO-8859-1. Errors are reported, but ignored.
|
|
|
|
*
|
|
|
|
* @param data File buffer to be converted from UTF8 to ISO-8859-1. The buffer
|
|
|
|
* is resized as appropriate. */
|
|
|
|
static void ConvertUTF8toISO8859_1(
|
|
|
|
std::string& data);
|
|
|
|
|
2016-11-11 11:49:05 +00:00
|
|
|
// -------------------------------------------------------------------
|
|
|
|
/// @brief Enum to define, if empty files are ok or not.
|
|
|
|
enum TextFileMode {
|
|
|
|
ALLOW_EMPTY,
|
|
|
|
FORBID_EMPTY
|
|
|
|
};
|
2015-12-14 06:13:25 +00:00
|
|
|
|
2015-05-19 03:57:13 +00:00
|
|
|
// -------------------------------------------------------------------
|
|
|
|
/** Utility for text file loaders which copies the contents of the
|
|
|
|
* file into a memory buffer and converts it to our UTF8
|
|
|
|
* representation.
|
|
|
|
* @param stream Stream to read from.
|
|
|
|
* @param data Output buffer to be resized and filled with the
|
|
|
|
* converted text file data. The buffer is terminated with
|
2015-12-14 06:13:25 +00:00
|
|
|
* a binary 0.
|
|
|
|
* @param mode Whether it is OK to load empty text files. */
|
2015-05-19 03:57:13 +00:00
|
|
|
static void TextFileToBuffer(
|
|
|
|
IOStream* stream,
|
2015-12-14 06:13:25 +00:00
|
|
|
std::vector<char>& data,
|
|
|
|
TextFileMode mode = FORBID_EMPTY);
|
2015-05-19 03:48:29 +00:00
|
|
|
|
2015-11-29 18:13:51 +00:00
|
|
|
// -------------------------------------------------------------------
|
|
|
|
/** Utility function to move a std::vector into a aiScene array
|
|
|
|
* @param vec The vector to be moved
|
|
|
|
* @param out The output pointer to the allocated array.
|
|
|
|
* @param numOut The output count of elements copied. */
|
|
|
|
template<typename T>
|
|
|
|
AI_FORCE_INLINE
|
|
|
|
static void CopyVector(
|
|
|
|
std::vector<T>& vec,
|
|
|
|
T*& out,
|
|
|
|
unsigned int& outLength)
|
|
|
|
{
|
2016-04-28 18:50:01 +00:00
|
|
|
outLength = unsigned(vec.size());
|
2015-11-29 18:13:51 +00:00
|
|
|
if (outLength) {
|
|
|
|
out = new T[outLength];
|
|
|
|
std::swap_ranges(vec.begin(), vec.end(), out);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-19 03:48:29 +00:00
|
|
|
protected:
|
2016-11-11 11:49:05 +00:00
|
|
|
/// Error description in case there was one.
|
2015-09-23 22:57:47 +00:00
|
|
|
std::string m_ErrorText;
|
2016-11-11 11:49:05 +00:00
|
|
|
/// Currently set progress handler.
|
2015-09-23 22:57:47 +00:00
|
|
|
ProgressHandler* m_progress;
|
2015-05-19 03:48:29 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
} // end of namespace Assimp
|
|
|
|
|
|
|
|
#endif // AI_BASEIMPORTER_H_INC
|