/* * Copyright 2006-2012 ICEsoft Technologies Inc. * * Licensed 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. */ package org.icepdf.core.util; import org.icepdf.core.io.SeekableInput; import org.icepdf.core.pobjects.*; import java.util.ArrayList; import java.util.logging.Level; import java.util.logging.Logger; /** * @author Mark Collette * @since 2.0 */ public class LazyObjectLoader implements MemoryManagerDelegate { private static final Logger logger = Logger.getLogger(LazyObjectLoader.class.toString()); private Library library; private SeekableInput m_SeekableInput; private CrossReference m_CrossReference; protected ArrayList<ObjectStream> leastRecentlyUsed; // ArrayList<ObjectStream> private final Object leastRectlyUsedLock = new Object(); public LazyObjectLoader(Library lib, SeekableInput seekableInput, CrossReference xref) { library = lib; m_SeekableInput = seekableInput; m_CrossReference = xref; leastRecentlyUsed = new ArrayList<ObjectStream>(256); } public boolean loadObject(Reference reference) { if (reference == null || library == null || m_CrossReference == null) return false; int objNum = reference.getObjectNumber(); CrossReference.Entry entry = m_CrossReference.getEntryForObject(objNum); if (entry == null) return false; boolean gotSomething = false; if (entry instanceof CrossReference.UsedEntry) { try { if (m_SeekableInput != null) { m_SeekableInput.beginThreadAccess(); CrossReference.UsedEntry usedEntry = (CrossReference.UsedEntry) entry; long position = usedEntry.getFilePositionOfObject(); long savedPosition = m_SeekableInput.getAbsolutePosition(); m_SeekableInput.seekAbsolute(position); Parser parser = new Parser(m_SeekableInput); Object ob = parser.getObject(library); gotSomething = (ob != null); m_SeekableInput.seekAbsolute(savedPosition); } } catch (Exception e) { logger.log(Level.SEVERE, "Error loading object instance: " + reference.toString(), e); } finally { if (m_SeekableInput != null) m_SeekableInput.endThreadAccess(); } } else if (entry instanceof CrossReference.CompressedEntry) { try { CrossReference.CompressedEntry compressedEntry = (CrossReference.CompressedEntry) entry; int objectStreamsObjectNumber = compressedEntry.getObjectNumberOfContainingObjectStream(); int objectIndex = compressedEntry.getIndexWithinObjectStream(); Reference objectStreamRef = new Reference(objectStreamsObjectNumber, 0); ObjectStream objectStream = (ObjectStream) library.getObject(objectStreamRef); if (objectStream != null) { synchronized (leastRectlyUsedLock) { leastRecentlyUsed.remove(objectStream); leastRecentlyUsed.add(objectStream); } gotSomething = objectStream.loadObject(library, objectIndex); } } catch (Exception e) { logger.log(Level.SEVERE, "Error loading object instance: " + reference.toString(), e); } } return gotSomething; } public boolean haveEntry(Reference reference) { if (reference == null || m_CrossReference == null) return false; int objNum = reference.getObjectNumber(); CrossReference.Entry entry = m_CrossReference.getEntryForObject(objNum); return (entry != null); } public PTrailer loadTrailer(long position) { PTrailer trailer = null; try { if (m_SeekableInput != null) { m_SeekableInput.beginThreadAccess(); long savedPosition = m_SeekableInput.getAbsolutePosition(); m_SeekableInput.seekAbsolute(position); Parser parser = new Parser(m_SeekableInput); Object obj = parser.getObject(library); if (obj instanceof PObject) obj = ((PObject) obj).getObject(); trailer = (PTrailer) obj; if (trailer != null) trailer.setPosition(position); m_SeekableInput.seekAbsolute(savedPosition); } } catch (Exception e) { logger.log(Level.FINE, "Error loading PTrailer instance: " + position, e); } finally { if (m_SeekableInput != null) m_SeekableInput.endThreadAccess(); } return trailer; } public void dispose() { library = null; m_SeekableInput = null; m_CrossReference = null; if (leastRecentlyUsed != null) { leastRecentlyUsed.clear(); leastRecentlyUsed = null; } } // // MemoryManagerDelegate interface // public boolean reduceMemory(int reductionPolicy) { int numToDo = 0; synchronized (leastRectlyUsedLock) { int lruSize = leastRecentlyUsed.size(); if (reductionPolicy == MemoryManagerDelegate.REDUCE_AGGRESSIVELY) { numToDo = lruSize * 75 / 100; } else if (reductionPolicy == MemoryManagerDelegate.REDUCE_SOMEWHAT) { if (lruSize > 5) numToDo = lruSize * 50 / 100; else if (lruSize > 0) numToDo = 1; } //System.out.println("LazyObjectLoader.reduceMemory() reductionPolicy: " + reductionPolicy + ", numToDo: " + numToDo + ", lruSize: " + lruSize); for (int i = 0; i < numToDo; i++) { ObjectStream objStm = leastRecentlyUsed.remove(0); if (objStm != null) objStm.dispose(true); } } return numToDo > 0; } /** * Get the documents library object. * * @return documents library object. */ public Library getLibrary() { return library; } }