summaryrefslogtreecommitdiff
path: root/canvas/source/tools/pagemanager.cxx
diff options
context:
space:
mode:
Diffstat (limited to 'canvas/source/tools/pagemanager.cxx')
-rw-r--r--canvas/source/tools/pagemanager.cxx219
1 files changed, 219 insertions, 0 deletions
diff --git a/canvas/source/tools/pagemanager.cxx b/canvas/source/tools/pagemanager.cxx
new file mode 100644
index 000000000000..456e1a7519d2
--- /dev/null
+++ b/canvas/source/tools/pagemanager.cxx
@@ -0,0 +1,219 @@
+/*************************************************************************
+ *
+ * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
+ *
+ * Copyright 2008 by Sun Microsystems, Inc.
+ *
+ * OpenOffice.org - a multi-platform office productivity suite
+ *
+ * $RCSfile: pagemanager.cxx,v $
+ * $Revision: 1.5 $
+ *
+ * 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.
+ *
+ ************************************************************************/
+
+// MARKER(update_precomp.py): autogen include statement, do not remove
+#include "precompiled_canvas.hxx"
+
+#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));
+ 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)->isNaked()))
+ break;
+ ++candidate;
+ }
+
+ const ::basegfx::B2ISize& rSize((*candidate)->getSize());
+ sal_uInt32 nMaxArea(rSize.getX()*rSize.getY());
+
+ FragmentContainer_t::const_iterator it(candidate);
+ while(it != aEnd)
+ {
+ if(!((*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);
+ }
+ }
+
+ //////////////////////////////////////////////////////////////////////////////////
+ // 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;
+ }
+}