1
0
mirror of https://github.com/exaloop/codon.git synced 2025-06-03 15:03:52 +08:00
codon/stdlib/internal/format.codon

218 lines
5.5 KiB
Python
Raw Normal View History

# Copyright (C) 2022-2024 Exaloop Inc. <https://exaloop.io>
def _format_error(ret: str):
raise ValueError(f"invalid format specifier: {ret}")
def python_to_fmt_format(s):
i = 0
fill, align = None, None
if i + 1 < len(s) and (s[i + 1] == '<' or s[i + 1] == '>' or s[i + 1] == '=' or s[i + 1] == '^'):
fill = s[i]
align = s[i + 1]
i += 2
elif i < len(s) and (s[i] == '<' or s[i] == '>' or s[i] == '=' or s[i] == '^'):
align = s[i]
i += 1
if align and align == '=':
raise NotImplementedError("'=' alignment not yet supported")
sign = None
if i < len(s) and (s[i] == '+' or s[i] == '-' or s[i] == ' '):
sign = s[i]
i += 1
coerce_negative_float = False
if i < len(s) and s[i] == 'z':
coerce_negative_float = True
i += 1
if coerce_negative_float:
raise NotImplementedError("'z' not yet supported")
alternate_form = False
if i < len(s) and s[i] == '#':
alternate_form = True
i += 1
width_pre_zero = False
if i < len(s) and s[i] == '#':
width_pre_zero = True
i += 1
width = 0
while i < len(s) and str._isdigit(s.ptr[i]):
width = 10 * width + ord(s[i]) - ord('0')
i += 1
grouping = None
if i < len(s) and (s[i] == '_' or s[i] == ','):
grouping = s[i]
i += 1
if grouping == '_':
raise NotImplementedError("'_' grouping not yet supported")
precision = None
if i < len(s) and s[i] == '.':
i += 1
precision = 0
while i < len(s) and str._isdigit(s.ptr[i]):
precision = 10 * precision + ord(s[i]) - ord('0')
i += 1
type = None
if i < len(s):
type = s[i]
i += 1
if i != len(s):
raise ValueError("bad format string")
# Construct fmt::format-compatible string
ns = ""
if align:
if fill: ns += fill
ns += align
if sign: ns += sign
if alternate_form: ns += "#"
if width_pre_zero: ns += "0"
if width: ns += str(width)
if precision is not None: ns += f".{precision}"
if grouping: ns += "L"
if type: ns += type
return ns
@extend
class int:
def __format__(self, format_spec: str) -> str:
err = False
ret = _C.seq_str_int(self, python_to_fmt_format(format_spec), __ptr__(err))
if format_spec and err:
_format_error(ret)
return ret
@extend
class Int:
def __format__(self, format_spec: str) -> str:
err = False
ret = _C.seq_str_int(self.__int__(), python_to_fmt_format(format_spec), __ptr__(err))
if format_spec and err:
_format_error(ret)
return ret
@extend
class UInt:
def __format__(self, format_spec: str) -> str:
err = False
ret = _C.seq_str_uint(self.__int__(), python_to_fmt_format(format_spec), __ptr__(err))
if format_spec and err:
_format_error(ret)
return ret
@extend
class float:
def __format__(self, format_spec: str) -> str:
err = False
ret = _C.seq_str_float(self, python_to_fmt_format(format_spec), __ptr__(err))
if format_spec and err:
_format_error(ret)
return ret if ret != "-nan" else "nan"
@extend
class str:
def __format__(self, format_spec: str) -> str:
err = False
ret = _C.seq_str_str(self, python_to_fmt_format(format_spec), __ptr__(err))
if format_spec and err:
_format_error(ret)
return ret
@extend
class Ptr:
def __format__(self, format_spec: str) -> str:
err = False
ret = _C.seq_str_ptr(self.as_byte(), python_to_fmt_format(format_spec), __ptr__(err))
if format_spec and err:
_format_error(ret)
return ret
v0.16 (#335) * Add Python extension lowering pass * Add DocstringAttribute * Add extension module codegen * Handle different argument counts efficiently * Add warnings to extension lowering * Fix module name * Fix extension codegen * Fix argument check * Auto-convert Codon exceptions to Python exceptions * Fix #183 * Fix #162; Fix #135 * Fix #155 * Fix CPython interface in codegen * Fix #191 * Fix #187 * Fix #189 * Generate object file in pyext mode * Convert Codon exceptions to Python exceptions * Fix vtable init; Fix failing tests on Linux * Fix #190 * Fix #156 * Fix union routing * Remove need for import python * Automatic @export and wrapping for toplevel functions * Reorganize API * Add Python extension IR structs * Add special calls for no-suspend yield-expr * Add special calls for no-suspend yield-expr * pyextension.h support [wip] * pyextension.h support [wip] * pyextension.h support * pyextension.h support for toplevel functions * clang-format * Add PyFunction::nargs field * Update pyextension codegen (WIP) * SUpport nargs * Add support for @pycapture * PyType codegen (WIP) * Py method codegen (WIP) * Add type ptr hook * Add getset codegen * Add type alloc function * Add type pointer hook codegen * Re-organize codegen * Add member codegen * Update module init codegen * Update module init codegen * Add support for typePtrHook and new to/from_py hooks * Fix extension codegen * Fix init codegen * Fix init codegen; add "tp_new" slot * Fix type hook * Add extra flags * Specialized wrappers (PyType specs) * Add static Python link option * Fix C imports * Add guards * Remove unused field * Python mode only when pyExt set * Update python module * Fix assert * Update codegen/passes * Fix tuple parsing in index expression * Fix empty tuple unification * Do not Cythonize underscore fns * clang-format * Fix switch * Add Py support for cmp/setitem * Add Py support for cmp/setitem * Add type is support * GetSet support * clang-format * GetSet support (fixes) * Avoid useless vtable alloc * Add iter support * Fix size_t capture bug * clang-format * Fix POD type unification with tuples * Add __try_from_py__ API * Fix annotation * Add static reflection methods (setattr; internal.static.*); refactor PyExt to python.codon; handle errors and kwargs in PyExt * Python compat fixes * Update Python object conversions * Fix PyErrors * clang-format; add copyright * Add PyFunction::keywords field * Fix JIT MRO handling; Refactor out Jupyter support * Refactor out Jupyter support * Add support for custom linking args (link=[]) to TOML plugins * Fix tests * Use g++ instead of gcc * Fix Jupyter CMAKE * Fix Jupyter CMAKE * Add _PyArg_Parser definition * Add complex64 type * Add extra complex64 tests * Fix Python calls; add staticenumerate * Fix call * Fix calls * Update pyext wrappers * Fix staticenumerate; Support static calls in tuple() * Fix pyext routing * Add add/mul for tuples * clang-format * Fix pyext codegen * Fix wrap_multiple * Add seq_alloc_atomic_uncollectable * Fix default generics issue * Add binary/ternary ops * Fix missing generic issue * Fix number slots * Update pow * Remove unnecessary pyobj * Fix allocation * Refactor errors * Add test extension * Fix formatting * clang-format * Fix getitem/setitem/delitem in pyext * Fix pyext iterators * Add builtin pow() (fix #294) * Fix #244 * Fix #231 * Fix #229 * Fix #205 * Update docs * Fix error message * Add pyext tests * Add pyext support for @property * Add pyext support for toplevel fns and @tuple classes * More pyext tests * More pyext tests * Fix file error checking * More pyext tests * Update pyext tests * Update docs * Add pyext test to CI * Add pyext support for @tuple.__new__ * Add pyext support for @tuple.__new__ * Fix hetero-tuple issue with fn_overloads * More pyext tests * Bump versions * Fix del magic in pyext * Fix init magic for tuples in pyext * Have test-pypi only run on develop branch * Make exception type indices unnamed-addr * Fix #316; Fix #317 (slash issue) * Use uncollectible-alloc for vtable * Fix #249 * Add pyext docs * Fix #249; Fix clashing vtables; Fix super() and class_copy * Add content-atomic type property instruction * __contents_atomic__ support * Update internal functions * Use PIC when generating Python extension * Cleanup * Add Dockerfile & fix -fPIC * Cleanup * Fix setup.py * Fix pyext fn iteration * Fix CI * clang-format * Update long conversions in Py bridge * Support wide-int to str conversions * Fix test * Add pow for arbitrary-width ints * Fix Linux backtraces * Cleanup * Add more tests * Fix docs; Remove tuple.__add__ for scalars * Update docs --------- Co-authored-by: Ibrahim Numanagić <ibrahimpasa@gmail.com>
2023-04-12 18:13:54 -04:00
def _divmod_10(dividend, N: Static[int]):
T = type(dividend)
zero, one = T(0), T(1)
neg = dividend < zero
dvd = dividend.__abs__()
remainder = 0
quotient = zero
# Euclidean division
for bit_idx in range(N - 1, -1, -1):
mask = int((dvd & (one << T(bit_idx))) != zero)
remainder = (remainder << 1) + mask
if remainder >= 10:
quotient = (quotient << one) + one
remainder -= 10
else:
quotient = quotient << one
if neg:
quotient = -quotient
remainder = -remainder
return quotient, remainder
@extend
class Int:
def __str__(self) -> str:
if N <= 64:
return str(int(self))
if not self:
return '0'
s = _strbuf()
d = self
if d >= Int[N](0):
while True:
d, m = _divmod_10(d, N)
b = byte(48 + m) # 48 == ord('0')
s.append(str(__ptr__(b), 1))
if not d:
break
else:
while True:
d, m = _divmod_10(d, N)
b = byte(48 - m) # 48 == ord('0')
s.append(str(__ptr__(b), 1))
if not d:
break
s.append('-')
s.reverse()
return s.__str__()
@extend
class UInt:
def __str__(self) -> str:
if N <= 64:
return self.__format__("")
s = _strbuf()
d = self
while True:
d, m = _divmod_10(d, N)
b = byte(48 + int(m)) # 48 == ord('0')
s.append(str(__ptr__(b), 1))
if not d:
break
s.reverse()
return s.__str__()
@extend
class __magic__:
def repr_partial(slf) -> str:
h = slf.__class__.__name__
return f"partial({h})"