Skip to content

Commit

Permalink
Treat non-AbstractArrays as scalars in broadcast
Browse files Browse the repository at this point in the history
  • Loading branch information
pabloferz committed Jun 20, 2016
1 parent 858371f commit 5032224
Show file tree
Hide file tree
Showing 5 changed files with 39 additions and 10 deletions.
15 changes: 12 additions & 3 deletions base/abstractarray.jl
Original file line number Diff line number Diff line change
Expand Up @@ -148,10 +148,11 @@ abstract IndicesBehavior
immutable IndicesStartAt1 <: IndicesBehavior end # indices 1:size(A,d)
immutable IndicesUnitRange <: IndicesBehavior end # arb UnitRange indices
immutable IndicesList <: IndicesBehavior end # indices like (:cat, :dog, :mouse)
immutable NoIndices <: IndicesBehavior end # default

indicesbehavior(A::AbstractArray) = indicesbehavior(typeof(A))
indicesbehavior{T<:AbstractArray}(::Type{T}) = IndicesStartAt1()
indicesbehavior(::Number) = IndicesStartAt1()
indicesbehavior(::Any) = NoIndices()

abstract IndicesPerformance
immutable IndicesFast1D <: IndicesPerformance end # indices(A, d) is fast
Expand All @@ -178,6 +179,8 @@ start at something different from 1), it is equivalent to `indices(A,
d)`.
"""
shape(a, d) = shape(indicesbehavior(a), a, d)
shape(::NoIndices, a) = (1,)
shape(::NoIndices, a, d) = (1,)
shape(::IndicesStartAt1, a) = size(a)
shape(::IndicesStartAt1, a, d) = size(a, d)
shape(::IndicesBehavior, a) = indices(a)
Expand Down Expand Up @@ -412,8 +415,9 @@ end
promote_indices(a::AbstractArray, b::AbstractArray) = _promote_indices(indicesbehavior(a), indicesbehavior(b), a, b)
_promote_indices(::IndicesStartAt1, ::IndicesStartAt1, a, b) = a
_promote_indices(::IndicesBehavior, ::IndicesBehavior, a, b) = throw(ArgumentError("types $(typeof(a)) and $(typeof(b)) do not have promote_indices defined"))
promote_indices(a::Number, b::AbstractArray) = b
promote_indices(a::AbstractArray, b::Number) = a
promote_indices(a, b::AbstractArray) = b
promote_indices(a::AbstractArray, b) = a
promote_indices(a, b) = a

# Strip off the index-changing container---this assumes that `parent`
# performs such an operation. TODO: since few things in Base need this, it
Expand Down Expand Up @@ -1459,9 +1463,14 @@ end
promote_eltype_op(::Any) = (@_pure_meta; Bottom)
promote_eltype_op{T}(op, ::AbstractArray{T}) = (@_pure_meta; promote_op(op, T))
promote_eltype_op{T}(op, ::T ) = (@_pure_meta; promote_op(op, T))
promote_eltype_op{T}(op, Ts::AbstractArray{DataType}, ::AbstractArray{T}) = typejoin((promote_op(op, S, T) for S in Ts)...)
promote_eltype_op{T}(op, Ts::AbstractArray{DataType}, ::T ) = typejoin((promote_op(op, S, T) for S in Ts)...)
promote_eltype_op{R,S}(op, ::AbstractArray{R}, ::AbstractArray{S}) = (@_pure_meta; promote_op(op, R, S))
promote_eltype_op{R,S}(op, ::AbstractArray{R}, ::S) = (@_pure_meta; promote_op(op, R, S))
promote_eltype_op{R,S}(op, ::R, ::AbstractArray{S}) = (@_pure_meta; promote_op(op, R, S))
promote_eltype_op{R,S}(op, ::Type{R}, ::AbstractArray{S}) = (@_pure_meta; promote_op(op, R, S))
promote_eltype_op{R,S}(op, ::Type{R}, ::S) = (@_pure_meta; promote_op(op, S, R))
promote_eltype_op{R,S}(op, ::R, ::S) = (@_pure_meta; promote_op(op, S, R))
promote_eltype_op(op, A, B, C, D...) = (@_pure_meta; promote_op(op, eltype(A), promote_eltype_op(op, B, C, D...)))

## 1 argument
Expand Down
17 changes: 11 additions & 6 deletions base/broadcast.jl
Original file line number Diff line number Diff line change
Expand Up @@ -13,8 +13,9 @@ export broadcast_getindex, broadcast_setindex!
## Calculate the broadcast shape of the arguments, or error if incompatible
# array inputs
broadcast_shape() = ()
broadcast_shape(A) = shape(A)
@inline broadcast_shape(A, B...) = broadcast_shape((), shape(A), map(shape, B)...)
broadcast_shape(A) = ()
broadcast_shape(A::AbstractArray) = shape(A)
@inline broadcast_shape(A, B...) = broadcast_shape((), broadcast_shape(A), map(broadcast_shape, B)...)
# shape inputs
broadcast_shape(shape::Tuple) = shape
@inline broadcast_shape(shape::Tuple, shape1::Tuple, shapes::Tuple...) = broadcast_shape(_bcs((), shape, shape1), shapes...)
Expand All @@ -40,7 +41,7 @@ _bcsm(a::Number, b::Number) = a == b || b == 1
## Check that all arguments are broadcast compatible with shape
# comparing one input against a shape
check_broadcast_shape(shp) = nothing
check_broadcast_shape(shp, A) = check_broadcast_shape(shp, shape(A))
check_broadcast_shape(shp, A) = check_broadcast_shape(shp, broadcast_shape(A))
check_broadcast_shape(::Tuple{}, ::Tuple{}) = nothing
check_broadcast_shape(shp, ::Tuple{}) = nothing
check_broadcast_shape(::Tuple{}, Ashp::Tuple) = throw(DimensionMismatch("cannot broadcast array to have fewer dimensions"))
Expand All @@ -63,8 +64,8 @@ end
@inline _newindex(out, I) = out # can truncate if indexmap is shorter than I
@inline _newindex(out, I, keep::Bool, indexmap...) = _newindex((out..., ifelse(keep, I[1], 1)), tail(I), indexmap...)

newindexer(sz, x::Number) = ()
@inline newindexer(sz, A) = _newindexer(sz, size(A))
newindexer(sz, x) = ()
@inline newindexer(sz, A::AbstractArray) = _newindexer(sz, size(A))
@inline _newindexer(sz, szA::Tuple{}) = ()
@inline _newindexer(sz, szA) = (sz[1] == szA[1], _newindexer(tail(sz), tail(szA))...)

Expand All @@ -79,6 +80,10 @@ const bitcache_size = 64 * bitcache_chunks # do not change this
dumpbitcache(Bc::Vector{UInt64}, bind::Int, C::Vector{Bool}) =
Base.copy_to_bitarray_chunks!(Bc, ((bind - 1) << 6) + 1, C, 1, min(bitcache_size, (length(Bc)-bind+1) << 6))

# Since we can't make T[1] return T, use this inside `_broadcast!`
@inline _broadcast_getvals(A, I) = A
@inline _broadcast_getvals(A::AbstractArray, I) = A[I]

## Broadcasting core
# nargs encodes the number of As arguments (which matches the number
# of indexmaps). The first two type parameters are to ensure specialization.
Expand All @@ -92,7 +97,7 @@ dumpbitcache(Bc::Vector{UInt64}, bind::Int, C::Vector{Bool}) =
# reverse-broadcast the indices
@nexprs $nargs i->(I_i = newindex(I, imap_i))
# extract array values
@nexprs $nargs i->(@inbounds val_i = A_i[I_i])
@nexprs $nargs i->(@inbounds val_i = _broadcast_getvals(A_i, I_i))
# call the function and store the result
@inbounds B[I] = @ncall $nargs f val
end
Expand Down
11 changes: 11 additions & 0 deletions base/float.jl
Original file line number Diff line number Diff line change
Expand Up @@ -199,6 +199,17 @@ promote_rule(::Type{Float64}, ::Type{Float32}) = Float64
widen(::Type{Float16}) = Float32
widen(::Type{Float32}) = Float64

promote_op{T<:Union{Float32,Float64}}(::typeof(trunc), ::Type{Signed}, ::Type{T}) = Int
promote_op{T<:Union{Float32,Float64}}(::typeof(trunc), ::Type{Unsigned}, ::Type{T}) = UInt
for Ti in (Int8, Int16, Int32, Int64, Int128, UInt8, UInt16, UInt32, UInt64, UInt128)
for Tf in (Float32, Float64)
@eval promote_op(::$(typeof(trunc)), ::Type{$Ti}, ::Type{$Tf}) = $Ti
end
end
for f in (ceil, floor, round)
@eval promote_op{R,S}(::$(typeof(f)), ::Type{R}, ::Type{S}) = promote_op($trunc, R, S)
end

## floating point arithmetic ##
-(x::Float32) = box(Float32,neg_float(unbox(Float32,x)))
-(x::Float64) = box(Float64,neg_float(unbox(Float64,x)))
Expand Down
3 changes: 3 additions & 0 deletions base/parse.jl
Original file line number Diff line number Diff line change
Expand Up @@ -194,3 +194,6 @@ function parse(str::AbstractString; raise::Bool=true)
end
return ex
end

promote_op{R<:AbstractFloat,S<:AbstractString}(::typeof(parse), ::Type{R}, ::Type{S}) = R
promote_op{R<:Integer,S<:Union{AbstractString,Char}}(::typeof(parse), ::Type{R}, ::Type{S}) = R
3 changes: 2 additions & 1 deletion base/promotion.jl
Original file line number Diff line number Diff line change
Expand Up @@ -222,8 +222,9 @@ minmax(x::Real, y::Real) = minmax(promote(x, y)...)
# for the multiplication of two types,
# promote_op{R<:MyType,S<:MyType}(::typeof(*), ::Type{R}, ::Type{S}) = MyType{multype(R,S)}
promote_op(::Any) = (@_pure_meta; Bottom)
promote_op(::Any, T) = (@_pure_meta; T)
promote_op(::Any, T) = (@_pure_meta; Bottom)
promote_op{T}(::Type{T}, ::Any) = (@_pure_meta; T)
promote_op{R,S}(::typeof(convert), ::Type{R}, ::Type{S}) = (@_pure_meta; R)
promote_op{R,S}(::Any, ::Type{R}, ::Type{S}) = (@_pure_meta; promote_type(R, S))
promote_op(op, T, S, U, V...) = (@_pure_meta; promote_op(op, T, promote_op(op, S, U, V...)))

Expand Down

0 comments on commit 5032224

Please sign in to comment.