mirror of
				https://github.com/espressif/esp-idf.git
				synced 2025-11-04 14:14:11 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			184 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			184 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * SPDX-FileCopyrightText: 2013-2019 Tom G. Huang
 | 
						|
 *
 | 
						|
 * SPDX-License-Identifier: BSD-3-Clause
 | 
						|
 */
 | 
						|
/*******************************************************************************
 | 
						|
 * arg_utils: Implements memory, panic, and other utility functions
 | 
						|
 *
 | 
						|
 * This file is part of the argtable3 library.
 | 
						|
 *
 | 
						|
 * Copyright (C) 2013-2019 Tom G. Huang
 | 
						|
 * <tomghuang@gmail.com>
 | 
						|
 * 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 STEWART HEITMANN 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 STEWART HEITMANN 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 "argtable3.h"
 | 
						|
 | 
						|
#ifndef ARG_AMALGAMATION
 | 
						|
#include "argtable3_private.h"
 | 
						|
#endif
 | 
						|
 | 
						|
#include <stdarg.h>
 | 
						|
#include <stdio.h>
 | 
						|
#include <stdlib.h>
 | 
						|
#include <string.h>
 | 
						|
 | 
						|
static void panic(const char* fmt, ...);
 | 
						|
static arg_panicfn* s_panic = panic;
 | 
						|
 | 
						|
void dbg_printf(const char* fmt, ...) {
 | 
						|
    va_list args;
 | 
						|
    va_start(args, fmt);
 | 
						|
    vfprintf(stderr, fmt, args);
 | 
						|
    va_end(args);
 | 
						|
}
 | 
						|
 | 
						|
static void panic(const char* fmt, ...) {
 | 
						|
    va_list args;
 | 
						|
    char* s;
 | 
						|
 | 
						|
    va_start(args, fmt);
 | 
						|
    vfprintf(stderr, fmt, args);
 | 
						|
    va_end(args);
 | 
						|
 | 
						|
#if defined(_MSC_VER)
 | 
						|
#pragma warning(push)
 | 
						|
#pragma warning(disable : 4996)
 | 
						|
#endif
 | 
						|
    s = getenv("EF_DUMPCORE");
 | 
						|
#if defined(_MSC_VER)
 | 
						|
#pragma warning(pop)
 | 
						|
#endif
 | 
						|
 | 
						|
    if (s != NULL && *s != '\0') {
 | 
						|
        abort();
 | 
						|
    } else {
 | 
						|
        exit(EXIT_FAILURE);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
void arg_set_panic(arg_panicfn* proc) {
 | 
						|
    s_panic = proc;
 | 
						|
}
 | 
						|
 | 
						|
void* xmalloc(size_t size) {
 | 
						|
    void* ret = malloc(size);
 | 
						|
    if (!ret) {
 | 
						|
        s_panic("Out of memory!\n");
 | 
						|
    }
 | 
						|
    return ret;
 | 
						|
}
 | 
						|
 | 
						|
void* xcalloc(size_t count, size_t size) {
 | 
						|
    size_t allocated_count = count && size ? count : 1;
 | 
						|
    size_t allocated_size = count && size ? size : 1;
 | 
						|
    void* ret = calloc(allocated_count, allocated_size);
 | 
						|
    if (!ret) {
 | 
						|
        s_panic("Out of memory!\n");
 | 
						|
    }
 | 
						|
    return ret;
 | 
						|
}
 | 
						|
 | 
						|
void* xrealloc(void* ptr, size_t size) {
 | 
						|
    size_t allocated_size = size ? size : 1;
 | 
						|
    void* ret = realloc(ptr, allocated_size);
 | 
						|
    if (!ret) {
 | 
						|
        s_panic("Out of memory!\n");
 | 
						|
    }
 | 
						|
    return ret;
 | 
						|
}
 | 
						|
 | 
						|
void xfree(void* ptr) {
 | 
						|
    free(ptr);
 | 
						|
}
 | 
						|
 | 
						|
static void merge(void* data, int esize, int i, int j, int k, arg_comparefn* comparefn) {
 | 
						|
    char* a = (char*)data;
 | 
						|
    char* m;
 | 
						|
    int ipos, jpos, mpos;
 | 
						|
 | 
						|
    /* Initialize the counters used in merging. */
 | 
						|
    ipos = i;
 | 
						|
    jpos = j + 1;
 | 
						|
    mpos = 0;
 | 
						|
 | 
						|
    /* Allocate storage for the merged elements. */
 | 
						|
    m = (char*)xmalloc((size_t)(esize * ((k - i) + 1)));
 | 
						|
 | 
						|
    /* Continue while either division has elements to merge. */
 | 
						|
    while (ipos <= j || jpos <= k) {
 | 
						|
        if (ipos > j) {
 | 
						|
            /* The left division has no more elements to merge. */
 | 
						|
            while (jpos <= k) {
 | 
						|
                memcpy(&m[mpos * esize], &a[jpos * esize], (size_t)esize);
 | 
						|
                jpos++;
 | 
						|
                mpos++;
 | 
						|
            }
 | 
						|
 | 
						|
            continue;
 | 
						|
        } else if (jpos > k) {
 | 
						|
            /* The right division has no more elements to merge. */
 | 
						|
            while (ipos <= j) {
 | 
						|
                memcpy(&m[mpos * esize], &a[ipos * esize], (size_t)esize);
 | 
						|
                ipos++;
 | 
						|
                mpos++;
 | 
						|
            }
 | 
						|
 | 
						|
            continue;
 | 
						|
        }
 | 
						|
 | 
						|
        /* Append the next ordered element to the merged elements. */
 | 
						|
        if (comparefn(&a[ipos * esize], &a[jpos * esize]) < 0) {
 | 
						|
            memcpy(&m[mpos * esize], &a[ipos * esize], (size_t)esize);
 | 
						|
            ipos++;
 | 
						|
            mpos++;
 | 
						|
        } else {
 | 
						|
            memcpy(&m[mpos * esize], &a[jpos * esize], (size_t)esize);
 | 
						|
            jpos++;
 | 
						|
            mpos++;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    /* Prepare to pass back the merged data. */
 | 
						|
    memcpy(&a[i * esize], m, (size_t)(esize * ((k - i) + 1)));
 | 
						|
    xfree(m);
 | 
						|
}
 | 
						|
 | 
						|
void arg_mgsort(void* data, int size, int esize, int i, int k, arg_comparefn* comparefn) {
 | 
						|
    int j;
 | 
						|
 | 
						|
    /* Stop the recursion when no more divisions can be made. */
 | 
						|
    if (i < k) {
 | 
						|
        /* Determine where to divide the elements. */
 | 
						|
        j = (int)(((i + k - 1)) / 2);
 | 
						|
 | 
						|
        /* Recursively sort the two divisions. */
 | 
						|
        arg_mgsort(data, size, esize, i, j, comparefn);
 | 
						|
        arg_mgsort(data, size, esize, j + 1, k, comparefn);
 | 
						|
        merge(data, esize, i, j, k, comparefn);
 | 
						|
    }
 | 
						|
}
 |