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

macro documentation and implementation mismatch #4873

Closed
RauliRuohonen opened this issue Nov 20, 2013 · 5 comments
Closed

macro documentation and implementation mismatch #4873

RauliRuohonen opened this issue Nov 20, 2013 · 5 comments
Assignees
Labels
compiler:lowering Syntax lowering (compiler front end, 2nd stage)

Comments

@RauliRuohonen
Copy link

From the docs (Manual/Metaprogramming):

macro time(ex)
  quote
    local t0 = time()
    local val = $ex
    local t1 = time()
    println("elapsed time: ", t1-t0, " seconds")
    val
  end
end

Julia’s macro expander solves these problems in the following way. [...] Local variables are then renamed to be unique (using the gensym function, which generates new symbols), and global variables are resolved within the macro definition environment. Therefore both of the above concerns are handled; the macro’s locals will not conflict with any user variables, and time and println will refer to the standard library definitions.

Yet if you put the above code verbatim into a script and tack this to the end:

println("I'm println in the macro definition environment and I'm all right.")

function g()  
    println("That same println still works. Functions' hygiene is impeccable!")
end

function f()
    println = "I'm println the user variable! Thou shalt not conflict with me!"
    g()
    @time sleep(1)
end

f()  

You get the output:

I'm println in the macro definition environment and I'm all right.
That same println still works. Functions' hygiene is impeccable!
ERROR: type: f: in apply, expected Function, got ASCIIString

That println sure doesn't refer to the standard library one like the docs say. Comparing the result to g(), one can clearly see macros' lack of hygiene compared to functions. If the macro is defined in a different module, then it works because println is expanded to TheOtherModule.println. This could be fixed by expanding it to TheCurrentModule.println when it's not from a different module, so that in any case the globals lookups are always done from some module's globals.

Another one: the definition of assert in error.jl is different than claimed in the docs. One might say in the docs that this is a simplified version of the one in error.jl. More importantly, the use of $ex instead of $(esc(ex)) would be better fixed or remarked upon ("this is subtly broken as we see in the next section" or something) lest someone imitate it because the official docs say that this is good enough for the standard library! The unescaped assert from the docs is not that hard to break:

x = 1
@assert (x -> x)(x) == 1

results in:

ERROR: x#12 not defined
JeffBezanson added a commit that referenced this issue Nov 21, 2013
unfortunately I botched the fix for #4771, which led to this problem
@ghost ghost assigned JeffBezanson Nov 21, 2013
@staticfloat
Copy link
Member

@JeffBezanson is the best thing to do here to just update the docs? Or ask users to directly use Base.println() etc... inside of macros?

@JeffBezanson
Copy link
Member

No, this is a bug.

@staticfloat
Copy link
Member

Since macros are executed at parse-time in the caller's scope, how could this work any other way? In order to ignore the function-local println you'd have to define the macro's println function as global somehow, no?

@JeffBezanson
Copy link
Member

Yes, but that is (should be) done automatically by the macro expander, which rewrites identifiers as e.g. Base.println as necessary.

JeffBezanson added a commit that referenced this issue Nov 30, 2013
unfortunately I botched the fix for #4771, which led to this problem
@ihnorton ihnorton added the compiler:lowering Syntax lowering (compiler front end, 2nd stage) label Nov 30, 2015
@vtjnash
Copy link
Member

vtjnash commented Nov 30, 2015

maybe the new GlobalRef can be inserted now to fix the first macro issue?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
compiler:lowering Syntax lowering (compiler front end, 2nd stage)
Projects
None yet
Development

No branches or pull requests

6 participants