/* ide-macros.h * * Copyright 2018-2019 Christian Hergert * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . * * SPDX-License-Identifier: GPL-3.0-or-later */ #pragma once #if !defined (IDE_CORE_INSIDE) && !defined (IDE_CORE_COMPILATION) # error "Only can be included directly." #endif #ifndef __GI_SCANNER__ #include #include "ide-global.h" #include "ide-object.h" #include "ide-version-macros.h" G_BEGIN_DECLS #define ide_str_empty0(str) (!(str) || !*(str)) #define ide_str_equal(str1,str2) (strcmp(str1,str2)==0) #define ide_str_equal0(str1,str2) (g_strcmp0(str1,str2)==0) #define ide_strv_empty0(strv) (((strv) == NULL) || ((strv)[0] == NULL)) #define ide_set_string(ptr,str) (ide_take_string((ptr), g_strdup(str))) #define ide_clear_param(pptr, pval) \ G_STMT_START { if (pptr) { *(pptr) = pval; }; } G_STMT_END #define IDE_IS_MAIN_THREAD() (g_thread_self() == ide_get_main_thread()) #define IDE_PTR_ARRAY_CLEAR_FREE_FUNC(ar) \ IDE_PTR_ARRAY_SET_FREE_FUNC(ar, NULL) #define IDE_PTR_ARRAY_SET_FREE_FUNC(ar, func) \ G_STMT_START { \ if ((ar) != NULL) \ g_ptr_array_set_free_func ((ar), (GDestroyNotify)(func)); \ } G_STMT_END #define IDE_PTR_ARRAY_STEAL_FULL(arptr) \ ({ IDE_PTR_ARRAY_CLEAR_FREE_FUNC (*(arptr)); \ g_steal_pointer ((arptr)); }) static inline gpointer ide_ptr_array_steal_index (GPtrArray *array, guint index, GDestroyNotify free_func) { gpointer ret; if (index >= array->len) return NULL; g_ptr_array_set_free_func (array, NULL); ret = g_ptr_array_index (array, index); g_ptr_array_remove_index (array, index); g_ptr_array_set_free_func (array, free_func); return ret; } static inline gboolean ide_error_ignore (const GError *error) { return g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED) || g_error_matches (error, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED); } static inline void _g_object_unref0 (gpointer instance) { if (instance) g_object_unref (instance); } static inline gboolean ide_take_string (gchar **ptr, gchar *str) { if (*ptr != str) { g_free (*ptr); *ptr = str; return TRUE; } return FALSE; } static inline void ide_clear_string (gchar **ptr) { g_free (*ptr); *ptr = NULL; } static inline GList * _g_list_insert_before_link (GList *list, GList *sibling, GList *link_) { g_return_val_if_fail (link_ != NULL, list); if (!list) { g_return_val_if_fail (sibling == NULL, list); return link_; } else if (sibling) { link_->prev = sibling->prev; link_->next = sibling; sibling->prev = link_; if (link_->prev) { link_->prev->next = link_; return list; } else { g_return_val_if_fail (sibling == list, link_); return link_; } } else { GList *last; last = list; while (last->next) last = last->next; last->next = link_; last->next->prev = last; last->next->next = NULL; return list; } } static inline void _g_queue_insert_before_link (GQueue *queue, GList *sibling, GList *link_) { g_return_if_fail (queue != NULL); g_return_if_fail (link_ != NULL); if G_UNLIKELY (sibling == NULL) { /* We don't use g_list_insert_before_link() with a NULL sibling because it * would be a O(n) operation and we would need to update manually the tail * pointer. */ g_queue_push_tail_link (queue, link_); } else { queue->head = _g_list_insert_before_link (queue->head, sibling, link_); queue->length++; } } static inline void _g_queue_insert_after_link (GQueue *queue, GList *sibling, GList *link_) { g_return_if_fail (queue != NULL); g_return_if_fail (link_ != NULL); if (sibling == NULL) g_queue_push_head_link (queue, link_); else _g_queue_insert_before_link (queue, sibling->next, link_); } static inline GPtrArray * _g_ptr_array_copy_objects (GPtrArray *ar) { if (ar != NULL) { GPtrArray *copy = g_ptr_array_new_full (ar->len, g_object_unref); for (guint i = 0; i < ar->len; i++) g_ptr_array_add (copy, g_object_ref (g_ptr_array_index (ar, i))); return g_steal_pointer (©); } return NULL; } static void ide_object_unref_and_destroy (IdeObject *object) { if (object != NULL) { if (!ide_object_in_destruction (object)) ide_object_destroy (object); g_object_unref (object); } } typedef GPtrArray IdeObjectArray; static inline void ide_clear_and_destroy_object (gpointer pptr) { IdeObject **ptr = pptr; if (ptr && *ptr) { if (!ide_object_in_destruction (*ptr)) ide_object_destroy (*ptr); g_clear_object (ptr); } } static inline GPtrArray * ide_object_array_new (void) { return g_ptr_array_new_with_free_func ((GDestroyNotify)ide_object_unref_and_destroy); } static inline gpointer ide_object_array_steal_index (IdeObjectArray *array, guint position) { gpointer ret = g_ptr_array_index (array, position); g_ptr_array_index (array, position) = NULL; g_ptr_array_remove_index (array, position); return ret; } static inline gpointer ide_object_array_index (IdeObjectArray *array, guint position) { return g_ptr_array_index (array, position); } static inline void ide_object_array_add (IdeObjectArray *ar, gpointer instance) { g_ptr_array_add (ar, g_object_ref (IDE_OBJECT (instance))); } static inline void ide_object_array_unref (IdeObjectArray *ar) { g_ptr_array_unref (ar); } #define IDE_OBJECT_ARRAY_STEAL_FULL(ar) IDE_PTR_ARRAY_STEAL_FULL(ar) G_DEFINE_AUTOPTR_CLEANUP_FUNC (IdeObjectArray, g_ptr_array_unref) G_END_DECLS #endif /* __GI_SCANNER__ */