Sign Up

Sign Up to our social questions and Answers Engine to ask questions, answer people’s questions, and connect with other people.

Have an account? Sign In

Have an account? Sign In Now

Sign In

Login to our social questions & Answers Engine to ask questions answer people’s questions & connect with other people.

Sign Up Here

Forgot Password?

Don't have account, Sign Up Here

Forgot Password

Lost your password? Please enter your email address. You will receive a link and will create a new password via email.

Have an account? Sign In Now

You must login to ask a question.

Forgot Password?

Need An Account, Sign Up Here

Please briefly explain why you feel this question should be reported.

Please briefly explain why you feel this answer should be reported.

Please briefly explain why you feel this user should be reported.

Sign InSign Up

The Archive Base

The Archive Base Logo The Archive Base Logo

The Archive Base Navigation

  • SEARCH
  • Home
  • About Us
  • Blog
  • Contact Us
Search
Ask A Question

Mobile menu

Close
Ask a Question
  • Home
  • Add group
  • Groups page
  • Feed
  • User Profile
  • Communities
  • Questions
    • New Questions
    • Trending Questions
    • Must read Questions
    • Hot Questions
  • Polls
  • Tags
  • Badges
  • Buy Points
  • Users
  • Help
  • Buy Theme
  • SEARCH
Home/ Questions/Q 4023908
In Process

The Archive Base Latest Questions

Editorial Team
  • 0
Editorial Team
Asked: May 20, 20262026-05-20T10:39:22+00:00 2026-05-20T10:39:22+00:00

I thought expressions like this would cause Haskell to evaluate forever. But the behaviors

  • 0

I thought expressions like this would cause Haskell to evaluate forever. But the behaviors in both GHCi and the compiled program surprised me.

For example, in GHCi, these expressions blocked until I Control+C, but consumed no CPU. Looked like it was sleeping.

let loop = loop
let loop = 1 + loop

I tried compiling these programs with GHC:

main = print loop
  where loop = 1 + loop

main = print loop
  where loop = if True then loop else 1

What was printed was:

Main: <<loop>>

So my question is: Obviously these expressions are compiled to something different than loops or recursive calls in imperative languages. What are they compiled to? Is this a special rule to handle 0-arg functions that have themselves in the right hand side, or it’s a special case of something more general that I don’t know?

[EDIT]:

One more question: If this happens to be a special handling from the compiler, what is the reason behind doing this when it’s impossible to check for all infinite loops? ‘Familiar’ languages don’t care about cases like while (true); or int f() { return f();}, right?

Many thanks.

  • 1 1 Answer
  • 0 Views
  • 0 Followers
  • 0
Share
  • Facebook
  • Report

Leave an answer
Cancel reply

You must login to add an answer.

Forgot Password?

Need An Account, Sign Up Here

1 Answer

  • Voted
  • Oldest
  • Recent
  • Random
  1. Editorial Team
    Editorial Team
    2026-05-20T10:39:22+00:00Added an answer on May 20, 2026 at 10:39 am

    GHC implements Haskell as a graph reduction machine. Imagine your program as a graph with each value as a node, and lines from it to each value that value depends on. Except, we’re lazy, so you really start with just one node — and to evaluate that node, GHC has to “enter” it and open it up to a function with arguments. It then replaces the function call with the body of the function, and attempts to reduce it enough to get it into head normal form, etc.

    The above being very handwavy and I’m sure eliding some necessary detail in the interest of brevity.

    In any case, when GHC enters a value, it generally replaces it with a black hole while the node is being evaluated (or, depending on your terminology, while the closure is being reduced) This has a number of purposes. First, it plugs a potential space leak. If the node references a value which is used nowhere else, the black hole allows that value to be garbage-collected even while the node is being evaluated. Second, this prevents certain types of duplicate work, since in a multi-threaded environment, two threads may attempt to enter the same value. The black-hole will cause the second thread to block rather than evaluate the value already being evaluated. Finally, this happens to allow for a limited form of loop detection, since if a thread attempts to re-enter its own black hole, we can throw an exception.

    Here’s a bit of a more metaphorical explanation. If I have a series of instructions that moves a turtle (in logo) around the screen, there’s no one way to tell what shape they will produce, or whether that shape terminates without running them. But if, while running them, I notice that the path of the turtle has crossed itself, I can indicate to the user “aha! the turtle has crossed its path!” So I know that the turtle has reached a spot it has been before — if the path is a circuit through evaluating the nodes of a graph, then that tells us we’re in a loop. However, the turtle can also go in, for example, an expanding spiral. And it will never terminate, but it will also never cross its prior path.

    So, because of the use of black holes, for multiple reasons, we have some notion of a marked “path” that evaluation has followed. And if the path crosses itself, we can tell and throw an exception. However, there are a million ways for things to diverge that don’t involve the path crossing itself. And in those cases, we can’t tell, and don’t throw an exception.

    For super-geeky technical detail about the current implementation of black holes, see Simon Marlow’s talk from the recent Haskell Implementors Workshop, “Scheduling Lazy Evaluation on Multicore” at the bottom of http://haskell.org/haskellwiki/HaskellImplementorsWorkshop/2010.

    • 0
    • Reply
    • Share
      Share
      • Share on Facebook
      • Share on Twitter
      • Share on LinkedIn
      • Share on WhatsApp
      • Report

Sidebar

Related Questions

I thought this would be fairly simple but it turns out not to work
I thought I would post this here not so much as a question but
Thought my range of search options would easily find this. I wish to combine
I thought I had resolved this but I obviously haven't and was hoping someone
I thought I'd find more about this topic but I didn't. I have to
I thought the following script will create div element but I got nothing output
I thought I would be smart and create member functions that accepted output iterators.
I thought I was pretty well acquainted with Subversion, but now something I thought
This might sound stupid, but why doesn't the Java compiler warn about the expression
I am new to both Perl and complicated regular expressions. I mean I've used

Explore

  • Home
  • Add group
  • Groups page
  • Communities
  • Questions
    • New Questions
    • Trending Questions
    • Must read Questions
    • Hot Questions
  • Polls
  • Tags
  • Badges
  • Users
  • Help
  • SEARCH

Footer

© 2021 The Archive Base. All Rights Reserved
With Love by The Archive Base

Insert/edit link

Enter the destination URL

Or link to existing content

    No search term specified. Showing recent items. Search or use up and down arrow keys to select an item.