src/hbtools/hbbincssmaker/hbcssconverterutils.cpp
changeset 2 06ff229162e9
child 5 627c4a0fd0e7
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/hbtools/hbbincssmaker/hbcssconverterutils.cpp	Fri May 14 16:09:54 2010 +0300
@@ -0,0 +1,204 @@
+/****************************************************************************
+**
+** Copyright (C) 2008-2010 Nokia Corporation and/or its subsidiary(-ies).
+** All rights reserved.
+** Contact: Nokia Corporation (developer.feedback@nokia.com)
+**
+** This file is part of the HbTools module of the UI Extensions for Mobile.
+**
+** GNU Lesser General Public License Usage
+** This file may be used under the terms of the GNU Lesser General Public
+** License version 2.1 as published by the Free Software Foundation and
+** appearing in the file LICENSE.LGPL included in the packaging of this file.
+** Please review the following information to ensure the GNU Lesser General
+** Public License version 2.1 requirements will be met:
+** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
+**
+** In addition, as a special exception, Nokia gives you certain additional
+** rights.  These rights are described in the Nokia Qt LGPL Exception
+** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
+**
+** If you have questions regarding the use of this file, please contact
+** Nokia at developer.feedback@nokia.com.
+**
+****************************************************************************/
+
+#include <QMap>
+#include "hbcssconverterutils_p.h"
+#include "hbsharedmemorymanager_p.h"
+#include "hbsharedmemoryallocators_p.h"
+
+// Global list that stores pointers to the member variables where shared container instances
+// store offsets returned my memory allocator.
+// CSS converter utilizes it to automatically adjust the offsets if allocated cells are moved.
+
+// Map is used only to get faster lookups, item's value is obsolete
+static QMap<int *, int> registered;
+
+
+// Shared chunk allocation information for css data
+static int totalAllocated = 0;
+// Using map instead of hash to guarantee that the items are in order
+// so the cell with offset 0 which is the HbCss::StyleSheet structure
+// is always first in the cell list, so its offset does not get changed
+// when the chunk is defragmented.
+static QMap<int, int> cells;
+
+
+void HbCssConverterUtils::registerOffsetHolder(int *offset)
+{
+    registered.insert(offset, 1);
+}
+
+void HbCssConverterUtils::unregisterOffsetHolder(int *offset)
+{
+    registered.remove(offset);
+}
+
+
+QList<int *> HbCssConverterUtils::registeredOffsetHolders()
+{
+    return registered.keys();
+}
+
+void HbCssConverterUtils::unregisterAll()
+{
+    registered.clear();
+}
+
+
+void HbCssConverterUtils::cellAllocated(int offset, int size)
+{
+    cells.insert(offset, ALIGN(size));
+    totalAllocated += ALIGN(size);
+}
+
+void HbCssConverterUtils::cellFreed(int offset)
+{
+    int size = cells.value(offset, 0);
+    totalAllocated -= size;
+    cells.remove(offset);
+
+    if (size > 0) {
+        // Make sure there are no registered offset holders in the freed cell any more
+        GET_MEMORY_MANAGER(HbMemoryManager::SharedMemory);
+        HbSharedMemoryManager *shared = static_cast<HbSharedMemoryManager*>(manager);
+        const char *chunkBase = static_cast<const char *>(shared->base());
+
+        QList<int *> offsetHolders = HbCssConverterUtils::registeredOffsetHolders();
+        for (int i = 0; i<offsetHolders.count(); ++i) {
+            int *holder = offsetHolders.at(i);
+            if ((char*)holder >= chunkBase + offset && (char*)holder < chunkBase + offset + size) {
+                HbCssConverterUtils::unregisterOffsetHolder(holder);
+            }
+        }
+    }
+}
+
+void HbCssConverterUtils::cellMoved(int offset, int newOffset)
+{
+    int size = cells.value(offset, 0);
+
+    if (size > 0) {
+        // Check if there were registered offset holders in the old cell
+        // and register corresponding ones in the reallocated cell.
+        QList<int *> holders = HbCssConverterUtils::registeredOffsetHolders();
+
+        GET_MEMORY_MANAGER(HbMemoryManager::SharedMemory);
+        HbSharedMemoryManager *shared = static_cast<HbSharedMemoryManager*>(manager);
+        const char *chunkBase = static_cast<const char *>(shared->base());    
+        
+        for (int i=0; i<holders.count(); i++) {
+            int *holder = holders.at(i);
+            char *holderC = (char*)holder;
+            if (holderC >= chunkBase + offset && holderC < chunkBase + offset + size) {
+                HbCssConverterUtils::unregisterOffsetHolder(holder);
+                HbCssConverterUtils::registerOffsetHolder((int*)(holderC + newOffset - offset));
+            }
+        }
+    }
+}
+
+/**
+* Defragments the shared chunk contents and places defragmented buffer in the beginning of the chunk.
+* Registered chunk offset holders are updated during the process.
+* Returns the next free offset in the chunk.
+*/
+int HbCssConverterUtils::defragmentChunk()
+{
+    GET_MEMORY_MANAGER(HbMemoryManager::SharedMemory);
+    HbSharedMemoryManager *shared = static_cast<HbSharedMemoryManager*>(manager);
+
+    // Register shared cache pointer in chunk header as shared cache may also be moved in defragmentation
+    HbSharedChunkHeader *chunkHeader = static_cast<HbSharedChunkHeader*>(shared->base());    
+    HbCssConverterUtils::registerOffsetHolder(reinterpret_cast<int *>(&chunkHeader->sharedCacheOffset));
+
+    QList<int *> offsetHolders = HbCssConverterUtils::registeredOffsetHolders();
+
+    // Create new buffer where the current chunk contents are defragmented
+    void *buffer = ::malloc(shared->size());
+    int newCurrentOffset = 0;
+
+    // Create new cell order and update offset holders
+    QMap<int,int>::const_iterator i = cells.constBegin();
+
+    while (i != cells.constEnd()) {
+        // Get the old cell
+        int offset = i.key();
+        int size = i.value();
+        
+        // Update registered offset holders
+
+        // TODO: optimize this, now there's linear search for each cell!
+		for (int j=0; j<offsetHolders.count(); ++j) {
+			int *holder = offsetHolders.at(j);
+			if (*holder == offset) {
+				// Change stored offset value
+				*holder = newCurrentOffset + sizeof(HbSharedChunkHeader);
+			}
+		}
+
+        newCurrentOffset += size;
+        i++;
+    }
+
+    i = cells.constBegin();
+    newCurrentOffset = 0;
+
+    // Move allocated cells to a linear buffer
+    while (i != cells.constEnd()) {
+        // Get the old cell
+        int offset = i.key();
+        int size = i.value();
+        // Copy to new chunk
+        memcpy((char*)buffer + newCurrentOffset, (char*)shared->base() + offset, size);
+
+        newCurrentOffset += size;
+        i++;
+    }
+
+    // Free all cells from the shared chunk and move the defragmented buffer in the beginning of the chunk.
+    // Note that chunk memory management is screwed up after this point, so no more allocations should be
+    // done in it after this.
+
+    HbCssConverterUtils::unregisterAll();
+    QList<int> keys = cells.keys();
+
+    for (int j=0; j<keys.count(); ++j) {
+        shared->free(keys.at(j));
+    }
+
+    // CSS binary data is placed after the chunk header.
+    int cssBinaryOffset = sizeof(HbSharedChunkHeader);
+    char *address = HbMemoryUtils::getAddress<char>(HbMemoryManager::SharedMemory, cssBinaryOffset);
+    memcpy(address, buffer, newCurrentOffset);
+
+    cells.clear();
+    totalAllocated = 0;
+
+    // Free the temp buffer
+    ::free(buffer);
+
+    // Return the next free address in the chunk
+    return cssBinaryOffset + newCurrentOffset;
+}