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

linear diophantine equations #220

Open
wants to merge 6 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 2 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
42 changes: 42 additions & 0 deletions Math/NumberTheory/Diophantine.hs
Original file line number Diff line number Diff line change
@@ -1,8 +1,15 @@
-- Module for Diophantine Equations and related functions

{-# LANGUAGE RecordWildCards, PartialTypeSignatures #-}
{-# OPTIONS_GHC -Wno-partial-type-signatures #-}
Copy link
Owner

Choose a reason for hiding this comment

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

Is it difficult to provide full type signatures?


module Math.NumberTheory.Diophantine
( cornacchiaPrimitive
, cornacchia
, Linear (..)
, LinearSolution (..)
, solveLinear
, runLinearSolution
)
where

Expand All @@ -14,6 +21,9 @@ import Math.NumberTheory.Primes ( factorise
import Math.NumberTheory.Roots ( integerSquareRoot )
import Math.NumberTheory.Utils.FromIntegral

import Control.Monad (guard)
import Data.Euclidean (gcdExt)

-- | See `cornacchiaPrimitive`, this is the internal algorithm implementation
-- | as described at https://en.wikipedia.org/wiki/Cornacchia%27s_algorithm
cornacchiaPrimitive' :: Integer -> Integer -> [(Integer, Integer)]
Expand Down Expand Up @@ -64,3 +74,35 @@ cornacchia d m
where
candidates = map (\sf -> (sf, m `div` (sf * sf))) (squareFactors m)
solve (sf, m') = map (\(x, y) -> (x * sf, y * sf)) (cornacchiaPrimitive d m')

----

-- | A linear diophantine equation `ax + by = c`
-- | where `x` and `y` are unknown
data Linear a = Lin { a,b,c :: a }
Copy link
Owner

Choose a reason for hiding this comment

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

This makes a, b and c globally defined functions, stealing them from any other potential usage. Could you please use something more specific and less likely to clash? Same applies to LinearSolution.

Copy link
Author

@BlackCapCoder BlackCapCoder Nov 19, 2022

Choose a reason for hiding this comment

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

If that is a problem I am more inclined to not have names at all, or something like a_var, or removing Linear all together in favor of function arguments? The current names are what is used in the literature

Copy link
Author

Choose a reason for hiding this comment

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

I am not as attached to the names in LinearSolution. The only reason I have it rather than returning Maybe (Int -> (Int, Int)) is that runLinearSolution is a very simple function, and applying it is technically lossy

Copy link
Author

@BlackCapCoder BlackCapCoder Nov 19, 2022

Choose a reason for hiding this comment

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

Sorry for the spam.

I am leaning towards just removing Linear. It is only used in one function- the user can create their own data type if they really want it.

On the other hand, I know that Linear has a sensible Semigroup instance, and many analogues of liftA2 f makes sense for it:

-- Basically `liftA2 (+)`
-- (a1+a2)*x + (b1+b2)*y = c1 + c2
--
add (Lin a1 b1 c1) (Lin a2 b2 c2) =
  Lin (a1+a2) (b1+b2) (c1+c2)

I didn't add instances because I didn't want to bloat this PR.

deriving
( Show, Eq, Ord
)

-- | A solution to a linear equation
data LinearSolution a = LS { x,v,y,u :: a }
deriving
( Show, Eq, Ord
)

-- | Produces an unique solution given any
-- | arbitrary number k
runLinearSolution :: _ => LinearSolution a -> a -> (a, a)
runLinearSolution LS {..} k =
( x + k*v, y - k*u )

solveLinear :: _ => Linear a -> Maybe (LinearSolution a)
solveLinear Lin {..} =
LS {..} <$ guard (b /= 0 && q == 0)
where
(d, e) = gcdExt a b
(h, q) = divMod c d
f = div (a*e-d) (-b)
(x, y) = (e*h, f*h)
(u, v) = (quot a d, quot b d)

10 changes: 10 additions & 0 deletions test-suite/Math/NumberTheory/DiophantineTests.hs
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@

{-# LANGUAGE CPP #-}

{-# LANGUAGE RecordWildCards, GADTs #-}
{-# OPTIONS_GHC -fno-warn-type-defaults #-}

module Math.NumberTheory.DiophantineTests
Expand Down Expand Up @@ -33,8 +34,17 @@ cornacchiaBruteForce (Positive d) (Positive a) = gcd d m /= 1 || findSolutions [
where x2 = m - d*y*y
x = integerSquareRoot x2

linearTest :: (a ~ Integer) => a -> a -> a -> a -> Bool
linearTest a b c k =
case solveLinear Lin {..} of
Nothing -> True -- Disproving this would require a counter example
Copy link
Owner

Choose a reason for hiding this comment

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

The problem is that solveLinear = const Nothing passes this test. Can we come up with a test which requires solveLinear return Just{} at least in some cases?

Copy link
Author

@BlackCapCoder BlackCapCoder Nov 16, 2022

Choose a reason for hiding this comment

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

Yes! I think this is easier to see in terms of my Pogo problem D = H*L (mod C) where H is the solution. All we have to do is make sure L is coprime to C, because sort [mod (h*l) c | h <- [0..pred c]] == [0..pred c], which includes D.

Just ls | (x, y) <- runLinearSolution ls k
-> a*x + b*y == c


testSuite :: TestTree
testSuite = testGroup "Diophantine"
[ testSmallAndQuick "Cornacchia correct" cornacchiaTest
, testSmallAndQuick "Cornacchia same solutions as brute force" cornacchiaBruteForce
, testSmallAndQuick "Linear correct" linearTest
]