mirror of
https://gitlab.gnome.org/GNOME/gtk.git
synced 2025-01-15 23:00:08 +00:00
d4ba57fcc3
This is just the API. Users will come later. I considered putting it into gdkmemoryformat.c because it's likely gonna be the only user and this one function is so little code, but it didn't fit at all. So now it's a new file.
87 lines
2.3 KiB
C
87 lines
2.3 KiB
C
/*
|
|
* Copyright © 2024 Benjamin Otte
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library 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
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
* Authors: Benjamin Otte <otte@gnome.org>
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "gdkparalleltaskprivate.h"
|
|
|
|
typedef struct _TaskData TaskData;
|
|
|
|
struct _TaskData
|
|
{
|
|
GdkTaskFunc task_func;
|
|
gpointer task_data;
|
|
int n_running_tasks;
|
|
};
|
|
|
|
static void
|
|
gdk_parallel_task_thread_func (gpointer data,
|
|
gpointer unused)
|
|
{
|
|
TaskData *task = data;
|
|
|
|
task->task_func (task->task_data);
|
|
|
|
g_atomic_int_add (&task->n_running_tasks, -1);
|
|
}
|
|
|
|
/**
|
|
* gdk_parallel_task_run:
|
|
* @task_func: the function to spawn
|
|
* @task_data: data to pass to the function
|
|
*
|
|
* Spawns the given function in many threads.
|
|
* Once all functions have exited, this function returns.
|
|
**/
|
|
void
|
|
gdk_parallel_task_run (GdkTaskFunc task_func,
|
|
gpointer task_data)
|
|
{
|
|
static GThreadPool *pool;
|
|
TaskData task = {
|
|
.task_func = task_func,
|
|
.task_data = task_data,
|
|
};
|
|
int i, n_tasks;
|
|
|
|
if (g_once_init_enter (&pool))
|
|
{
|
|
GThreadPool *the_pool = g_thread_pool_new (gdk_parallel_task_thread_func,
|
|
NULL,
|
|
MAX (2, g_get_num_processors ()) - 1,
|
|
FALSE,
|
|
NULL);
|
|
g_once_init_leave (&pool, the_pool);
|
|
}
|
|
|
|
n_tasks = g_get_num_processors ();
|
|
task.n_running_tasks = n_tasks;
|
|
/* Start with 1 because we run 1 task ourselves */
|
|
for (i = 1; i < n_tasks; i++)
|
|
{
|
|
g_thread_pool_push (pool, &task, NULL);
|
|
}
|
|
|
|
gdk_parallel_task_thread_func (&task, NULL);
|
|
|
|
while (g_atomic_int_get (&task.n_running_tasks) > 0)
|
|
g_thread_yield ();
|
|
}
|
|
|