Home | History | Annotate | Download | only in CoreIPC
      1 /*
      2  * Copyright (C) 2010 Apple Inc. All rights reserved.
      3  *
      4  * Redistribution and use in source and binary forms, with or without
      5  * modification, are permitted provided that the following conditions
      6  * are met:
      7  * 1. Redistributions of source code must retain the above copyright
      8  *    notice, this list of conditions and the following disclaimer.
      9  * 2. Redistributions in binary form must reproduce the above copyright
     10  *    notice, this list of conditions and the following disclaimer in the
     11  *    documentation and/or other materials provided with the distribution.
     12  *
     13  * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
     14  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
     15  * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
     16  * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
     17  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
     18  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
     19  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
     20  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
     21  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
     22  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
     23  * THE POSSIBILITY OF SUCH DAMAGE.
     24  */
     25 
     26 #include "config.h"
     27 #include "ArgumentEncoder.h"
     28 
     29 #include <algorithm>
     30 #include <stdio.h>
     31 
     32 namespace CoreIPC {
     33 
     34 PassOwnPtr<ArgumentEncoder> ArgumentEncoder::create(uint64_t destinationID)
     35 {
     36     return adoptPtr(new ArgumentEncoder(destinationID));
     37 }
     38 
     39 ArgumentEncoder::ArgumentEncoder(uint64_t destinationID)
     40     : m_buffer(0)
     41     , m_bufferPointer(0)
     42     , m_bufferSize(0)
     43     , m_bufferCapacity(0)
     44 {
     45     // Encode the destination ID.
     46     encodeUInt64(destinationID);
     47 }
     48 
     49 ArgumentEncoder::~ArgumentEncoder()
     50 {
     51     if (m_buffer)
     52         fastFree(m_buffer);
     53 #if !PLATFORM(QT) && !PLATFORM(GTK)
     54     // FIXME: We need to dispose of the attachments in cases of failure.
     55 #else
     56     for (int i = 0; i < m_attachments.size(); ++i)
     57         m_attachments[i].dispose();
     58 #endif
     59 }
     60 
     61 static inline size_t roundUpToAlignment(size_t value, unsigned alignment)
     62 {
     63     return ((value + alignment - 1) / alignment) * alignment;
     64 }
     65 
     66 uint8_t* ArgumentEncoder::grow(unsigned alignment, size_t size)
     67 {
     68     size_t alignedSize = roundUpToAlignment(m_bufferSize, alignment);
     69 
     70     if (alignedSize + size > m_bufferCapacity) {
     71         size_t newCapacity = std::max(alignedSize + size, std::max(static_cast<size_t>(32), m_bufferCapacity + m_bufferCapacity / 4 + 1));
     72         if (!m_buffer)
     73             m_buffer = static_cast<uint8_t*>(fastMalloc(newCapacity));
     74         else
     75             m_buffer = static_cast<uint8_t*>(fastRealloc(m_buffer, newCapacity));
     76 
     77         // FIXME: What should we do if allocating memory fails?
     78 
     79         m_bufferCapacity = newCapacity;
     80     }
     81 
     82     m_bufferSize = alignedSize + size;
     83     m_bufferPointer = m_buffer + alignedSize + size;
     84 
     85     return m_buffer + alignedSize;
     86 }
     87 
     88 void ArgumentEncoder::encodeBytes(const uint8_t* bytes, size_t size)
     89 {
     90     // Encode the size.
     91     encodeUInt64(static_cast<uint64_t>(size));
     92 
     93     uint8_t* buffer = grow(1, size);
     94 
     95     memcpy(buffer, bytes, size);
     96 }
     97 
     98 void ArgumentEncoder::encodeBool(bool n)
     99 {
    100     uint8_t* buffer = grow(sizeof(n), sizeof(n));
    101 
    102     *reinterpret_cast<bool*>(buffer) = n;
    103 }
    104 
    105 void ArgumentEncoder::encodeUInt32(uint32_t n)
    106 {
    107     uint8_t* buffer = grow(sizeof(n), sizeof(n));
    108 
    109     *reinterpret_cast<uint32_t*>(buffer) = n;
    110 }
    111 
    112 void ArgumentEncoder::encodeUInt64(uint64_t n)
    113 {
    114     uint8_t* buffer = grow(sizeof(n), sizeof(n));
    115 
    116     *reinterpret_cast<uint64_t*>(buffer) = n;
    117 }
    118 
    119 void ArgumentEncoder::encodeInt32(int32_t n)
    120 {
    121     uint8_t* buffer = grow(sizeof(n), sizeof(n));
    122 
    123     *reinterpret_cast<int32_t*>(buffer) = n;
    124 }
    125 
    126 void ArgumentEncoder::encodeInt64(int64_t n)
    127 {
    128     uint8_t* buffer = grow(sizeof(n), sizeof(n));
    129 
    130     *reinterpret_cast<int64_t*>(buffer) = n;
    131 }
    132 
    133 void ArgumentEncoder::encodeFloat(float n)
    134 {
    135     uint8_t* buffer = grow(sizeof(n), sizeof(n));
    136 
    137     *reinterpret_cast<float*>(buffer) = n;
    138 }
    139 
    140 void ArgumentEncoder::encodeDouble(double n)
    141 {
    142     uint8_t* buffer = grow(sizeof(n), sizeof(n));
    143 
    144     *reinterpret_cast<double*>(buffer) = n;
    145 }
    146 
    147 void ArgumentEncoder::addAttachment(const Attachment& attachment)
    148 {
    149     m_attachments.append(attachment);
    150 }
    151 
    152 Vector<Attachment> ArgumentEncoder::releaseAttachments()
    153 {
    154     Vector<Attachment> newList;
    155     newList.swap(m_attachments);
    156     return newList;
    157 }
    158 
    159 #ifndef NDEBUG
    160 void ArgumentEncoder::debug()
    161 {
    162     printf("ArgumentEncoder::debug()\n");
    163     printf("Number of Attachments: %d\n", (int)m_attachments.size());
    164     printf("Size of buffer: %d\n", (int)m_bufferSize);
    165 }
    166 #endif
    167 
    168 } // namespace CoreIPC
    169