Mercurial > hgrepos > Python2 > PyMuPDF
view mupdf-source/include/mupdf/fitz/heap-imp.h @ 46:7ee69f120f19 default tip
>>>>> tag v1.26.5+1 for changeset b74429b0f5c4
| author | Franz Glasner <fzglas.hg@dom66.de> |
|---|---|
| date | Sat, 11 Oct 2025 17:17:30 +0200 |
| parents | b50eed0cc0ef |
| children |
line wrap: on
line source
// Copyright (C) 2004-2025 Artifex Software, Inc. // // This file is part of MuPDF. // // MuPDF is free software: you can redistribute it and/or modify it under the // terms of the GNU Affero General Public License as published by the Free // Software Foundation, either version 3 of the License, or (at your option) // any later version. // // MuPDF is distributed in the hope that it will be useful, but WITHOUT ANY // WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS // FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more // details. // // You should have received a copy of the GNU Affero General Public License // along with MuPDF. If not, see <https://www.gnu.org/licenses/agpl-3.0.en.html> // // Alternative licensing terms are available from the licensor. // For commercial licensing, see <https://www.artifex.com/> or contact // Artifex Software, Inc., 39 Mesa Street, Suite 108A, San Francisco, // CA 94129, USA, for further information. /* This file has preprocessor magic in it to instantiate both * prototypes and implementations for heap sorting structures * of various different types. Effectively, it's templating for * C. * * If you are including this file directly without intending to * be instantiating a new set of heap sort functions, you are * doing the wrong thing. */ #ifndef MUPDF_FITZ_HEAP_I_KNOW_WHAT_IM_DOING #error Do not include heap-imp.h unless you know what you are doing #endif #define HEAP_XCAT(A,B) A##B #define HEAP_CAT(A,B) HEAP_XCAT(A,B) #ifndef MUPDF_FITZ_HEAP_IMPLEMENT typedef struct { int max; int len; HEAP_CONTAINER_TYPE *heap; } HEAP_TYPE_NAME; #endif void HEAP_CAT(HEAP_TYPE_NAME,_insert)(fz_context *ctx, HEAP_TYPE_NAME *heap, HEAP_CONTAINER_TYPE v #ifndef HEAP_CMP , int (*HEAP_CMP)(HEAP_CONTAINER_TYPE *a, HEAP_CONTAINER_TYPE *b) #endif ) #ifndef MUPDF_FITZ_HEAP_IMPLEMENT ; #else { int i; HEAP_CONTAINER_TYPE *h; if (heap->max == heap->len) { int m = heap->max * 2; if (m == 0) m = 32; heap->heap = (HEAP_CONTAINER_TYPE *)fz_realloc(ctx, heap->heap, sizeof(*heap->heap) * m); heap->max = m; } h = heap->heap; /* Insert it into the heap. Consider inserting at position i, and * then 'heapify' back. We can delay the actual insertion to the * end of the process. */ i = heap->len++; while (i != 0) { int parent_idx = (i-1)/2; HEAP_CONTAINER_TYPE *parent_val = &h[parent_idx]; if (HEAP_CMP(parent_val, &v) > 0) break; h[i] = h[parent_idx]; i = parent_idx; } h[i] = v; } #endif void HEAP_CAT(HEAP_TYPE_NAME,_sort)(fz_context *ctx, HEAP_TYPE_NAME *heap #ifndef HEAP_CMP , int (*HEAP_CMP)(HEAP_CONTAINER_TYPE *a, HEAP_CONTAINER_TYPE *b) #endif ) #ifndef MUPDF_FITZ_HEAP_IMPLEMENT ; #else { int j; HEAP_CONTAINER_TYPE *h = heap->heap; /* elements j to len are always sorted. 0 to j are always a valid heap. Gradually move j to 0. */ for (j = heap->len-1; j > 0; j--) { int k; HEAP_CONTAINER_TYPE val; /* Swap max element with j. Invariant valid for next value to j. */ val = h[j]; h[j] = h[0]; /* Now reform the heap. 0 to k is a valid heap. */ k = 0; while (1) { int kid = k*2+1; if (kid >= j) break; if (kid+1 < j && (HEAP_CMP(&h[kid+1], &h[kid])) > 0) kid++; if ((HEAP_CMP(&val, &h[kid])) > 0) break; h[k] = h[kid]; k = kid; } h[k] = val; } } #endif void HEAP_CAT(HEAP_TYPE_NAME,_uniq)(fz_context *ctx, HEAP_TYPE_NAME *heap #ifndef HEAP_CMP , int (*HEAP_CMP)(HEAP_CONTAINER_TYPE *a, HEAP_CONTAINER_TYPE *b) #endif ) #ifndef MUPDF_FITZ_HEAP_IMPLEMENT ; #else { int n = heap->len; int i, j = 0; HEAP_CONTAINER_TYPE *h = heap->heap; if (n == 0) return; j = 0; for (i = 1; i < n; i++) { if (HEAP_CMP(&h[j], &h[i]) == 0) continue; j++; if (i != j) h[j] = h[i]; } heap->len = j+1; } #endif #ifdef HEAP_DUMP void HEAP_CAT(HEAP_TYPE_NAME,_dump)(fz_context *ctx, fz_output *out, HEAP_TYPE_NAME *heap) #ifndef MUPDF_FITZ_HEAP_IMPLEMENT ; #else { int n = heap->len; int i; HEAP_CONTAINER_TYPE *h = heap->heap; fz_write_printf(ctx, out, "Heap %p len %d:\n", heap, n); for (i = 0; i < n; i++) HEAP_DUMP(ctx, out, i, &h[i]); } #endif void HEAP_CAT(HEAP_TYPE_NAME,_debug)(fz_context *ctx, HEAP_TYPE_NAME *heap) #ifndef MUPDF_FITZ_HEAP_IMPLEMENT ; #else { HEAP_CAT(HEAP_TYPE_NAME,_dump)(ctx, fz_stddbg(ctx), heap); } #endif #endif #undef HEAP_CONTAINER_TYPE #undef HEAP_TYPE_NAME #undef HEAP_CMP #undef HEAP_XCAT #undef HEAP_CAT #undef HEAP_DUMP
