223 lines
8.2 KiB
C++
223 lines
8.2 KiB
C++
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
|
/*************************************************************************
|
|
*
|
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
|
*
|
|
* Copyright 2000, 2010 Oracle and/or its affiliates.
|
|
*
|
|
* OpenOffice.org - a multi-platform office productivity suite
|
|
*
|
|
* This file is part of OpenOffice.org.
|
|
*
|
|
* OpenOffice.org is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Lesser General Public License version 3
|
|
* only, as published by the Free Software Foundation.
|
|
*
|
|
* OpenOffice.org 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 version 3 for more details
|
|
* (a copy is included in the LICENSE file that accompanied this code).
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
* version 3 along with OpenOffice.org. If not, see
|
|
* <http://www.openoffice.org/license.html>
|
|
* for a copy of the LGPLv3 License.
|
|
*
|
|
************************************************************************/
|
|
|
|
|
|
#include <boost/bind.hpp>
|
|
#include "pagemanager.hxx"
|
|
|
|
namespace canvas
|
|
{
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////
|
|
// PageManager
|
|
//////////////////////////////////////////////////////////////////////////////////
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////
|
|
// PageManager::allocateSpace
|
|
//////////////////////////////////////////////////////////////////////////////////
|
|
|
|
FragmentSharedPtr PageManager::allocateSpace( const ::basegfx::B2ISize& rSize )
|
|
{
|
|
// we are asked to find a location for the requested size.
|
|
// first we try to satisfy the request from the
|
|
// remaining space in the existing pages.
|
|
const PageContainer_t::iterator aEnd(maPages.end());
|
|
PageContainer_t::iterator it(maPages.begin());
|
|
while(it != aEnd)
|
|
{
|
|
FragmentSharedPtr pFragment((*it)->allocateSpace(rSize));
|
|
if(pFragment)
|
|
{
|
|
// the page created a new fragment, since we maybe want
|
|
// to consolidate sparse pages we keep a reference to
|
|
// the fragment.
|
|
maFragments.push_back(pFragment);
|
|
return pFragment;
|
|
}
|
|
|
|
++it;
|
|
}
|
|
|
|
// otherwise try to create a new page and allocate space there...
|
|
PageSharedPtr pPage(new Page(mpRenderModule));
|
|
if(pPage->isValid())
|
|
{
|
|
maPages.push_back(pPage);
|
|
FragmentSharedPtr pFragment(pPage->allocateSpace(rSize));
|
|
if (pFragment)
|
|
maFragments.push_back(pFragment);
|
|
return pFragment;
|
|
}
|
|
|
|
// the rendermodule failed to create a new page [maybe out
|
|
// of videomemory], and all other pages could not take
|
|
// the new request. we decide to create a 'naked' fragment
|
|
// which will receive its location later.
|
|
FragmentSharedPtr pFragment(new PageFragment(rSize));
|
|
maFragments.push_back(pFragment);
|
|
return pFragment;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////
|
|
// PageManager::free
|
|
//////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void PageManager::free( const FragmentSharedPtr& pFragment )
|
|
{
|
|
// erase the reference to the given fragment from our
|
|
// internal container.
|
|
FragmentContainer_t::iterator it(
|
|
std::remove(
|
|
maFragments.begin(),maFragments.end(),pFragment));
|
|
maFragments.erase(it,maFragments.end());
|
|
|
|
// let the fragment itself know about it...
|
|
// we need to pass 'this' as argument since the fragment
|
|
// needs to pass this to the page and can't create
|
|
// shared_ptr from itself...
|
|
pFragment->free(pFragment);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////
|
|
// PageManager::nakedFragment
|
|
//////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void PageManager::nakedFragment( const FragmentSharedPtr& pFragment )
|
|
{
|
|
if(maPages.empty())
|
|
return;
|
|
|
|
// okay, one last chance is left, we try all available
|
|
// pages again. maybe some other fragment was deleted
|
|
// and we can exploit the space.
|
|
while(!(relocate(pFragment)))
|
|
{
|
|
// no way, we need to free up some space...
|
|
// TODO(F1): this is a heuristic, could
|
|
// be designed as a policy.
|
|
const FragmentContainer_t::const_iterator aEnd(maFragments.end());
|
|
FragmentContainer_t::const_iterator candidate(maFragments.begin());
|
|
while(candidate != aEnd)
|
|
{
|
|
if(*candidate && !((*candidate)->isNaked()))
|
|
break;
|
|
++candidate;
|
|
}
|
|
|
|
if (candidate != aEnd)
|
|
{
|
|
const ::basegfx::B2ISize& rSize((*candidate)->getSize());
|
|
sal_uInt32 nMaxArea(rSize.getX()*rSize.getY());
|
|
|
|
FragmentContainer_t::const_iterator it(candidate);
|
|
while(it != aEnd)
|
|
{
|
|
if (*it && !((*it)->isNaked()))
|
|
{
|
|
const ::basegfx::B2ISize& rCandidateSize((*it)->getSize());
|
|
const sal_uInt32 nArea(rCandidateSize.getX()*rCandidateSize.getY());
|
|
if(nArea > nMaxArea)
|
|
{
|
|
candidate=it;
|
|
nMaxArea=nArea;
|
|
}
|
|
}
|
|
|
|
++it;
|
|
}
|
|
|
|
// this does not erase the candidate,
|
|
// but makes it 'naked'...
|
|
(*candidate)->free(*candidate);
|
|
}
|
|
else
|
|
break;
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////
|
|
// PageManager::relocate
|
|
//////////////////////////////////////////////////////////////////////////////////
|
|
|
|
bool PageManager::relocate( const FragmentSharedPtr& pFragment )
|
|
{
|
|
// the fragment passed as argument is assumed to
|
|
// be naked, that is it is not located on any page.
|
|
// we try all available pages again, maybe some
|
|
// other fragment was deleted and we can exploit the space.
|
|
const PageContainer_t::iterator aEnd(maPages.end());
|
|
PageContainer_t::iterator it(maPages.begin());
|
|
while(it != aEnd)
|
|
{
|
|
// if the page at hand takes the fragment, we immediatelly
|
|
// call select() to pull the information from the associated
|
|
// image to the hardware surface.
|
|
if((*it)->nakedFragment(pFragment))
|
|
{
|
|
// dirty, since newly allocated.
|
|
pFragment->select(true);
|
|
return true;
|
|
}
|
|
|
|
++it;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////
|
|
// PageManager::validatePages
|
|
//////////////////////////////////////////////////////////////////////////////////
|
|
|
|
void PageManager::validatePages()
|
|
{
|
|
::std::for_each( maPages.begin(),
|
|
maPages.end(),
|
|
::boost::mem_fn(&Page::validate));
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////
|
|
// PageManager::getPageSize
|
|
//////////////////////////////////////////////////////////////////////////////////
|
|
|
|
::basegfx::B2ISize PageManager::getPageSize() const
|
|
{
|
|
return mpRenderModule->getPageSize();
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////
|
|
// PageManager::getRenderModule
|
|
//////////////////////////////////////////////////////////////////////////////////
|
|
|
|
canvas::IRenderModuleSharedPtr PageManager::getRenderModule() const
|
|
{
|
|
return mpRenderModule;
|
|
}
|
|
}
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|