2021-01-28 22:47:25 -08:00
|
|
|
//
|
2021-01-28 22:58:17 -08:00
|
|
|
// utils.h
|
2021-07-28 22:34:51 -07:00
|
|
|
// Copyright 2021 Bruce Hill
|
2021-01-28 23:00:34 -08:00
|
|
|
// Released under the MIT license with the Commons Clause
|
2021-01-28 22:47:25 -08:00
|
|
|
//
|
2021-07-28 22:34:51 -07:00
|
|
|
// This file contains some definitions of some utility macros and functions.
|
2021-01-28 22:47:25 -08:00
|
|
|
//
|
2019-09-30 17:06:27 -07:00
|
|
|
|
2021-01-28 22:13:51 -08:00
|
|
|
#ifndef FILE_UTILS__H
|
|
|
|
#define FILE_UTILS__H
|
2021-01-28 21:53:48 -08:00
|
|
|
|
2021-04-17 00:09:49 -07:00
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#ifndef streq
|
|
|
|
#define streq(a,b) (strcmp(a,b)==0)
|
|
|
|
#endif
|
2020-04-10 00:12:57 -07:00
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
#define MAX(a,b) ((a) < (b) ? (b) : (a))
|
|
|
|
#define MIN(a,b) ((a) > (b) ? (b) : (a))
|
2021-01-28 22:13:51 -08:00
|
|
|
|
|
|
|
// Platform-dependent time strucutre accessors:
|
2019-09-30 17:06:27 -07:00
|
|
|
#ifdef __APPLE__
|
2024-06-09 21:56:26 -07:00
|
|
|
#define get_mtime(s) (s).st_mtimespec
|
|
|
|
#define get_atime(s) (s).st_atimespec
|
|
|
|
#define get_ctime(s) (s).st_ctimespec
|
2019-09-30 17:06:27 -07:00
|
|
|
#else
|
2024-06-09 21:56:26 -07:00
|
|
|
#define get_mtime(s) (s).st_mtim
|
|
|
|
#define get_atime(s) (s).st_atim
|
|
|
|
#define get_ctime(s) (s).st_ctim
|
2019-09-30 17:06:27 -07:00
|
|
|
#endif
|
|
|
|
|
2021-01-28 22:13:51 -08:00
|
|
|
// Entry macros
|
|
|
|
#define IS_SELECTED(e) (((e)->selected.atme) != NULL)
|
|
|
|
#define IS_VIEWED(e) ((e)->index >= 0)
|
|
|
|
#define IS_LOADED(e) ((e)->hash.atme != NULL)
|
|
|
|
|
|
|
|
#define E_ISDIR(e) (S_ISDIR(S_ISLNK((e)->info.st_mode) ? (e)->linkedmode : (e)->info.st_mode))
|
|
|
|
|
|
|
|
// Linked list macros
|
2019-10-27 14:58:23 -07:00
|
|
|
#define LL_PREPEND(head, node, name) do { \
|
|
|
|
((node)->name).atme = &(head); \
|
|
|
|
((node)->name).next = head; \
|
|
|
|
if (head) ((head)->name).atme = &(((node)->name).next); \
|
|
|
|
head = node; \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define LL_REMOVE(node, name) do { \
|
|
|
|
if (((node)->name).next) \
|
|
|
|
((__typeof__(node))(node)->name.next)->name.atme = ((node)->name).atme; \
|
2020-02-23 18:00:39 -08:00
|
|
|
if (((node)->name).atme) \
|
|
|
|
*(((node)->name).atme) = ((node)->name).next; \
|
2019-10-27 14:58:23 -07:00
|
|
|
((node)->name).atme = NULL; \
|
|
|
|
((node)->name).next = NULL; \
|
2019-10-13 19:34:16 -07:00
|
|
|
} while (0)
|
|
|
|
|
2021-07-29 00:23:32 -07:00
|
|
|
#define LEN(a) (sizeof(a)/sizeof(a[0]))
|
|
|
|
#define FOREACH(type, var, array) for (type var = array; (var) < &(array)[LEN(array)]; var++)
|
|
|
|
|
2021-07-28 22:34:51 -07:00
|
|
|
#define S1(x) #x
|
|
|
|
#define S2(x) S1(x)
|
|
|
|
#define __LOCATION__ __FILE__ ":" S2(__LINE__)
|
|
|
|
|
|
|
|
// Error checking helper macros:
|
|
|
|
#define nonnegative(exp, ...) check_nonnegative(exp, __LOCATION__ ": `" #exp "` " __VA_ARGS__)
|
|
|
|
#define nonnull(exp, ...) check_nonnull(exp, __LOCATION__ ": `" #exp "` " __VA_ARGS__)
|
|
|
|
// Error-checking memory allocation helper macros:
|
|
|
|
#define new(t) check_nonnull(calloc(1, sizeof(t)), __LOCATION__ ": new(" #t ") failed")
|
|
|
|
#define new_bytes(n) check_nonnull(calloc(1, n), __LOCATION__ ": new_bytes(" #n ") failed")
|
|
|
|
#define grow(obj, new_count) check_nonnull(reallocarray(obj, (new_count), sizeof(obj[0])), __LOCATION__ ": grow(" #obj ", " #new_count ") failed")
|
|
|
|
#define check_strdup(s) check_nonnull(strdup(s), __LOCATION__ ": check_strdup(" #s ") failed")
|
|
|
|
|
|
|
|
int check_nonnegative(int negative_err, const char *err_msg, ...);
|
|
|
|
__attribute__((returns_nonnull))
|
|
|
|
void *check_nonnull(void *p, const char *err_msg, ...);
|
|
|
|
__attribute__((nonnull))
|
|
|
|
void delete(void *p);
|
2019-09-30 17:06:27 -07:00
|
|
|
|
2020-12-30 22:47:49 -08:00
|
|
|
#endif
|
2021-09-29 14:40:03 -07:00
|
|
|
// vim: ts=4 sw=0 et cino=L2,l1,(0,W4,m1,\:0
|