| /** |
| * This is a public domain version of qsort.d. All it does is call C's |
| * qsort(). |
| * |
| * Copyright: Copyright Digital Mars 2000 - 2010. |
| * License: $(WEB www.boost.org/LICENSE_1_0.txt, Boost License 1.0). |
| * Authors: Walter Bright, Martin Nowak |
| */ |
| |
| /* Copyright Digital Mars 2000 - 2010. |
| * Distributed under the Boost Software License, Version 1.0. |
| * (See accompanying file LICENSE_1_0.txt or copy at |
| * http://www.boost.org/LICENSE_1_0.txt) |
| */ |
| module rt.qsort; |
| |
| //debug=qsort; |
| |
| private import core.stdc.stdlib; |
| |
| version (OSX) |
| version = Darwin; |
| else version (iOS) |
| version = Darwin; |
| else version (TVOS) |
| version = Darwin; |
| else version (WatchOS) |
| version = Darwin; |
| |
| // qsort_r was added in glibc in 2.8. https://gcc.gnu.org/bugzilla/show_bug.cgi?id=88127 |
| version (CRuntime_Glibc) |
| { |
| version (GNU) |
| { |
| import gcc.config : Have_Qsort_R; |
| enum Glibc_Qsort_R = Have_Qsort_R; |
| } |
| else |
| { |
| enum Glibc_Qsort_R = true; |
| } |
| } |
| else |
| { |
| enum Glibc_Qsort_R = false; |
| } |
| |
| static if (Glibc_Qsort_R) |
| { |
| alias extern (C) int function(scope const void *, scope const void *, scope void *) Cmp; |
| extern (C) void qsort_r(scope void *base, size_t nmemb, size_t size, Cmp cmp, scope void *arg); |
| |
| extern (C) void[] _adSort(return scope void[] a, TypeInfo ti) |
| { |
| extern (C) int cmp(scope const void* p1, scope const void* p2, scope void* ti) |
| { |
| return (cast(TypeInfo)ti).compare(p1, p2); |
| } |
| qsort_r(a.ptr, a.length, ti.tsize, &cmp, cast(void*)ti); |
| return a; |
| } |
| } |
| else version (FreeBSD) |
| { |
| alias extern (C) int function(scope void *, scope const void *, scope const void *) Cmp; |
| extern (C) void qsort_r(scope void *base, size_t nmemb, size_t size, scope void *thunk, Cmp cmp); |
| |
| extern (C) void[] _adSort(return scope void[] a, TypeInfo ti) |
| { |
| extern (C) int cmp(scope void* ti, scope const void* p1, scope const void* p2) |
| { |
| return (cast(TypeInfo)ti).compare(p1, p2); |
| } |
| qsort_r(a.ptr, a.length, ti.tsize, cast(void*)ti, &cmp); |
| return a; |
| } |
| } |
| else version (DragonFlyBSD) |
| { |
| alias extern (C) int function(scope void *, scope const void *, scope const void *) Cmp; |
| extern (C) void qsort_r(scope void *base, size_t nmemb, size_t size, scope void *thunk, Cmp cmp); |
| |
| extern (C) void[] _adSort(return scope void[] a, TypeInfo ti) |
| { |
| extern (C) int cmp(scope void* ti, scope const void* p1, scope const void* p2) |
| { |
| return (cast(TypeInfo)ti).compare(p1, p2); |
| } |
| qsort_r(a.ptr, a.length, ti.tsize, cast(void*)ti, &cmp); |
| return a; |
| } |
| } |
| else version (Darwin) |
| { |
| alias extern (C) int function(scope void *, scope const void *, scope const void *) Cmp; |
| extern (C) void qsort_r(scope void *base, size_t nmemb, size_t size, scope void *thunk, Cmp cmp); |
| |
| extern (C) void[] _adSort(return scope void[] a, TypeInfo ti) |
| { |
| extern (C) int cmp(scope void* ti, scope const void* p1, scope const void* p2) |
| { |
| return (cast(TypeInfo)ti).compare(p1, p2); |
| } |
| qsort_r(a.ptr, a.length, ti.tsize, cast(void*)ti, &cmp); |
| return a; |
| } |
| } |
| else version (CRuntime_UClibc) |
| { |
| alias extern (C) int function(scope const void *, scope const void *, scope void *) __compar_d_fn_t; |
| extern (C) void qsort_r(scope void *base, size_t nmemb, size_t size, __compar_d_fn_t cmp, scope void *arg); |
| |
| extern (C) void[] _adSort(return scope void[] a, TypeInfo ti) |
| { |
| extern (C) int cmp(scope const void* p1, scope const void* p2, scope void* ti) |
| { |
| return (cast(TypeInfo)ti).compare(p1, p2); |
| } |
| qsort_r(a.ptr, a.length, ti.tsize, &cmp, cast(void*)ti); |
| return a; |
| } |
| } |
| else |
| { |
| private TypeInfo tiglobal; |
| |
| extern (C) void[] _adSort(return scope void[] a, TypeInfo ti) |
| { |
| extern (C) int cmp(scope const void* p1, scope const void* p2) |
| { |
| return tiglobal.compare(p1, p2); |
| } |
| tiglobal = ti; |
| qsort(a.ptr, a.length, ti.tsize, &cmp); |
| return a; |
| } |
| } |
| |
| |
| |
| unittest |
| { |
| debug(qsort) printf("array.sort.unittest()\n"); |
| |
| int[] a = new int[10]; |
| |
| a[0] = 23; |
| a[1] = 1; |
| a[2] = 64; |
| a[3] = 5; |
| a[4] = 6; |
| a[5] = 5; |
| a[6] = 17; |
| a[7] = 3; |
| a[8] = 0; |
| a[9] = -1; |
| |
| _adSort(*cast(void[]*)&a, typeid(a[0])); |
| |
| for (int i = 0; i < a.length - 1; i++) |
| { |
| //printf("i = %d", i); |
| //printf(" %d %d\n", a[i], a[i + 1]); |
| assert(a[i] <= a[i + 1]); |
| } |
| } |