#region PDFsharp - A .NET library for processing PDF // // Authors: // Stefan Lange // // Copyright (c) 2005-2017 empira Software GmbH, Cologne Area (Germany) // // http://www.pdfsharp.com // http://sourceforge.net/projects/pdfsharp // // Permission is hereby granted, free of charge, to any person obtaining a // copy of this software and associated documentation files (the "Software"), // to deal in the Software without restriction, including without limitation // the rights to use, copy, modify, merge, publish, distribute, sublicense, // and/or sell copies of the Software, and to permit persons to whom the // Software is furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included // in all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL // THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER // DEALINGS IN THE SOFTWARE. #endregion using System; using System.Collections.Generic; using System.Diagnostics; namespace PdfSharp.Pdf.IO { /// /// Represents the stack for the shift-reduce parser. It seems that it is only needed for /// reduction of indirect references. /// internal class ShiftStack { // TODO: make Lexer.PeekChars(20) and scan for 'R' to detect indirect references public ShiftStack() { _items = new List(); } public PdfItem[] ToArray(int start, int length) { PdfItem[] items = new PdfItem[length]; for (int i = 0, j = start; i < length; i++, j++) items[i] = _items[j]; return items; } /// /// Gets the stack pointer index. /// // ReSharper disable InconsistentNaming public int SP // ReSharper restore InconsistentNaming { get { return _sp; } } /// /// Gets the value at the specified index. Valid index is in range 0 up to sp-1. /// public PdfItem this[int index] { get { if (index >= _sp) throw new ArgumentOutOfRangeException("index", index, "Value greater than stack index."); return _items[index]; } } /// /// Gets an item relative to the current stack pointer. The index must be a negative value (-1, -2, etc.). /// public PdfItem GetItem(int relativeIndex) { if (relativeIndex >= 0 || -relativeIndex > _sp) throw new ArgumentOutOfRangeException("relativeIndex", relativeIndex, "Value out of stack range."); return _items[_sp + relativeIndex]; } /// /// Gets an item relative to the current stack pointer. The index must be a negative value (-1, -2, etc.). /// public int GetInteger(int relativeIndex) { if (relativeIndex >= 0 || -relativeIndex > _sp) throw new ArgumentOutOfRangeException("relativeIndex", relativeIndex, "Value out of stack range."); return ((PdfInteger)_items[_sp + relativeIndex]).Value; } /// /// Pushes the specified item onto the stack. /// public void Shift(PdfItem item) { Debug.Assert(item != null); _items.Add(item); _sp++; } /// /// Replaces the last 'count' items with the specified item. /// public void Reduce(int count) { if (count > _sp) throw new ArgumentException("count causes stack underflow."); _items.RemoveRange(_sp - count, count); _sp -= count; } /// /// Replaces the last 'count' items with the specified item. /// public void Reduce(PdfItem item, int count) { Debug.Assert(item != null); Reduce(count); _items.Add(item); _sp++; } /// /// The stack pointer index. Points to the next free item. /// int _sp; /// /// An array representing the stack. /// readonly List _items; } }