|
1 | 1 | from ._graphblas import ffi, lib # noqa
|
2 | 2 | from . import utils
|
3 | 3 | from ._version import get_versions
|
| 4 | +from . import exceptions as ex |
4 | 5 |
|
5 | 6 |
|
6 | 7 | def is_initialized():
|
7 | 8 | """Is GraphBLAS initialized via GrB_init or GxB_init?"""
|
8 | 9 | return lib.GxB_Global_Option_get(lib.GxB_MODE, ffi.new("GrB_Mode*")) != lib.GrB_PANIC
|
9 | 10 |
|
10 | 11 |
|
| 12 | +def supports_complex(): |
| 13 | + """Does this package support complex numbers?""" |
| 14 | + return hasattr(lib, "GrB_FC64") or hasattr(lib, "GxB_FC64") |
| 15 | + |
| 16 | + |
11 | 17 | def initialize(*, blocking=False, memory_manager="numpy"):
|
12 | 18 | """Initialize GraphBLAS via GrB_init or GxB_init.
|
13 | 19 |
|
@@ -37,8 +43,132 @@ def initialize(*, blocking=False, memory_manager="numpy"):
|
37 | 43 | elif memory_manager == "c":
|
38 | 44 | lib.GrB_init(blocking)
|
39 | 45 | else:
|
40 |
| - raise ValueError(f'memory_manager argument must be "numpy" or "c"; got: {memory_manager!r}') |
| 46 | + raise ValueError( |
| 47 | + f'memory_manager argument must be "numpy" or "c"; got: ' "{memory_manager!r}" # noqa |
| 48 | + ) |
41 | 49 |
|
42 | 50 |
|
43 | 51 | __version__ = get_versions()["version"]
|
44 | 52 | del get_versions
|
| 53 | + |
| 54 | + |
| 55 | +def libget(name): |
| 56 | + """Helper to get items from GraphBLAS which might be GrB or GxB""" |
| 57 | + try: |
| 58 | + return getattr(lib, name) |
| 59 | + except AttributeError: |
| 60 | + ext_name = f"GxB_{name[4:]}" |
| 61 | + try: |
| 62 | + return getattr(lib, ext_name) |
| 63 | + except AttributeError: |
| 64 | + pass |
| 65 | + raise |
| 66 | + |
| 67 | + |
| 68 | +bool_types = frozenset((lib.GrB_BOOL,)) |
| 69 | + |
| 70 | +signed_integer_types = frozenset( |
| 71 | + ( |
| 72 | + lib.GrB_INT8, |
| 73 | + lib.GrB_INT16, |
| 74 | + lib.GrB_INT32, |
| 75 | + lib.GrB_INT64, |
| 76 | + ) |
| 77 | +) |
| 78 | + |
| 79 | +unsigned_integer_types = frozenset( |
| 80 | + ( |
| 81 | + lib.GrB_UINT8, |
| 82 | + lib.GrB_UINT16, |
| 83 | + lib.GrB_UINT32, |
| 84 | + lib.GrB_UINT64, |
| 85 | + ) |
| 86 | +) |
| 87 | + |
| 88 | +integer_types = signed_integer_types | unsigned_integer_types |
| 89 | + |
| 90 | +real_types = frozenset( |
| 91 | + ( |
| 92 | + lib.GrB_FP32, |
| 93 | + lib.GrB_FP64, |
| 94 | + ) |
| 95 | +) |
| 96 | + |
| 97 | +if supports_complex(): |
| 98 | + complex_types = frozenset( |
| 99 | + ( |
| 100 | + lib.GxB_FC32, |
| 101 | + lib.GxB_FC64, |
| 102 | + ) |
| 103 | + ) |
| 104 | +else: |
| 105 | + complex_types = frozenset() |
| 106 | + |
| 107 | + |
| 108 | +grb_types = bool_types | integer_types | real_types | complex_types |
| 109 | + |
| 110 | + |
| 111 | +_error_code_lookup = { |
| 112 | + # Warning |
| 113 | + lib.GrB_NO_VALUE: ex.NoValue, |
| 114 | + # API Errors |
| 115 | + lib.GrB_UNINITIALIZED_OBJECT: ex.UninitializedObject, |
| 116 | + lib.GrB_INVALID_OBJECT: ex.InvalidObject, |
| 117 | + lib.GrB_NULL_POINTER: ex.NullPointer, |
| 118 | + lib.GrB_INVALID_VALUE: ex.InvalidValue, |
| 119 | + lib.GrB_INVALID_INDEX: ex.InvalidIndex, |
| 120 | + lib.GrB_DOMAIN_MISMATCH: ex.DomainMismatch, |
| 121 | + lib.GrB_DIMENSION_MISMATCH: ex.DimensionMismatch, |
| 122 | + lib.GrB_OUTPUT_NOT_EMPTY: ex.OutputNotEmpty, |
| 123 | + # Execution Errors |
| 124 | + lib.GrB_OUT_OF_MEMORY: ex.OutOfMemory, |
| 125 | + lib.GrB_INSUFFICIENT_SPACE: ex.InsufficientSpace, |
| 126 | + lib.GrB_INDEX_OUT_OF_BOUNDS: ex.IndexOutOfBound, |
| 127 | + lib.GrB_PANIC: ex.Panic, |
| 128 | +} |
| 129 | +GrB_SUCCESS = lib.GrB_SUCCESS |
| 130 | +GrB_NO_VALUE = lib.GrB_NO_VALUE |
| 131 | + |
| 132 | + |
| 133 | +_error_func_lookup = { |
| 134 | + "struct GB_Type_opaque *": lib.GrB_Type_error, |
| 135 | + "struct GB_UnaryOp_opaque *": lib.GrB_UnaryOp_error, |
| 136 | + "struct GB_BinaryOp_opaque *": lib.GrB_BinaryOp_error, |
| 137 | + "struct GB_SelectOp_opaque *": lib.GxB_SelectOp_error, |
| 138 | + "struct GB_Monoid_opaque *": lib.GrB_Monoid_error, |
| 139 | + "struct GB_Semiring_opaque *": lib.GrB_Semiring_error, |
| 140 | + "struct GB_Scalar_opaque *": lib.GxB_Scalar_error, |
| 141 | + "struct GB_Matrix_opaque *": lib.GrB_Matrix_error, |
| 142 | + "struct GB_Vector_opaque *": lib.GrB_Vector_error, |
| 143 | + "struct GB_Descriptor_opaque *": lib.GrB_Descriptor_error, |
| 144 | +} |
| 145 | + |
| 146 | + |
| 147 | +def check_status(obj, response_code): |
| 148 | + """Check the return code of the GraphBLAS function. |
| 149 | +
|
| 150 | + If the operation was successful, return None. |
| 151 | +
|
| 152 | + If the operation returned no value return `exceptions.NoValue`. |
| 153 | +
|
| 154 | + Otherwise it is an error, lookup the exception and the error |
| 155 | + description, and throw the exception. |
| 156 | +
|
| 157 | + """ |
| 158 | + if response_code == GrB_SUCCESS: |
| 159 | + return |
| 160 | + if response_code == GrB_NO_VALUE: |
| 161 | + return ex.NoValue |
| 162 | + |
| 163 | + if ffi.typeof(obj).item.kind == "pointer": |
| 164 | + obj = obj[0] |
| 165 | + |
| 166 | + cname = ffi.typeof(obj).cname |
| 167 | + error_func = _error_func_lookup.get(cname) |
| 168 | + if error_func is None: |
| 169 | + raise TypeError(f"Unknown cname {cname} looking up error string.") |
| 170 | + |
| 171 | + string = ffi.new("char**") |
| 172 | + error_func(string, obj) |
| 173 | + text = ffi.string(string[0]).decode() |
| 174 | + raise _error_code_lookup[response_code](text) |
0 commit comments