mirror of
https://github.com/ggerganov/llama.cpp.git
synced 2025-01-11 11:11:46 +00:00
434 lines
12 KiB
C
434 lines
12 KiB
C
// https://github.com/ggerganov/ggml/issues/291
|
|
// https://github.com/ggerganov/llama.cpp/pull/1507
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <time.h>
|
|
|
|
#if defined(_MSC_VER) || defined(__MINGW32__)
|
|
#include <malloc.h> // using malloc.h with MSC/MINGW
|
|
#elif !defined(__FreeBSD__) && !defined(__NetBSD__) && !defined(__OpenBSD__)
|
|
#include <alloca.h>
|
|
#endif
|
|
|
|
#define GGML_ASSERT(x) \
|
|
do { \
|
|
if (!(x)) { \
|
|
fprintf(stderr, "GGML_ASSERT: %s:%d: %s\n", __FILE__, __LINE__, \
|
|
#x); \
|
|
abort(); \
|
|
} \
|
|
} while (0)
|
|
|
|
#define GGML_DEBUG 1
|
|
#if (GGML_DEBUG >= 1)
|
|
#define GGML_PRINT_DEBUG(...) printf(__VA_ARGS__)
|
|
#else
|
|
#define GGML_PRINT_DEBUG(...)
|
|
#endif
|
|
|
|
#if (GGML_DEBUG >= 5)
|
|
#define GGML_PRINT_DEBUG_5(...) printf(__VA_ARGS__)
|
|
#else
|
|
#define GGML_PRINT_DEBUG_5(...)
|
|
#endif
|
|
|
|
#define UNUSED(x) (void)(x)
|
|
#define MIN(a, b) ((a) < (b) ? (a) : (b))
|
|
|
|
#if defined(_WIN32)
|
|
|
|
#include <windows.h>
|
|
|
|
typedef volatile LONG atomic_int;
|
|
typedef atomic_int atomic_bool;
|
|
|
|
typedef HANDLE pthread_t;
|
|
typedef int thread_ret_t;
|
|
|
|
static void atomic_store(atomic_int *ptr, LONG val) {
|
|
Intechan_lockedExchange(ptr, val);
|
|
}
|
|
|
|
static LONG atomic_load(atomic_int *ptr) {
|
|
return Intechan_lockedCompareExchange(ptr, 0, 0);
|
|
}
|
|
|
|
static LONG atomic_fetch_add(atomic_int *ptr, LONG inc) {
|
|
return Intechan_lockedExchangeAdd(ptr, inc);
|
|
}
|
|
|
|
static LONG atomic_fetch_sub(atomic_int *ptr, LONG dec) {
|
|
return atomic_fetch_add(ptr, -(dec));
|
|
}
|
|
|
|
static int pthread_create(pthread_t *out, void *unused,
|
|
thread_ret_t (*func)(void *), void *arg) {
|
|
(void)unused;
|
|
HANDLE handle =
|
|
CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE)func, arg, 0, NULL);
|
|
if (handle == NULL) {
|
|
return EAGAIN;
|
|
}
|
|
|
|
*out = handle;
|
|
return 0;
|
|
}
|
|
|
|
static int pthread_join(pthread_t thread, void *unused) {
|
|
(void)unused;
|
|
return (int)WaitForSingleObject(thread, INFINITE);
|
|
}
|
|
|
|
static int sched_yield(void) {
|
|
// https://learn.microsoft.com/en-us/windows/win32/api/winnt/nf-winnt-yieldprocessor
|
|
YieldProcessor();
|
|
return 0;
|
|
}
|
|
|
|
#else // ! _WIN32
|
|
|
|
typedef void *thread_ret_t;
|
|
|
|
#include <pthread.h>
|
|
#include <stdatomic.h>
|
|
#include <stdbool.h>
|
|
|
|
#endif
|
|
|
|
typedef pthread_t ggml_thread_t;
|
|
|
|
//-----------------------------------------------------------------------------
|
|
/// Most of the above codes are taken from
|
|
/// https://github.com/ggerganov/llama.cpp/tree/master/ggml.c
|
|
/// Copyright original authors.
|
|
//-----------------------------------------------------------------------------
|
|
|
|
_Thread_local int32_t thread_local_id;
|
|
|
|
#define MAX_THREADS 16
|
|
|
|
struct task_allocator {
|
|
int nth;
|
|
|
|
int n_multiplier; // >= 1
|
|
|
|
atomic_int lock; // 0 unlocked, 1 locked
|
|
|
|
// total assigned.
|
|
atomic_int global_counter;
|
|
|
|
atomic_int thread_queue_heads[MAX_THREADS];
|
|
atomic_int thread_queue_tails[MAX_THREADS];
|
|
};
|
|
|
|
static void task_allocator_reset(struct task_allocator *a) {
|
|
for (int i = 0; i < a->nth; ++i) {
|
|
atomic_store(&a->thread_queue_heads[i], 0);
|
|
atomic_store(&a->thread_queue_tails[i], a->n_multiplier);
|
|
}
|
|
|
|
atomic_store(&a->lock, 0);
|
|
atomic_store(&a->global_counter, 0);
|
|
}
|
|
|
|
static void task_allocator_init(struct task_allocator *a, int nth,
|
|
int n_multiplier) {
|
|
GGML_ASSERT(nth <= MAX_THREADS);
|
|
a->nth = nth;
|
|
a->n_multiplier = n_multiplier;
|
|
task_allocator_reset(a);
|
|
}
|
|
|
|
static void allocate_chunk(struct task_allocator *a, int ith, int *chunk_idx,
|
|
int *n_chunks) {
|
|
GGML_ASSERT(a->nth > 0);
|
|
GGML_ASSERT(a->n_multiplier > 0);
|
|
|
|
*chunk_idx = -1;
|
|
*n_chunks = 0;
|
|
|
|
while (atomic_fetch_add(&a->lock, 1) != 0) { // lock
|
|
atomic_fetch_sub(&a->lock, 1);
|
|
}
|
|
|
|
int M = a->n_multiplier;
|
|
int nth = a->nth;
|
|
int total_chunks = M * nth;
|
|
|
|
// all assigned?
|
|
if (atomic_load(&a->global_counter) == total_chunks) {
|
|
GGML_PRINT_DEBUG_5("[#_%d] %s(): nothing to do.\n", thread_local_id,
|
|
__func__);
|
|
atomic_fetch_sub(&a->lock, 1); // unlock
|
|
return;
|
|
}
|
|
|
|
// try take its own, pop front.
|
|
{
|
|
int head = atomic_load(&a->thread_queue_heads[ith]);
|
|
int tail = atomic_load(&a->thread_queue_tails[ith]);
|
|
|
|
GGML_PRINT_DEBUG_5("[#_%d] %s(): head: %d, tail: %d.\n",
|
|
thread_local_id, __func__, head, tail);
|
|
|
|
if (head < tail) {
|
|
int idx = ith * M + head;
|
|
|
|
atomic_fetch_add(&a->thread_queue_heads[ith], 1);
|
|
atomic_fetch_add(&a->global_counter, 1);
|
|
|
|
GGML_PRINT_DEBUG("[#_%d] %s(): take the %3d-th trunk of its own.\n",
|
|
thread_local_id, __func__, head + 1);
|
|
|
|
*chunk_idx = idx;
|
|
*n_chunks = total_chunks;
|
|
atomic_fetch_sub(&a->lock, 1); // unlock
|
|
return;
|
|
}
|
|
}
|
|
|
|
// steal from others.
|
|
for (int i = 0; i < nth; ++i) {
|
|
if (i == ith) {
|
|
continue;
|
|
}
|
|
|
|
int tail = atomic_load(&a->thread_queue_tails[i]);
|
|
if (tail == atomic_load(&a->thread_queue_heads[i])) {
|
|
continue;
|
|
}
|
|
|
|
// pop back
|
|
int idx = i * M + tail;
|
|
atomic_fetch_sub(&a->thread_queue_tails[i], 1);
|
|
atomic_fetch_add(&a->global_counter, 1);
|
|
|
|
GGML_PRINT_DEBUG("[#_%d] %s(): steal the %d-th trunk from #_%d\n",
|
|
thread_local_id, __func__, tail, i);
|
|
|
|
*chunk_idx = idx;
|
|
*n_chunks = total_chunks;
|
|
atomic_fetch_sub(&a->lock, 1); // unlock
|
|
return;
|
|
}
|
|
|
|
fprintf(stderr, "%s:%d should be unreachable!\n", __FILE__, __LINE__);
|
|
abort();
|
|
}
|
|
|
|
struct state_shared {
|
|
int n_threads;
|
|
int n_multiplier;
|
|
|
|
int n_nodes;
|
|
struct ggml_tensor *nodes;
|
|
|
|
// thread done counter for single node
|
|
atomic_int done_counter;
|
|
|
|
struct task_allocator task_allocator;
|
|
};
|
|
|
|
struct state {
|
|
ggml_thread_t thrd;
|
|
int ith;
|
|
struct state_shared *shared;
|
|
};
|
|
|
|
// simulate tensor that can be compute in parallel
|
|
struct ggml_tensor {
|
|
// simulate actual compute workload, e.g. src0 rows
|
|
int n_compute_units;
|
|
};
|
|
|
|
struct params {
|
|
int ith;
|
|
int nth;
|
|
|
|
// simulate performance jitters related to: OS workload, thread affinity,
|
|
// economic cores, ...
|
|
int jitter_percent;
|
|
|
|
struct task_allocator *task_allocator;
|
|
};
|
|
|
|
void compute_tensor(struct params params, struct ggml_tensor *node) {
|
|
GGML_PRINT_DEBUG_5("[#_%d] %s(): enter.\n", thread_local_id, __func__);
|
|
|
|
const int ith = params.ith;
|
|
int chunk_idx;
|
|
int n_chunks;
|
|
|
|
while (true) {
|
|
allocate_chunk(params.task_allocator, ith, &chunk_idx, &n_chunks);
|
|
if (chunk_idx < 0 || n_chunks <= 0) {
|
|
break;
|
|
}
|
|
|
|
const int nr = node->n_compute_units;
|
|
const int dr = (nr + n_chunks - 1) / n_chunks;
|
|
const int ir0 = dr * chunk_idx;
|
|
const int ir1 = MIN(ir0 + dr, nr);
|
|
const int n_loops = 10000 * (100 + params.jitter_percent);
|
|
|
|
volatile int64_t x = 0;
|
|
|
|
for (int i = ir0; i <= ir1; ++i) {
|
|
for (int j = 0; j < n_loops; ++j) {
|
|
++x;
|
|
}
|
|
}
|
|
UNUSED(x);
|
|
}
|
|
|
|
GGML_PRINT_DEBUG_5("[#_%d] %s(): exit.\n", thread_local_id, __func__);
|
|
}
|
|
|
|
static thread_ret_t demo_compute_thread(void *data) {
|
|
struct state *state = (struct state *)data;
|
|
GGML_ASSERT(state);
|
|
|
|
struct state_shared *shared = state->shared;
|
|
GGML_ASSERT(shared);
|
|
|
|
struct task_allocator *allocator = &shared->task_allocator;
|
|
GGML_ASSERT(allocator);
|
|
|
|
int ith = state->ith;
|
|
int n_threads = shared->n_threads;
|
|
|
|
thread_local_id = ith;
|
|
|
|
atomic_int *done_counter = &shared->done_counter;
|
|
|
|
for (int i = 0; i < shared->n_nodes; ++i) {
|
|
// Just slow down the last thread.
|
|
struct params params = {
|
|
.ith = state->ith,
|
|
.nth = n_threads, // suppose parallel
|
|
.task_allocator = allocator,
|
|
.jitter_percent = ith + 1 < n_threads ? 0 : 50,
|
|
};
|
|
|
|
struct ggml_tensor *node = &shared->nodes[i];
|
|
|
|
compute_tensor(params, node);
|
|
atomic_fetch_add(done_counter, 1);
|
|
|
|
while (atomic_load(done_counter) != n_threads) {
|
|
sched_yield();
|
|
// main: go here --> later, main saw cond matched, break out loop
|
|
// --> reset counter
|
|
// current: go here --stall for a thousand years --> check condition
|
|
// OOPS! will never break out.
|
|
|
|
// So we have to check if the counter has been reset.
|
|
if (atomic_load(done_counter) == 0) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
GGML_PRINT_DEBUG_5(
|
|
"[#_%d] %s(): saw all threads finished computing the node.\n",
|
|
thread_local_id, __func__);
|
|
|
|
if (ith == 0) {
|
|
task_allocator_reset(allocator);
|
|
atomic_store(done_counter, 0);
|
|
} else {
|
|
while (atomic_load(done_counter) != 0) {
|
|
sched_yield();
|
|
}
|
|
}
|
|
}
|
|
|
|
GGML_PRINT_DEBUG_5("[#_%d] %s(): exited\n", thread_local_id, __func__);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void test_task_allocator(int n_threads, int n_nodes, int n_compute_units,
|
|
int n_multiplier) {
|
|
fprintf(stderr,
|
|
"\n[#_%d] %s(): n_threads: %d, n_nodes: %d, n_compute_units: %d, "
|
|
"n_multiplier: %d ===>\n\n",
|
|
thread_local_id, __func__, n_threads, n_nodes, n_compute_units,
|
|
n_multiplier);
|
|
|
|
struct ggml_tensor *nodes = alloca(n_nodes * sizeof(struct ggml_tensor));
|
|
|
|
for (int i = 0; i < n_nodes; ++i) {
|
|
nodes[i].n_compute_units = n_compute_units;
|
|
}
|
|
|
|
struct state_shared shared = {
|
|
.n_threads = n_threads,
|
|
.n_nodes = n_nodes,
|
|
.nodes = nodes,
|
|
.done_counter = 0,
|
|
};
|
|
|
|
task_allocator_init(&shared.task_allocator, n_threads, n_multiplier);
|
|
|
|
struct state *workers = alloca(n_threads * sizeof(struct state));
|
|
|
|
for (int i = 0; i < n_threads; ++i) {
|
|
workers[i].ith = i;
|
|
workers[i].shared = &shared;
|
|
if (i > 0) {
|
|
pthread_create(&workers[i].thrd, NULL, demo_compute_thread,
|
|
&workers[i]);
|
|
}
|
|
}
|
|
|
|
demo_compute_thread(&workers[0]);
|
|
|
|
for (int i = 1; i < n_threads; ++i) {
|
|
pthread_join(workers[i].thrd, NULL);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Conclusions:
|
|
//
|
|
// - Given workers A and B, and the accumulated time T_a and T_b:
|
|
// B can steal a chunk from A only if T_a > T_b + T_b_per_chunk.
|
|
// - Saw this situation: A steal B, B steal C.
|
|
// - n_chunks plays a key role, similar to choosing the best n_threads, it's
|
|
// difficult choose the ideal n_chunks value. Performance drops with per-chunk
|
|
// compute time exceeds the scheduling overhead.
|
|
// - Work stealing chunked task allocator can save the response time
|
|
// significantly when the majority threads runs fast but a few suddenly or
|
|
// constantly slow.
|
|
//
|
|
int main(void) {
|
|
if (false) { // the most simple one: only main thread, one node
|
|
int n_threads = 1;
|
|
int n_nodes = 1;
|
|
int n_multiplier = 1; // trunks per thread.
|
|
int n_compute_units = 1;
|
|
|
|
test_task_allocator(n_threads, n_nodes, n_compute_units, n_multiplier);
|
|
}
|
|
|
|
if (false) {
|
|
int n_threads = 2;
|
|
int n_nodes = 2;
|
|
int n_multiplier = 1; // trunks per thread.
|
|
int n_compute_units = 2;
|
|
|
|
test_task_allocator(n_threads, n_nodes, n_compute_units, n_multiplier);
|
|
}
|
|
|
|
if (true) {
|
|
int n_threads = 4;
|
|
int n_nodes = 2;
|
|
int n_multiplier = 8; // trunks per thread.
|
|
int n_compute_units = 32;
|
|
|
|
test_task_allocator(n_threads, n_nodes, n_compute_units, n_multiplier);
|
|
}
|
|
}
|