Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Prevent crash on zero arity def in protocol #14191

Merged
merged 7 commits into from
Jan 17, 2025

Conversation

sabiwara
Copy link
Contributor

@sabiwara sabiwara commented Jan 15, 2025

We can feel the excitement in the commits 😄 1763865.

I tried main on a project and got a compiler crash due to this dependency which defines regular functions inside a protocol, basically:

defprotocol Msgpax.Packer do
  def pack(term)

  Kernel.def(pack_nan(), do: <<0xCB, -1::64>>)
end

The lib should probably be fixed and we should probably not allow this in the first place (same spirit as #14158), but right now this crashes the compiler on consolidation: (fun: :pack_nan, arity: 0)

08:40:08.671 [error] Task #PID<0.7509.0> started from #PID<0.94.0> terminating
** (FunctionClauseError) no function clause matching in :lists.duplicate/2
    (stdlib 6.2) lists.erl:510: :lists.duplicate(-1, :term)
    (elixir 1.19.0-dev) lib/protocol.ex:670: anonymous fn/3 in Protocol.new_signatures/3
    (elixir 1.19.0-dev) lib/enum.ex:2546: Enum."-reduce/3-lists^foldl/2-0-"/3
    (elixir 1.19.0-dev) lib/protocol.ex:669: Protocol.new_signatures/3
    (elixir 1.19.0-dev) lib/protocol.ex:620: Protocol.consolidate/5
    (elixir 1.19.0-dev) lib/protocol.ex:565: Protocol.consolidate/2
    (mix 1.19.0-dev) lib/mix/compilers/protocol.ex:153: Mix.Compilers.Protocol.consolidate_each/4
    (elixir 1.19.0-dev) lib/task/supervised.ex:101: Task.Supervised.invoke_mfa/2
Function: #Function<8.117671614/0 in Mix.Compilers.Protocol.consolidate/4>
    Args: []

I tried to add a test but consolidation tests are a bit high ceremony and I wasn't sure it was worth defining a fixture etc. especially if this is going to be forbidden.

@@ -666,7 +666,7 @@ defmodule Protocol do
end

new_signatures =
for {{fun, arity}, :def, _, _} <- definitions do
for {{fun, arity}, :def, _, _} when arity > 0 <- definitions do
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Unfortunately this is not enough to fix it. If people are defining custom functions, then we need to do this modification only for protocol functions. So we need to store somewhere, either in a module attribute or perhaps in the function metadata, the protocol functions and traverse only them here.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Maybe we can replace this line:

Kernel.def(unquote(name)(unquote_splicing(args)))

By:

Kernel.def(unquote({name, [protocol: true], args}))

And see if [protocol: true]appears in the metadata here.


Another option, which perhaps I like most, is to traverse the implementation of the __protocol__ and find the return for the :functions clause:

Kernel.def(__protocol__(:functions), do: unquote(:lists.sort(@__functions__)))

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

And then we need to add a test here that the signature for the user generated function does not have a :strong triplet:

test "defines signatures without fallback to any" do

It should most likely be an :infer tuple (you can match on {:infer, _, _}).

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Another option, which perhaps I like most, is to traverse the implementation of the protocol and find the return for the :functions clause:

I like it, seems straightforward. This is what you mean right? 609f7cd

And then we need to add a test here that the signature for the user generated function does not have a :strong triplet:

Will have a look!

Copy link
Contributor Author

@sabiwara sabiwara Jan 17, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@josevalim done ba95908 but I'm getting {sig: :none}.

Indeed it used to be :strong without the fix.

Screenshot 2025-01-17 at 11 10 58

@@ -665,10 +665,13 @@ defmodule Protocol do
end
end

fun_arities = :sets.from_list(protocol.__protocol__(:functions), version: 2)
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The set might be overkill since most protocols should define very few functions, so perhaps just a list is fine?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Unfortunately we cannot load the code, that would add a bunch of unnecessary work, given we would have to purge it immediately after consolidation anyway. What I proposed is to find the {:__protocol__, 1} entry inside definitions, then find the :functions clauses in there, and get its return type, which should be a keyword list. :)

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Got it! 6ff5011

Copy link
Member

@josevalim josevalim left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looks great, we just need updated tests and we can ship it!

Comment on lines 1 to 4
defprotocol Protocol.ConsolidationTest.ExtraDef do
def protocol_fun(term)

Kernel.def(regular_fun(term), do: term + 1)
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Or I can add it to NoImpl if you prefer to avoid fixture proliferation?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, let's add it to to an existing one with a comment on top.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done 9adad46

Copy link
Member

@josevalim josevalim left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ship it!!!!

Co-authored-by: José Valim <[email protected]>
@sabiwara sabiwara merged commit 618d03b into elixir-lang:main Jan 17, 2025
9 checks passed
@sabiwara sabiwara deleted the fix-protocol-crash branch January 17, 2025 11:47
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Development

Successfully merging this pull request may close these issues.

2 participants