/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements. See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership. The ASF licenses this file
 * to you under the Apache License, Version 2.0 (the  "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

// Class header file.
#include "XalanSourceTreeElementANSAllocator.hpp"



namespace XALAN_CPP_NAMESPACE {



XalanSourceTreeElementANSAllocator::XalanSourceTreeElementANSAllocator(
            MemoryManager&  theManager,
            size_type       theBlockCount) :
    m_allocator(theManager, theBlockCount)
{
}



XalanSourceTreeElementANSAllocator::~XalanSourceTreeElementANSAllocator()
{
}



XalanSourceTreeElementANSAllocator::ObjectType*
XalanSourceTreeElementANSAllocator::create(
            const XalanDOMString&       theTagName,
            const XalanDOMString&       theLocalName,
            const XalanDOMString&       theNamespaceURI,
            const XalanDOMString&       thePrefix,
            XalanSourceTreeDocument*    theOwnerDocument,
            XalanSourceTreeAttr**       theAttributes,
            XalanSize_t                 theAttributeCount,
            XalanNode*                  theParentNode,
            XalanNode*                  thePreviousSibling,
            XalanNode*                  theNextSibling,
            IndexType                   theIndex)
{
    ObjectType* const   theBlock = m_allocator.allocateBlock();
    assert(theBlock != 0);

    new(theBlock) ObjectType(
                        m_allocator.getMemoryManager(),
                        theTagName,
                        theLocalName,
                        theNamespaceURI,
                        thePrefix,
                        theOwnerDocument,
                        theAttributes,
                        theAttributeCount,
                        theParentNode,
                        thePreviousSibling,
                        theNextSibling,
                        theIndex);

    m_allocator.commitAllocation(theBlock);

    return theBlock;
}



void 
XalanSourceTreeElementANSAllocator::reset()
{
    m_allocator.reset();
}



}
