|  | /* | 
|  | * Copyright (C) 2005, 2006, 2008 Apple Inc. All rights reserved. | 
|  | * Copyright (C) 2010 Google Inc. All rights reserved. | 
|  | * | 
|  | * Redistribution and use in source and binary forms, with or without | 
|  | * modification, are permitted provided that the following conditions are | 
|  | * met: | 
|  | * | 
|  | *     * Redistributions of source code must retain the above copyright | 
|  | * notice, this list of conditions and the following disclaimer. | 
|  | *     * Redistributions in binary form must reproduce the above | 
|  | * copyright notice, this list of conditions and the following disclaimer | 
|  | * in the documentation and/or other materials provided with the | 
|  | * distribution. | 
|  | *     * Neither the name of Google Inc. nor the names of its | 
|  | * contributors may be used to endorse or promote products derived from | 
|  | * this software without specific prior written permission. | 
|  | * | 
|  | * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | 
|  | * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | 
|  | * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR | 
|  | * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT | 
|  | * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | 
|  | * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | 
|  | * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, | 
|  | * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY | 
|  | * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | 
|  | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | 
|  | * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
|  | */ | 
|  |  | 
|  | #include "sky/engine/config.h" | 
|  | #include "sky/engine/platform/text/LineEnding.h" | 
|  |  | 
|  | #include "sky/engine/wtf/text/CString.h" | 
|  | #include "sky/engine/wtf/text/WTFString.h" | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | class OutputBuffer { | 
|  | public: | 
|  | virtual char* allocate(size_t) = 0; | 
|  | virtual void copy(const CString&) = 0; | 
|  | virtual ~OutputBuffer() { } | 
|  | }; | 
|  |  | 
|  | class CStringBuffer final : public OutputBuffer { | 
|  | public: | 
|  | CStringBuffer(CString& buffer) | 
|  | : m_buffer(buffer) | 
|  | { | 
|  | } | 
|  | virtual ~CStringBuffer() { } | 
|  |  | 
|  | virtual char* allocate(size_t size) override | 
|  | { | 
|  | char* ptr; | 
|  | m_buffer = CString::newUninitialized(size, ptr); | 
|  | return ptr; | 
|  | } | 
|  |  | 
|  | virtual void copy(const CString& source) override | 
|  | { | 
|  | m_buffer = source; | 
|  | } | 
|  |  | 
|  | const CString& buffer() const { return m_buffer; } | 
|  |  | 
|  | private: | 
|  | CString m_buffer; | 
|  | }; | 
|  |  | 
|  | class VectorCharAppendBuffer final : public OutputBuffer { | 
|  | public: | 
|  | VectorCharAppendBuffer(Vector<char>& buffer) | 
|  | : m_buffer(buffer) | 
|  | { | 
|  | } | 
|  | virtual ~VectorCharAppendBuffer() { } | 
|  |  | 
|  | virtual char* allocate(size_t size) override | 
|  | { | 
|  | size_t oldSize = m_buffer.size(); | 
|  | m_buffer.grow(oldSize + size); | 
|  | return m_buffer.data() + oldSize; | 
|  | } | 
|  |  | 
|  | virtual void copy(const CString& source) override | 
|  | { | 
|  | m_buffer.append(source.data(), source.length()); | 
|  | } | 
|  |  | 
|  | private: | 
|  | Vector<char>& m_buffer; | 
|  | }; | 
|  |  | 
|  | void internalNormalizeLineEndingsToCRLF(const CString& from, OutputBuffer& buffer) | 
|  | { | 
|  | // Compute the new length. | 
|  | size_t newLen = 0; | 
|  | const char* p = from.data(); | 
|  | while (p < from.data() + from.length()) { | 
|  | char c = *p++; | 
|  | if (c == '\r') { | 
|  | // Safe to look ahead because of trailing '\0'. | 
|  | if (*p != '\n') { | 
|  | // Turn CR into CRLF. | 
|  | newLen += 2; | 
|  | } | 
|  | } else if (c == '\n') { | 
|  | // Turn LF into CRLF. | 
|  | newLen += 2; | 
|  | } else { | 
|  | // Leave other characters alone. | 
|  | newLen += 1; | 
|  | } | 
|  | } | 
|  | if (newLen < from.length()) | 
|  | return; | 
|  |  | 
|  | if (newLen == from.length()) { | 
|  | buffer.copy(from); | 
|  | return; | 
|  | } | 
|  |  | 
|  | p = from.data(); | 
|  | char* q = buffer.allocate(newLen); | 
|  |  | 
|  | // Make a copy of the string. | 
|  | while (p < from.data() + from.length()) { | 
|  | char c = *p++; | 
|  | if (c == '\r') { | 
|  | // Safe to look ahead because of trailing '\0'. | 
|  | if (*p != '\n') { | 
|  | // Turn CR into CRLF. | 
|  | *q++ = '\r'; | 
|  | *q++ = '\n'; | 
|  | } | 
|  | } else if (c == '\n') { | 
|  | // Turn LF into CRLF. | 
|  | *q++ = '\r'; | 
|  | *q++ = '\n'; | 
|  | } else { | 
|  | // Leave other characters alone. | 
|  | *q++ = c; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | }; | 
|  |  | 
|  | namespace blink { | 
|  |  | 
|  | void normalizeToCROrLF(const CString& from, Vector<char>& result, bool toCR); | 
|  |  | 
|  | // Normalize all line-endings to CR or LF. | 
|  | void normalizeToCROrLF(const CString& from, Vector<char>& result, bool toCR) | 
|  | { | 
|  | // Compute the new length. | 
|  | size_t newLen = 0; | 
|  | bool needFix = false; | 
|  | const char* p = from.data(); | 
|  | char fromEndingChar = toCR ? '\n' : '\r'; | 
|  | char toEndingChar = toCR ? '\r' : '\n'; | 
|  | while (p < from.data() + from.length()) { | 
|  | char c = *p++; | 
|  | if (c == '\r' && *p == '\n') { | 
|  | // Turn CRLF into CR or LF. | 
|  | p++; | 
|  | needFix = true; | 
|  | } else if (c == fromEndingChar) { | 
|  | // Turn CR/LF into LF/CR. | 
|  | needFix = true; | 
|  | } | 
|  | newLen += 1; | 
|  | } | 
|  |  | 
|  | // Grow the result buffer. | 
|  | p = from.data(); | 
|  | size_t oldResultSize = result.size(); | 
|  | result.grow(oldResultSize + newLen); | 
|  | char* q = result.data() + oldResultSize; | 
|  |  | 
|  | // If no need to fix the string, just copy the string over. | 
|  | if (!needFix) { | 
|  | memcpy(q, p, from.length()); | 
|  | return; | 
|  | } | 
|  |  | 
|  | // Make a copy of the string. | 
|  | while (p < from.data() + from.length()) { | 
|  | char c = *p++; | 
|  | if (c == '\r' && *p == '\n') { | 
|  | // Turn CRLF or CR into CR or LF. | 
|  | p++; | 
|  | *q++ = toEndingChar; | 
|  | } else if (c == fromEndingChar) { | 
|  | // Turn CR/LF into LF/CR. | 
|  | *q++ = toEndingChar; | 
|  | } else { | 
|  | // Leave other characters alone. | 
|  | *q++ = c; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | CString normalizeLineEndingsToCRLF(const CString& from) | 
|  | { | 
|  | if (!from.length()) | 
|  | return from; | 
|  | CString result; | 
|  | CStringBuffer buffer(result); | 
|  | internalNormalizeLineEndingsToCRLF(from, buffer); | 
|  | return buffer.buffer(); | 
|  | } | 
|  |  | 
|  | void normalizeLineEndingsToCR(const CString& from, Vector<char>& result) | 
|  | { | 
|  | normalizeToCROrLF(from, result, true); | 
|  | } | 
|  |  | 
|  | void normalizeLineEndingsToLF(const CString& from, Vector<char>& result) | 
|  | { | 
|  | normalizeToCROrLF(from, result, false); | 
|  | } | 
|  |  | 
|  | void normalizeLineEndingsToNative(const CString& from, Vector<char>& result) | 
|  | { | 
|  | normalizeLineEndingsToLF(from, result); | 
|  | } | 
|  |  | 
|  | } // namespace blink |