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

Declare exported functions as IO, call initialize internally instead of exporting it. #118

Closed
wants to merge 0 commits into from

Conversation

adrianmay
Copy link

Previously, some of the exported haskell functions were declared pure but used unsafePerformIO internally. That just encourages the calling code to lazily skip the call. Now, everything correctly returns IO.

It had been obligatory for callers to call initialize before anything else, but when the ultimate calling code uses this library indirectly through other libraries, it's not clear which package should call initialize. The logical answer is that this library should do so itself, so now it does so in fec. It's not possible to call encode or decode without having obtained a handle from fec, so those functions don't need to worry about initialization. There is a slight overhead that subsequent calls to fec must call into C to check the boolean that says whether or not it's been called already, but then they bail out fast.

A better solution to all of this would be if the numbers that fec_init stuffs into the four tables were simply defined in C as initialized data. Then there'd be no initialize function, not even internally. Since the handle returned by fec is per-client and does not get modified by encode or decode, the whole library would then be implicitly thread safe but for the danger of a client freeing their own handle too early. It would also be pure, i.e., there'd be no IOs in sight. Both would be huge improvements. The numbers would be generated by a pre-build step that runs the logic of fec_init and dumps the results in C syntax. That should be considered for a future ticket, but the present goal is just to get everything working.

@hacklschorsch
Copy link
Member

hacklschorsch commented Feb 25, 2025

I like the description of this MR! Can you put the first two paragraphs into the ChangeLog.md and the last paragraph into a new ticket?

Comment on lines 18 to 16
, env (setupFEC 94 100) makeFECBenchmarks
[ env (fec 2 3) makeFECBenchmarks
Copy link
Member

Choose a reason for hiding this comment

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

I understand this as kind of a constructor - is this the naming convention in Haskell Land? No "init" or "setup" or "create"? (I know the function was called "fec" before).

I find it a bit confusing between all the different fec functions, but that might be my as-yet under-exposure to Haskell code.

Copy link
Author

Choose a reason for hiding this comment

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

There's only one fec and this is it. Previously it was wrapped by setupFEC which returned IO and called initialize. Now that both of those things are true of fec itself, the wrapper would be a synonym, so I removed it.

@hacklschorsch
Copy link
Member

A more speaking title would be nice - this PR changes the interface and quite a bit in general

@hacklschorsch
Copy link
Member

hacklschorsch commented Mar 7, 2025

Can we keep the old initializing interface (setupFEC(), more?) around, but make it a no-op (maybe emit a warning at compile time), so people can upgrade more easily?

@adrianmay adrianmay marked this pull request as draft March 7, 2025 11:39
@exarkun
Copy link
Member

exarkun commented Mar 7, 2025

See also #84

Copy link
Member

@hacklschorsch hacklschorsch left a comment

Choose a reason for hiding this comment

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

Doesn't the removing of the upper version constraints in the cabal file mean we can't publish the library to hackage anymore? (Asking because it's already there: https://hackage.haskell.org/package/fec)

fec.cabal Outdated
version: 0.2.0
version: 1.1.1
Copy link
Member

Choose a reason for hiding this comment

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

If we can keep the initialize call but make it a no-op we don't have to bump the major version, right?

Copy link
Author

Choose a reason for hiding this comment

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

Actually we need to bump the major version anyway cos everything returns IO now. That's also a breaking change, and an unavoidable one cos not returning IO from something that has side effects was probably the problem all along.

@adrianmay
Copy link
Author

See also #84

Shame it didn't happen. I think the unsafePerformIO is the real reason for the "race condition". Haskell can't prove that it needs to do something that returns () if it doesn't see it as IO.

@adrianmay
Copy link
Author

I like the description of this MR! Can you put the first two paragraphs into the ChangeLog.md and the last paragraph into a new ticket?

Done: #126

@adrianmay adrianmay changed the title Fix multithreading issues Declare exported functions as IO, call initialize internally instead of exporting it. Mar 10, 2025
@adrianmay adrianmay closed this Mar 17, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants