diff --git a/README.markdown b/README.markdown index 918aa18..80f9767 100644 --- a/README.markdown +++ b/README.markdown @@ -148,3 +148,12 @@ Even in inner blocks. | } | } ? undeclared location + +No duplicate declarations. + + | reserve word score + | assign word score 4000 + | routine main { + | nop + | } + ? duplicate declaration diff --git a/src/SixtyPical/Checker.hs b/src/SixtyPical/Checker.hs index 26ac73f..506abdb 100644 --- a/src/SixtyPical/Checker.hs +++ b/src/SixtyPical/Checker.hs @@ -14,11 +14,12 @@ routineDeclared routName (Program _ routines) = where getRoutineName (Routine name _) = name +getDeclLocationName (Assign name _ _) = name +getDeclLocationName (Reserve name _) = name + locationDeclared locName (Program decls _) = - elem locName (map (getLocationName) decls) + elem locName (map (getDeclLocationName) decls) where - getLocationName (Assign name _ _) = name - getLocationName (Reserve name _) = name routineUsedLocations (Routine _ instrs) = blockUsedLocations instrs @@ -41,6 +42,21 @@ allRoutineLocationsDeclared program routine = allUsedLocationsDeclared p@(Program _ routines) = allTrue (map (allRoutineLocationsDeclared p) routines) +noDuplicateDecls p@(Program decls routines) = + collectDecls decls [] + where + collectDecls [] acc = True + collectDecls (decl:decls) acc = + if + name `elem` acc + then + error ("duplicate declaration '" ++ name ++ "'") + else + collectDecls decls (name:acc) + where + name = getDeclLocationName decl + checkProgram program = trueOrDie "missing 'main' routine" (routineDeclared "main" program) && - trueOrDie "undeclared location" (allUsedLocationsDeclared program) + trueOrDie "undeclared location" (allUsedLocationsDeclared program) && + noDuplicateDecls program