# This file is a part of Julia. License is MIT: https://julialang.org/license # `ntuple`, for constructing tuples of a given length """ ntuple(f::Function, n::Integer) Create a tuple of length `n`, computing each element as `f(i)`, where `i` is the index of the element. # Examples ```jldoctest julia> ntuple(i -> 2*i, 4) (2, 4, 6, 8) ``` """ function ntuple(f::F, n::Integer) where F t = n == 0 ? () : n == 1 ? (f(1),) : n == 2 ? (f(1), f(2)) : n == 3 ? (f(1), f(2), f(3)) : n == 4 ? (f(1), f(2), f(3), f(4)) : n == 5 ? (f(1), f(2), f(3), f(4), f(5)) : n == 6 ? (f(1), f(2), f(3), f(4), f(5), f(6)) : n == 7 ? (f(1), f(2), f(3), f(4), f(5), f(6), f(7)) : n == 8 ? (f(1), f(2), f(3), f(4), f(5), f(6), f(7), f(8)) : n == 9 ? (f(1), f(2), f(3), f(4), f(5), f(6), f(7), f(8), f(9)) : n == 10 ? (f(1), f(2), f(3), f(4), f(5), f(6), f(7), f(8), f(9), f(10)) : _ntuple(f, n) return t end function _ntuple(f, n) @_noinline_meta (n >= 0) || throw(ArgumentError(string("tuple length should be ≥ 0, got ", n))) ([f(i) for i = 1:n]...,) end # inferrable ntuple (enough for bootstrapping) ntuple(f, ::Val{0}) = () ntuple(f, ::Val{1}) = (@_inline_meta; (f(1),)) ntuple(f, ::Val{2}) = (@_inline_meta; (f(1), f(2))) ntuple(f, ::Val{3}) = (@_inline_meta; (f(1), f(2), f(3))) @inline function ntuple(f::F, ::Val{N}) where {F,N} N::Int (N >= 0) || throw(ArgumentError(string("tuple length should be ≥ 0, got ", N))) if @generated quote @nexprs $N i -> t_i = f(i) @ncall $N tuple t end else Tuple(f(i) for i = 1:N) end end @inline function fill_to_length(t::Tuple, val, ::Val{N}) where {N} M = length(t) M > N && throw(ArgumentError("input tuple of length $M, requested $N")) if @generated quote (t..., $(fill(:val, N-length(t.parameters))...)) end else (t..., fill(val, N-M)...) end end