forked from GitHub/gf-core
Compare commits
42 Commits
js-binding
...
compact-pg
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
320ead943c | ||
|
|
c119d5e34b | ||
|
|
529635e0e9 | ||
|
|
a33a84df3d | ||
|
|
9e3512db81 | ||
|
|
8a419f66a6 | ||
|
|
29662350dc | ||
|
|
a27bcb8092 | ||
|
|
4d79aa8b19 | ||
|
|
e989cc69a2 | ||
|
|
5c5af8df79 | ||
|
|
084b345663 | ||
|
|
400aad1d07 | ||
|
|
a0cfe09e09 | ||
|
|
12912299be | ||
|
|
b3c07d45b9 | ||
|
|
acb70ccc1b | ||
|
|
4a71464ca7 | ||
|
|
e993ae59f8 | ||
|
|
f12557acf8 | ||
|
|
9d3badd8b2 | ||
|
|
e2ddea6c7d | ||
|
|
59a6e3cfdd | ||
|
|
1e8d684f9a | ||
|
|
72cfc1f48a | ||
|
|
724bf67295 | ||
|
|
a7a592d93e | ||
|
|
d1bb1de87f | ||
|
|
394d033d19 | ||
|
|
cb678dfdc8 | ||
|
|
4161bbf0ec | ||
|
|
148590927c | ||
|
|
85a81ef741 | ||
|
|
3e662475ee | ||
|
|
b77626b802 | ||
|
|
12f2520b3c | ||
|
|
941b4ddf1f | ||
|
|
85f12a5544 | ||
|
|
81362ed7b7 | ||
|
|
6a5053daeb | ||
|
|
5a2b200948 | ||
|
|
bf5abe2948 |
2
.gitignore
vendored
2
.gitignore
vendored
@@ -44,6 +44,8 @@ cabal.sandbox.config
|
|||||||
.stack-work
|
.stack-work
|
||||||
DATA_DIR
|
DATA_DIR
|
||||||
|
|
||||||
|
stack*.yaml.lock
|
||||||
|
|
||||||
# Generated documentation (not exhaustive)
|
# Generated documentation (not exhaustive)
|
||||||
demos/index-numbers.html
|
demos/index-numbers.html
|
||||||
demos/resourcegrammars.html
|
demos/resourcegrammars.html
|
||||||
|
|||||||
4
debian/rules
vendored
4
debian/rules
vendored
@@ -26,8 +26,8 @@ override_dh_auto_build:
|
|||||||
cd src/runtime/python && EXTRA_INCLUDE_DIRS=$(CURDIR)/src/runtime/c EXTRA_LIB_DIRS=$(CURDIR)/src/runtime/c/.libs python setup.py build
|
cd src/runtime/python && EXTRA_INCLUDE_DIRS=$(CURDIR)/src/runtime/c EXTRA_LIB_DIRS=$(CURDIR)/src/runtime/c/.libs python setup.py build
|
||||||
cd src/runtime/java && make CFLAGS="-I$(CURDIR)/src/runtime/c -L$(CURDIR)/src/runtime/c/.libs" INSTALL_PATH=/usr/lib
|
cd src/runtime/java && make CFLAGS="-I$(CURDIR)/src/runtime/c -L$(CURDIR)/src/runtime/c/.libs" INSTALL_PATH=/usr/lib
|
||||||
echo $(SET_LDL)
|
echo $(SET_LDL)
|
||||||
$(SET_LDL) cabal build # builds gf, fails to build example grammars
|
-$(SET_LDL) cabal build # builds gf, fails to build example grammars
|
||||||
PATH=$(CURDIR)/dist/build/gf:$$PATH && make -C ../gf-rgl build
|
export $(SET_LDL); PATH=$(CURDIR)/dist/build/gf:$$PATH && make -C ../gf-rgl build
|
||||||
GF_LIB_PATH=$(CURDIR)/../gf-rgl/dist $(SET_LDL) cabal build # have RGL now, ok to build example grammars
|
GF_LIB_PATH=$(CURDIR)/../gf-rgl/dist $(SET_LDL) cabal build # have RGL now, ok to build example grammars
|
||||||
make html
|
make html
|
||||||
|
|
||||||
|
|||||||
@@ -13,13 +13,13 @@ These binary packages include both the GF core (compiler and runtime) as well as
|
|||||||
| Platform | Download | Features | How to install |
|
| Platform | Download | Features | How to install |
|
||||||
|:----------------|:---------------------------------------------------|:---------------|:-----------------------------------|
|
|:----------------|:---------------------------------------------------|:---------------|:-----------------------------------|
|
||||||
| macOS | [gf-3.10.pkg](gf-3.10.pkg) | GF, S, C, J, P | Double-click on the package icon |
|
| macOS | [gf-3.10.pkg](gf-3.10.pkg) | GF, S, C, J, P | Double-click on the package icon |
|
||||||
|
| Raspbian 10 (buster) | [gf\_3.10-2\_armhf.deb](gf_3.10-2_armhf.deb) | GF,S,C,J,P | `sudo dpkg -i gf_3.10-2_armhf.deb` |
|
||||||
| Ubuntu (32-bit) | [gf\_3.10-2\_i386.deb](gf_3.10-2_i386.deb) | GF, S, C, J, P | `sudo dpkg -i gf_3.10-2_i386.deb` |
|
| Ubuntu (32-bit) | [gf\_3.10-2\_i386.deb](gf_3.10-2_i386.deb) | GF, S, C, J, P | `sudo dpkg -i gf_3.10-2_i386.deb` |
|
||||||
| Ubuntu (64-bit) | [gf\_3.10-2\_amd64.deb](gf_3.10-2_amd64.deb) | GF, S, C, J, P | `sudo dpkg -i gf_3.10-2_amd64.deb` |
|
| Ubuntu (64-bit) | [gf\_3.10-2\_amd64.deb](gf_3.10-2_amd64.deb) | GF, S, C, J, P | `sudo dpkg -i gf_3.10-2_amd64.deb` |
|
||||||
| Windows | [gf-3.10-bin-windows.zip](gf-3.10-bin-windows.zip) | GF, S | `unzip gf-3.10-bin-windows.zip` |
|
| Windows | [gf-3.10-bin-windows.zip](gf-3.10-bin-windows.zip) | GF, S | `unzip gf-3.10-bin-windows.zip` |
|
||||||
|
|
||||||
<!--
|
<!--
|
||||||
| macOS | [gf-3.10-bin-intel-mac.tar.gz](gf-3.10-bin-intel-mac.tar.gz) | GF,S,C,J,P | `sudo tar -C /usr/local -zxf gf-3.10-bin-intel-mac.tar.gz` |
|
| macOS | [gf-3.10-bin-intel-mac.tar.gz](gf-3.10-bin-intel-mac.tar.gz) | GF,S,C,J,P | `sudo tar -C /usr/local -zxf gf-3.10-bin-intel-mac.tar.gz` |
|
||||||
| Raspbian 9.1 | [gf\_3.10-1\_armhf.deb](gf_3.10-1_armhf.deb) | GF,S,C,J,P | `sudo dpkg -i gf_3.10-1_armhf.deb` |
|
|
||||||
-->
|
-->
|
||||||
|
|
||||||
**Features**
|
**Features**
|
||||||
|
|||||||
148
gf.cabal
148
gf.cabal
@@ -47,6 +47,10 @@ custom-setup
|
|||||||
filepath,
|
filepath,
|
||||||
process >=1.0.1.1
|
process >=1.0.1.1
|
||||||
|
|
||||||
|
--source-repository head
|
||||||
|
-- type: darcs
|
||||||
|
-- location: http://www.grammaticalframework.org/
|
||||||
|
|
||||||
source-repository head
|
source-repository head
|
||||||
type: git
|
type: git
|
||||||
location: https://github.com/GrammaticalFramework/gf-core.git
|
location: https://github.com/GrammaticalFramework/gf-core.git
|
||||||
@@ -63,17 +67,12 @@ flag network-uri
|
|||||||
description: Get Network.URI from the network-uri package
|
description: Get Network.URI from the network-uri package
|
||||||
default: True
|
default: True
|
||||||
|
|
||||||
--flag new-comp
|
executable gf
|
||||||
-- Description: Make -new-comp the default
|
hs-source-dirs: src/programs
|
||||||
-- Default: True
|
main-is: gf-main.hs
|
||||||
|
|
||||||
flag c-runtime
|
|
||||||
Description: Include functionality from the C run-time library (which must be installed already)
|
|
||||||
Default: False
|
|
||||||
|
|
||||||
Library
|
|
||||||
default-language: Haskell2010
|
default-language: Haskell2010
|
||||||
build-depends: base >= 4.6 && <5,
|
build-depends: pgf2,
|
||||||
|
base >= 4.6 && <5,
|
||||||
array,
|
array,
|
||||||
containers,
|
containers,
|
||||||
bytestring,
|
bytestring,
|
||||||
@@ -82,79 +81,27 @@ Library
|
|||||||
pretty,
|
pretty,
|
||||||
mtl,
|
mtl,
|
||||||
exceptions,
|
exceptions,
|
||||||
ghc-prim
|
ghc-prim,
|
||||||
hs-source-dirs: src/runtime/haskell
|
filepath, directory>=1.2, time,
|
||||||
|
|
||||||
other-modules:
|
|
||||||
-- not really part of GF but I have changed the original binary library
|
|
||||||
-- and we have to keep the copy for now.
|
|
||||||
Data.Binary
|
|
||||||
Data.Binary.Put
|
|
||||||
Data.Binary.Get
|
|
||||||
Data.Binary.Builder
|
|
||||||
Data.Binary.IEEE754
|
|
||||||
|
|
||||||
--ghc-options: -fwarn-unused-imports
|
|
||||||
--if impl(ghc>=7.8)
|
|
||||||
-- ghc-options: +RTS -A20M -RTS
|
|
||||||
ghc-prof-options: -fprof-auto
|
|
||||||
if impl(ghc>=8.6)
|
|
||||||
Default-extensions: NoMonadFailDesugaring
|
|
||||||
|
|
||||||
exposed-modules:
|
|
||||||
PGF
|
|
||||||
PGF.Internal
|
|
||||||
PGF.Haskell
|
|
||||||
|
|
||||||
other-modules:
|
|
||||||
PGF.Data
|
|
||||||
PGF.Macros
|
|
||||||
PGF.Binary
|
|
||||||
PGF.Optimize
|
|
||||||
PGF.Printer
|
|
||||||
PGF.CId
|
|
||||||
PGF.Expr
|
|
||||||
PGF.Generate
|
|
||||||
PGF.Linearize
|
|
||||||
PGF.Morphology
|
|
||||||
PGF.Paraphrase
|
|
||||||
PGF.Parse
|
|
||||||
PGF.Probabilistic
|
|
||||||
PGF.SortTop
|
|
||||||
PGF.Tree
|
|
||||||
PGF.Type
|
|
||||||
PGF.TypeCheck
|
|
||||||
PGF.Forest
|
|
||||||
PGF.TrieMap
|
|
||||||
PGF.VisualizeTree
|
|
||||||
PGF.ByteCode
|
|
||||||
PGF.OldBinary
|
|
||||||
PGF.Utilities
|
|
||||||
|
|
||||||
if flag(c-runtime)
|
|
||||||
exposed-modules: PGF2
|
|
||||||
other-modules: PGF2.FFI PGF2.Expr PGF2.Type
|
|
||||||
GF.Interactive2 GF.Command.Commands2
|
|
||||||
hs-source-dirs: src/runtime/haskell-bind
|
|
||||||
build-tools: hsc2hs
|
|
||||||
extra-libraries: pgf gu
|
|
||||||
c-sources: src/runtime/haskell-bind/utils.c
|
|
||||||
cc-options: -std=c99
|
|
||||||
|
|
||||||
---- GF compiler as a library:
|
|
||||||
|
|
||||||
build-depends: filepath, directory>=1.2, time,
|
|
||||||
process, haskeline, parallel>=3, json
|
process, haskeline, parallel>=3, json
|
||||||
|
ghc-options: -threaded
|
||||||
|
|
||||||
|
if impl(ghc>=7.0)
|
||||||
|
ghc-options: -rtsopts -with-rtsopts=-I5
|
||||||
|
if impl(ghc<7.8)
|
||||||
|
ghc-options: -with-rtsopts=-K64M
|
||||||
|
|
||||||
|
ghc-prof-options: -auto-all
|
||||||
|
|
||||||
hs-source-dirs: src/compiler
|
hs-source-dirs: src/compiler
|
||||||
exposed-modules:
|
|
||||||
|
other-modules:
|
||||||
GF
|
GF
|
||||||
GF.Support
|
GF.Support
|
||||||
GF.Text.Pretty
|
GF.Text.Pretty
|
||||||
GF.Text.Lexing
|
GF.Text.Lexing
|
||||||
GF.Grammar.Canonical
|
GF.Grammar.Canonical
|
||||||
|
|
||||||
other-modules:
|
|
||||||
GF.Main GF.Compiler GF.Interactive
|
GF.Main GF.Compiler GF.Interactive
|
||||||
|
|
||||||
GF.Compile GF.CompileInParallel GF.CompileOne GF.Compile.GetGrammar
|
GF.Compile GF.CompileInParallel GF.CompileOne GF.Compile.GetGrammar
|
||||||
@@ -177,7 +124,6 @@ Library
|
|||||||
GF.Compile.CheckGrammar
|
GF.Compile.CheckGrammar
|
||||||
GF.Compile.Compute.AppPredefined
|
GF.Compile.Compute.AppPredefined
|
||||||
GF.Compile.Compute.ConcreteNew
|
GF.Compile.Compute.ConcreteNew
|
||||||
-- GF.Compile.Compute.ConcreteNew1
|
|
||||||
GF.Compile.Compute.Predef
|
GF.Compile.Compute.Predef
|
||||||
GF.Compile.Compute.Value
|
GF.Compile.Compute.Value
|
||||||
GF.Compile.ExampleBased
|
GF.Compile.ExampleBased
|
||||||
@@ -187,16 +133,14 @@ Library
|
|||||||
GF.Compile.GrammarToPGF
|
GF.Compile.GrammarToPGF
|
||||||
GF.Compile.Multi
|
GF.Compile.Multi
|
||||||
GF.Compile.Optimize
|
GF.Compile.Optimize
|
||||||
|
GF.Compile.OptimizePGF
|
||||||
GF.Compile.PGFtoHaskell
|
GF.Compile.PGFtoHaskell
|
||||||
GF.Compile.PGFtoJava
|
GF.Compile.PGFtoJava
|
||||||
GF.Haskell
|
GF.Haskell
|
||||||
GF.Compile.ConcreteToHaskell
|
GF.Compile.ConcreteToHaskell
|
||||||
GF.Compile.GrammarToCanonical
|
GF.Compile.GrammarToCanonical
|
||||||
GF.Grammar.CanonicalJSON
|
GF.Grammar.CanonicalJSON
|
||||||
GF.Compile.PGFtoJS
|
|
||||||
GF.Compile.PGFtoJSON
|
GF.Compile.PGFtoJSON
|
||||||
GF.Compile.PGFtoProlog
|
|
||||||
GF.Compile.PGFtoPython
|
|
||||||
GF.Compile.ReadFiles
|
GF.Compile.ReadFiles
|
||||||
GF.Compile.Rename
|
GF.Compile.Rename
|
||||||
GF.Compile.SubExOpt
|
GF.Compile.SubExOpt
|
||||||
@@ -266,11 +210,17 @@ Library
|
|||||||
GF.System.Signal
|
GF.System.Signal
|
||||||
GF.Text.Clitics
|
GF.Text.Clitics
|
||||||
GF.Text.Coding
|
GF.Text.Coding
|
||||||
|
GF.Text.Lexing
|
||||||
GF.Text.Transliterations
|
GF.Text.Transliterations
|
||||||
Paths_gf
|
Paths_gf
|
||||||
|
|
||||||
if flag(c-runtime)
|
-- not really part of GF but I have changed the original binary library
|
||||||
cpp-options: -DC_RUNTIME
|
-- and we have to keep the copy for now.
|
||||||
|
Data.Binary
|
||||||
|
Data.Binary.Put
|
||||||
|
Data.Binary.Get
|
||||||
|
Data.Binary.Builder
|
||||||
|
Data.Binary.IEEE754
|
||||||
|
|
||||||
if flag(server)
|
if flag(server)
|
||||||
build-depends: httpd-shed>=0.4.0.3, network>=2.3 && <2.7,
|
build-depends: httpd-shed>=0.4.0.3, network>=2.3 && <2.7,
|
||||||
@@ -293,8 +243,6 @@ Library
|
|||||||
CGIUtils
|
CGIUtils
|
||||||
Cache
|
Cache
|
||||||
Fold
|
Fold
|
||||||
ExampleDemo
|
|
||||||
ExampleService
|
|
||||||
hs-source-dirs: src/server src/server/transfer src/example-based
|
hs-source-dirs: src/server src/server/transfer src/example-based
|
||||||
|
|
||||||
if flag(interrupt)
|
if flag(interrupt)
|
||||||
@@ -305,7 +253,6 @@ Library
|
|||||||
|
|
||||||
if impl(ghc>=7.8)
|
if impl(ghc>=7.8)
|
||||||
build-tools: happy>=1.19, alex>=3.1
|
build-tools: happy>=1.19, alex>=3.1
|
||||||
-- ghc-options: +RTS -A20M -RTS
|
|
||||||
else
|
else
|
||||||
build-tools: happy, alex>=3
|
build-tools: happy, alex>=3
|
||||||
|
|
||||||
@@ -316,36 +263,13 @@ Library
|
|||||||
else
|
else
|
||||||
build-depends: unix, terminfo>=0.4
|
build-depends: unix, terminfo>=0.4
|
||||||
|
|
||||||
if impl(ghc>=8.2)
|
|
||||||
ghc-options: -fhide-source-paths
|
|
||||||
|
|
||||||
Executable gf
|
test-suite rgl-tests
|
||||||
hs-source-dirs: src/programs
|
type: exitcode-stdio-1.0
|
||||||
main-is: gf-main.hs
|
main-is: run.hs
|
||||||
|
hs-source-dirs: lib/tests/
|
||||||
|
build-depends: base, HTF, process, HUnit, filepath, directory
|
||||||
default-language: Haskell2010
|
default-language: Haskell2010
|
||||||
build-depends: gf, base
|
|
||||||
ghc-options: -threaded
|
|
||||||
--ghc-options: -fwarn-unused-imports
|
|
||||||
|
|
||||||
if impl(ghc>=7.0)
|
|
||||||
ghc-options: -rtsopts -with-rtsopts=-I5
|
|
||||||
if impl(ghc<7.8)
|
|
||||||
ghc-options: -with-rtsopts=-K64M
|
|
||||||
|
|
||||||
ghc-prof-options: -auto-all
|
|
||||||
|
|
||||||
if impl(ghc>=8.2)
|
|
||||||
ghc-options: -fhide-source-paths
|
|
||||||
|
|
||||||
executable pgf-shell
|
|
||||||
--if !flag(c-runtime)
|
|
||||||
buildable: False
|
|
||||||
main-is: pgf-shell.hs
|
|
||||||
hs-source-dirs: src/runtime/haskell-bind/examples
|
|
||||||
build-depends: gf, base, containers, mtl, lifted-base
|
|
||||||
default-language: Haskell2010
|
|
||||||
if impl(ghc>=7.0)
|
|
||||||
ghc-options: -rtsopts
|
|
||||||
|
|
||||||
test-suite gf-tests
|
test-suite gf-tests
|
||||||
type: exitcode-stdio-1.0
|
type: exitcode-stdio-1.0
|
||||||
|
|||||||
24
index.html
24
index.html
@@ -29,9 +29,9 @@
|
|||||||
<ul class="mb-2">
|
<ul class="mb-2">
|
||||||
<li><a href="https://www.youtube.com/watch?v=x1LFbDQhbso">Google Tech Talk</a></li>
|
<li><a href="https://www.youtube.com/watch?v=x1LFbDQhbso">Google Tech Talk</a></li>
|
||||||
<li>
|
<li>
|
||||||
<a href="http://cloud.grammaticalframework.org/">
|
<a href="//cloud.grammaticalframework.org/">
|
||||||
GF Cloud
|
GF Cloud
|
||||||
<img src="http://www.grammaticalframework.org/src/www/P/gf-cloud.png" style="height:30px" class="ml-2" alt="Cloud logo">
|
<img src="src/www/P/gf-cloud.png" style="height:30px" class="ml-2" alt="Cloud logo">
|
||||||
</a>
|
</a>
|
||||||
</li>
|
</li>
|
||||||
<li>
|
<li>
|
||||||
@@ -83,7 +83,7 @@
|
|||||||
<li><a href="http://groups.google.com/group/gf-dev">Mailing List</a></li>
|
<li><a href="http://groups.google.com/group/gf-dev">Mailing List</a></li>
|
||||||
<li><a href="https://github.com/GrammaticalFramework/gf-core/issues">Issue Tracker</a></li>
|
<li><a href="https://github.com/GrammaticalFramework/gf-core/issues">Issue Tracker</a></li>
|
||||||
<li><a href="doc/gf-people.html">Authors</a></li>
|
<li><a href="doc/gf-people.html">Authors</a></li>
|
||||||
<li><a href="http://school.grammaticalframework.org/2018/">Summer School</a></li>
|
<li><a href="//school.grammaticalframework.org/2018/">Summer School</a></li>
|
||||||
</ul>
|
</ul>
|
||||||
<a href="https://github.com/GrammaticalFramework/" class="btn btn-primary ml-3">
|
<a href="https://github.com/GrammaticalFramework/" class="btn btn-primary ml-3">
|
||||||
<i class="fab fa-github mr-1"></i>
|
<i class="fab fa-github mr-1"></i>
|
||||||
@@ -152,9 +152,9 @@ least one, it may help you to get a first idea of what GF is.
|
|||||||
<h2>Applications & Availability</h2>
|
<h2>Applications & Availability</h2>
|
||||||
<p>
|
<p>
|
||||||
GF can be used for building
|
GF can be used for building
|
||||||
<a href="http://cloud.grammaticalframework.org/translator/">translation systems</a>,
|
<a href="//cloud.grammaticalframework.org/translator/">translation systems</a>,
|
||||||
<a href="http://cloud.grammaticalframework.org/minibar/minibar.html">multilingual web gadgets</a>,
|
<a href="//cloud.grammaticalframework.org/minibar/minibar.html">multilingual web gadgets</a>,
|
||||||
<a href="http://www.cs.chalmers.se/~hallgren/Alfa/Tutorial/GFplugin.html">natural-language interfaces</a>,
|
<a href="http://www.cse.chalmers.se/~hallgren/Alfa/Tutorial/GFplugin.html">natural-language interfaces</a>,
|
||||||
<a href="http://www.youtube.com/watch?v=1bfaYHWS6zU">dialogue systems</a>, and
|
<a href="http://www.youtube.com/watch?v=1bfaYHWS6zU">dialogue systems</a>, and
|
||||||
<a href="lib/doc/synopsis/index.html">natural language resources</a>.
|
<a href="lib/doc/synopsis/index.html">natural language resources</a>.
|
||||||
</p>
|
</p>
|
||||||
@@ -210,7 +210,7 @@ least one, it may help you to get a first idea of what GF is.
|
|||||||
<p>
|
<p>
|
||||||
We run the IRC channel <strong><code>#gf</code></strong> on the Freenode network, where you are welcome to look for help with small questions or just start a general discussion.
|
We run the IRC channel <strong><code>#gf</code></strong> on the Freenode network, where you are welcome to look for help with small questions or just start a general discussion.
|
||||||
You can <a href="https://webchat.freenode.net/?channels=gf">open a web chat</a>
|
You can <a href="https://webchat.freenode.net/?channels=gf">open a web chat</a>
|
||||||
or <a href="http://www.grammaticalframework.org/irc/">browse the channel logs</a>.
|
or <a href="/irc/">browse the channel logs</a>.
|
||||||
</p>
|
</p>
|
||||||
<p>
|
<p>
|
||||||
If you have a larger question which the community may benefit from, we recommend you ask it on the <a href="http://groups.google.com/group/gf-dev">mailing list</a>.
|
If you have a larger question which the community may benefit from, we recommend you ask it on the <a href="http://groups.google.com/group/gf-dev">mailing list</a>.
|
||||||
@@ -224,7 +224,7 @@ least one, it may help you to get a first idea of what GF is.
|
|||||||
<dl class="row">
|
<dl class="row">
|
||||||
<dt class="col-sm-3 text-center text-nowrap">2018-12-03</dt>
|
<dt class="col-sm-3 text-center text-nowrap">2018-12-03</dt>
|
||||||
<dd class="col-sm-9">
|
<dd class="col-sm-9">
|
||||||
<a href="http://school.grammaticalframework.org/2018/">Sixth GF Summer School</a> in Stellenbosch (South Africa), 3–14 December 2018
|
<a href="//school.grammaticalframework.org/2018/">Sixth GF Summer School</a> in Stellenbosch (South Africa), 3–14 December 2018
|
||||||
</dd>
|
</dd>
|
||||||
<dt class="col-sm-3 text-center text-nowrap">2018-12-02</dt>
|
<dt class="col-sm-3 text-center text-nowrap">2018-12-02</dt>
|
||||||
<dd class="col-sm-9">
|
<dd class="col-sm-9">
|
||||||
@@ -248,7 +248,7 @@ least one, it may help you to get a first idea of what GF is.
|
|||||||
GF is moving to <a href="https://github.com/GrammaticalFramework/GF/">GitHub</a>.</dd>
|
GF is moving to <a href="https://github.com/GrammaticalFramework/GF/">GitHub</a>.</dd>
|
||||||
<dt class="col-sm-3 text-center text-nowrap">2017-03-13</dt>
|
<dt class="col-sm-3 text-center text-nowrap">2017-03-13</dt>
|
||||||
<dd class="col-sm-9">
|
<dd class="col-sm-9">
|
||||||
<a href="http://school.grammaticalframework.org/2017/">GF Summer School</a> in Riga (Latvia), 14-25 August 2017
|
<a href="//school.grammaticalframework.org/2017/">GF Summer School</a> in Riga (Latvia), 14-25 August 2017
|
||||||
</dd>
|
</dd>
|
||||||
</dl>
|
</dl>
|
||||||
|
|
||||||
@@ -268,7 +268,7 @@ least one, it may help you to get a first idea of what GF is.
|
|||||||
</p>
|
</p>
|
||||||
<ul>
|
<ul>
|
||||||
<li>
|
<li>
|
||||||
<a href="http://www.cs.chalmers.se/~hallgren/Alfa/Tutorial/GFplugin.html">GF-Alfa</a>:
|
<a href="http://www.cse.chalmers.se/~hallgren/Alfa/Tutorial/GFplugin.html">GF-Alfa</a>:
|
||||||
natural language interface to formal proofs
|
natural language interface to formal proofs
|
||||||
</li>
|
</li>
|
||||||
<li>
|
<li>
|
||||||
@@ -293,11 +293,11 @@ least one, it may help you to get a first idea of what GF is.
|
|||||||
<a href="http://www.cse.chalmers.se/alumni/markus/FM/">Functional Morphology</a>
|
<a href="http://www.cse.chalmers.se/alumni/markus/FM/">Functional Morphology</a>
|
||||||
</li>
|
</li>
|
||||||
<li>
|
<li>
|
||||||
<a href="http://www.molto-project.eu">MOLTO</a>:
|
<a href="//www.molto-project.eu">MOLTO</a>:
|
||||||
multilingual online translation
|
multilingual online translation
|
||||||
</li>
|
</li>
|
||||||
<li>
|
<li>
|
||||||
<a href="http://remu.grammaticalframework.org">REMU</a>:
|
<a href="//remu.grammaticalframework.org">REMU</a>:
|
||||||
reliable multilingual digital communication
|
reliable multilingual digital communication
|
||||||
</li>
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
module GF.Command.Abstract(module GF.Command.Abstract,Expr,showExpr,Term) where
|
module GF.Command.Abstract(module GF.Command.Abstract,Expr,showExpr,Term) where
|
||||||
|
|
||||||
import PGF(CId,mkCId,Expr,showExpr)
|
import PGF2(Expr,showExpr)
|
||||||
import GF.Grammar.Grammar(Term)
|
import GF.Grammar.Grammar(Term)
|
||||||
|
|
||||||
type Ident = String
|
type Ident = String
|
||||||
@@ -11,7 +11,7 @@ type Pipe = [Command]
|
|||||||
|
|
||||||
data Command
|
data Command
|
||||||
= Command Ident [Option] Argument
|
= Command Ident [Option] Argument
|
||||||
deriving (Eq,Ord,Show)
|
deriving Show
|
||||||
|
|
||||||
data Option
|
data Option
|
||||||
= OOpt Ident
|
= OOpt Ident
|
||||||
@@ -29,13 +29,7 @@ data Argument
|
|||||||
| ATerm Term
|
| ATerm Term
|
||||||
| ANoArg
|
| ANoArg
|
||||||
| AMacro Ident
|
| AMacro Ident
|
||||||
deriving (Eq,Ord,Show)
|
deriving Show
|
||||||
|
|
||||||
valCIdOpts :: String -> CId -> [Option] -> CId
|
|
||||||
valCIdOpts flag def opts =
|
|
||||||
case [v | OFlag f (VId v) <- opts, f == flag] of
|
|
||||||
(v:_) -> mkCId v
|
|
||||||
_ -> def
|
|
||||||
|
|
||||||
valIntOpts :: String -> Int -> [Option] -> Int
|
valIntOpts :: String -> Int -> [Option] -> Int
|
||||||
valIntOpts flag def opts =
|
valIntOpts flag def opts =
|
||||||
@@ -49,6 +43,18 @@ valStrOpts flag def opts =
|
|||||||
v:_ -> valueString v
|
v:_ -> valueString v
|
||||||
_ -> def
|
_ -> def
|
||||||
|
|
||||||
|
maybeIntOpts :: String -> a -> (Int -> a) -> [Option] -> a
|
||||||
|
maybeIntOpts flag def fn opts =
|
||||||
|
case [v | OFlag f (VInt v) <- opts, f == flag] of
|
||||||
|
(v:_) -> fn v
|
||||||
|
_ -> def
|
||||||
|
|
||||||
|
maybeStrOpts :: String -> a -> (String -> a) -> [Option] -> a
|
||||||
|
maybeStrOpts flag def fn opts =
|
||||||
|
case listFlags flag opts of
|
||||||
|
v:_ -> fn (valueString v)
|
||||||
|
_ -> def
|
||||||
|
|
||||||
listFlags flag opts = [v | OFlag f v <- opts, f == flag]
|
listFlags flag opts = [v | OFlag f v <- opts, f == flag]
|
||||||
|
|
||||||
valueString v =
|
valueString v =
|
||||||
|
|||||||
@@ -3,8 +3,7 @@ import GF.Command.Abstract(Option,Expr,Term)
|
|||||||
import GF.Text.Pretty(render)
|
import GF.Text.Pretty(render)
|
||||||
import GF.Grammar.Printer() -- instance Pretty Term
|
import GF.Grammar.Printer() -- instance Pretty Term
|
||||||
import GF.Grammar.Macros(string2term)
|
import GF.Grammar.Macros(string2term)
|
||||||
import qualified PGF as H(showExpr)
|
import PGF2(mkStr,unStr,showExpr)
|
||||||
import qualified PGF.Internal as H(Literal(LStr),Expr(ELit)) ----
|
|
||||||
|
|
||||||
data CommandInfo m = CommandInfo {
|
data CommandInfo m = CommandInfo {
|
||||||
exec :: [Option] -> CommandArguments -> m CommandOutput,
|
exec :: [Option] -> CommandArguments -> m CommandOutput,
|
||||||
@@ -38,21 +37,19 @@ class Monad m => TypeCheckArg m where typeCheckArg :: Expr -> m Expr
|
|||||||
|
|
||||||
--------------------------------------------------------------------------------
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
data CommandArguments = Exprs [Expr] | Strings [String] | Term Term
|
data CommandArguments = Exprs [(Expr,Float)] | Strings [String] | Term Term
|
||||||
|
|
||||||
newtype CommandOutput = Piped (CommandArguments,String) ---- errors, etc
|
newtype CommandOutput = Piped (CommandArguments,String) ---- errors, etc
|
||||||
|
|
||||||
-- ** Converting command output
|
-- ** Converting command output
|
||||||
fromStrings ss = Piped (Strings ss, unlines ss)
|
fromStrings ss = Piped (Strings ss, unlines ss)
|
||||||
fromExprs es = Piped (Exprs es,unlines (map (H.showExpr []) es))
|
fromExprs show_p es = Piped (Exprs es,unlines (map (\(e,p) -> (if show_p then (++) ("["++show p++"] ") else id) (showExpr [] e)) es))
|
||||||
fromString s = Piped (Strings [s], s)
|
fromString s = Piped (Strings [s], s)
|
||||||
pipeWithMessage es msg = Piped (Exprs es,msg)
|
pipeWithMessage es msg = Piped (Exprs es,msg)
|
||||||
pipeMessage msg = Piped (Exprs [],msg)
|
pipeMessage msg = Piped (Exprs [],msg)
|
||||||
pipeExprs es = Piped (Exprs es,[]) -- only used in emptyCommandInfo
|
pipeExprs es = Piped (Exprs es,[]) -- only used in emptyCommandInfo
|
||||||
void = Piped (Exprs [],"")
|
void = Piped (Exprs [],"")
|
||||||
|
|
||||||
stringAsExpr = H.ELit . H.LStr -- should be a pattern macro
|
|
||||||
|
|
||||||
-- ** Converting command input
|
-- ** Converting command input
|
||||||
|
|
||||||
toStrings args =
|
toStrings args =
|
||||||
@@ -61,23 +58,23 @@ toStrings args =
|
|||||||
Exprs es -> zipWith showAsString (True:repeat False) es
|
Exprs es -> zipWith showAsString (True:repeat False) es
|
||||||
Term t -> [render t]
|
Term t -> [render t]
|
||||||
where
|
where
|
||||||
showAsString first t =
|
showAsString first (e,p) =
|
||||||
case t of
|
case unStr e of
|
||||||
H.ELit (H.LStr s) -> s
|
Just s -> s
|
||||||
_ -> ['\n'|not first] ++
|
Nothing -> ['\n'|not first] ++
|
||||||
H.showExpr [] t ---newline needed in other cases than the first
|
showExpr [] e ---newline needed in other cases than the first
|
||||||
|
|
||||||
toExprs args =
|
toExprs args =
|
||||||
case args of
|
case args of
|
||||||
Exprs es -> es
|
Exprs es -> map fst es
|
||||||
Strings ss -> map stringAsExpr ss
|
Strings ss -> map mkStr ss
|
||||||
Term t -> [stringAsExpr (render t)]
|
Term t -> [mkStr (render t)]
|
||||||
|
|
||||||
toTerm args =
|
toTerm args =
|
||||||
case args of
|
case args of
|
||||||
Term t -> t
|
Term t -> t
|
||||||
Strings ss -> string2term $ unwords ss -- hmm
|
Strings ss -> string2term $ unwords ss -- hmm
|
||||||
Exprs es -> string2term $ unwords $ map (H.showExpr []) es -- hmm
|
Exprs es -> string2term $ unwords $ map (showExpr [] . fst) es -- hmm
|
||||||
|
|
||||||
-- ** Creating documentation
|
-- ** Creating documentation
|
||||||
|
|
||||||
|
|||||||
@@ -1,16 +1,12 @@
|
|||||||
{-# LANGUAGE FlexibleInstances, UndecidableInstances #-}
|
{-# LANGUAGE FlexibleInstances, UndecidableInstances #-}
|
||||||
module GF.Command.Commands (
|
module GF.Command.Commands (
|
||||||
PGFEnv,HasPGFEnv(..),pgf,mos,pgfEnv,pgfCommands,
|
HasPGF(..),pgfCommands,
|
||||||
options,flags,
|
options,flags,
|
||||||
) where
|
) where
|
||||||
import Prelude hiding (putStrLn,(<>)) -- GHC 8.4.1 clash with Text.PrettyPrint
|
import Prelude hiding (putStrLn)
|
||||||
|
|
||||||
import PGF
|
import PGF2
|
||||||
|
import PGF2.Internal(writePGF)
|
||||||
import PGF.Internal(lookStartCat,functionsToCat,lookValCat,restrictPGF,hasLin)
|
|
||||||
import PGF.Internal(abstract,funs,cats,Expr(EFun)) ----
|
|
||||||
import PGF.Internal(ppFun,ppCat)
|
|
||||||
import PGF.Internal(optimizePGF)
|
|
||||||
|
|
||||||
import GF.Compile.Export
|
import GF.Compile.Export
|
||||||
import GF.Compile.ToAPI
|
import GF.Compile.ToAPI
|
||||||
@@ -28,27 +24,25 @@ import GF.Command.TreeOperations ---- temporary place for typecheck and compute
|
|||||||
|
|
||||||
import GF.Data.Operations
|
import GF.Data.Operations
|
||||||
|
|
||||||
import PGF.Internal (encodeFile)
|
import Data.Char
|
||||||
import Data.List(intersperse,nub)
|
import Data.List(intersperse,nub)
|
||||||
import Data.Maybe
|
import Data.Maybe
|
||||||
import qualified Data.Map as Map
|
import qualified Data.Map as Map
|
||||||
import GF.Text.Pretty
|
import GF.Text.Pretty
|
||||||
import Data.List (sort)
|
import Data.List (sort)
|
||||||
--import Debug.Trace
|
import Control.Monad(mplus)
|
||||||
|
|
||||||
|
class (Functor m,Monad m,MonadSIO m) => HasPGF m where getPGF :: m (Maybe PGF)
|
||||||
|
|
||||||
data PGFEnv = Env {pgf::PGF,mos::Map.Map Language Morpho}
|
instance (Monad m,HasPGF m) => TypeCheckArg m where
|
||||||
|
typeCheckArg e = do mb_pgf <- getPGF
|
||||||
|
case mb_pgf of
|
||||||
|
Just pgf -> either fail
|
||||||
|
(return . fst)
|
||||||
|
(inferExpr pgf e)
|
||||||
|
Nothing -> fail "Import a grammar before using this command"
|
||||||
|
|
||||||
pgfEnv pgf = Env pgf mos
|
pgfCommands :: HasPGF m => Map.Map String (CommandInfo m)
|
||||||
where mos = Map.fromList [(la,buildMorpho pgf la) | la <- languages pgf]
|
|
||||||
|
|
||||||
class (Functor m,Monad m,MonadSIO m) => HasPGFEnv m where getPGFEnv :: m PGFEnv
|
|
||||||
|
|
||||||
instance (Monad m,HasPGFEnv m) => TypeCheckArg m where
|
|
||||||
typeCheckArg e = (either (fail . render . ppTcError) (return . fst)
|
|
||||||
. flip inferExpr e . pgf) =<< getPGFEnv
|
|
||||||
|
|
||||||
pgfCommands :: HasPGFEnv m => Map.Map String (CommandInfo m)
|
|
||||||
pgfCommands = Map.fromList [
|
pgfCommands = Map.fromList [
|
||||||
("aw", emptyCommandInfo {
|
("aw", emptyCommandInfo {
|
||||||
longname = "align_words",
|
longname = "align_words",
|
||||||
@@ -61,7 +55,7 @@ pgfCommands = Map.fromList [
|
|||||||
"by the view flag. The target format is png, unless overridden by the",
|
"by the view flag. The target format is png, unless overridden by the",
|
||||||
"flag -format. Results from multiple trees are combined to pdf with convert (ImageMagick)."
|
"flag -format. Results from multiple trees are combined to pdf with convert (ImageMagick)."
|
||||||
],
|
],
|
||||||
exec = getEnv $ \ opts arg (Env pgf mos) -> do
|
exec = needPGF $ \ opts arg pgf -> do
|
||||||
let es = toExprs arg
|
let es = toExprs arg
|
||||||
let langs = optLangs pgf opts
|
let langs = optLangs pgf opts
|
||||||
if isOpt "giza" opts
|
if isOpt "giza" opts
|
||||||
@@ -73,7 +67,7 @@ pgfCommands = Map.fromList [
|
|||||||
let grph = if null es then [] else lsrc ++ "\n--end_source--\n\n"++ltrg++"\n-end_target--\n\n"++align
|
let grph = if null es then [] else lsrc ++ "\n--end_source--\n\n"++ltrg++"\n-end_target--\n\n"++align
|
||||||
return $ fromString grph
|
return $ fromString grph
|
||||||
else do
|
else do
|
||||||
let grphs = map (graphvizAlignment pgf langs) es
|
let grphs = map (graphvizWordAlignment langs graphvizDefaults) es
|
||||||
if isFlag "view" opts || isFlag "format" opts
|
if isFlag "view" opts || isFlag "format" opts
|
||||||
then do
|
then do
|
||||||
let view = optViewGraph opts
|
let view = optViewGraph opts
|
||||||
@@ -95,6 +89,7 @@ pgfCommands = Map.fromList [
|
|||||||
("view", "program to open the resulting file")
|
("view", "program to open the resulting file")
|
||||||
]
|
]
|
||||||
}),
|
}),
|
||||||
|
|
||||||
("ca", emptyCommandInfo {
|
("ca", emptyCommandInfo {
|
||||||
longname = "clitic_analyse",
|
longname = "clitic_analyse",
|
||||||
synopsis = "print the analyses of all words into stems and clitics",
|
synopsis = "print the analyses of all words into stems and clitics",
|
||||||
@@ -105,16 +100,17 @@ pgfCommands = Map.fromList [
|
|||||||
"by the flag '-clitics'. The list of stems is given as the list of words",
|
"by the flag '-clitics'. The list of stems is given as the list of words",
|
||||||
"of the language given by the '-lang' flag."
|
"of the language given by the '-lang' flag."
|
||||||
],
|
],
|
||||||
exec = getEnv $ \opts ts env -> case opts of
|
exec = needPGF $ \opts ts pgf -> do
|
||||||
_ | isOpt "raw" opts ->
|
concr <- optLang pgf opts
|
||||||
return . fromString .
|
case opts of
|
||||||
unlines . map (unwords . map (concat . intersperse "+")) .
|
_ | isOpt "raw" opts ->
|
||||||
map (getClitics (isInMorpho (optMorpho env opts)) (optClitics opts)) .
|
return . fromString .
|
||||||
concatMap words $ toStrings ts
|
unlines . map (unwords . map (concat . intersperse "+")) .
|
||||||
_ ->
|
map (getClitics (not . null . lookupMorpho concr) (optClitics opts)) .
|
||||||
return . fromStrings .
|
concatMap words $ toStrings ts
|
||||||
getCliticsText (isInMorpho (optMorpho env opts)) (optClitics opts) .
|
_ -> return . fromStrings .
|
||||||
concatMap words $ toStrings ts,
|
getCliticsText (not . null . lookupMorpho concr) (optClitics opts) .
|
||||||
|
concatMap words $ toStrings ts,
|
||||||
flags = [
|
flags = [
|
||||||
("clitics","the list of possible clitics (comma-separated, no spaces)"),
|
("clitics","the list of possible clitics (comma-separated, no spaces)"),
|
||||||
("lang", "the language of analysis")
|
("lang", "the language of analysis")
|
||||||
@@ -146,19 +142,19 @@ pgfCommands = Map.fromList [
|
|||||||
],
|
],
|
||||||
flags = [
|
flags = [
|
||||||
("file","the file to be converted (suffix .gfe must be given)"),
|
("file","the file to be converted (suffix .gfe must be given)"),
|
||||||
("lang","the language in which to parse"),
|
("lang","the language in which to parse")
|
||||||
("probs","file with probabilities to rank the parses")
|
|
||||||
],
|
],
|
||||||
exec = getEnv $ \ opts _ env@(Env pgf mos) -> do
|
exec = needPGF $ \opts _ pgf -> do
|
||||||
let file = optFile opts
|
let file = optFile opts
|
||||||
pgf <- optProbs opts pgf
|
|
||||||
let printer = if (isOpt "api" opts) then exprToAPI else (showExpr [])
|
let printer = if (isOpt "api" opts) then exprToAPI else (showExpr [])
|
||||||
let conf = configureExBased pgf (optMorpho env opts) (optLang pgf opts) printer
|
concr <- optLang pgf opts
|
||||||
|
let conf = configureExBased pgf concr printer
|
||||||
(file',ws) <- restricted $ parseExamplesInGrammar conf file
|
(file',ws) <- restricted $ parseExamplesInGrammar conf file
|
||||||
if null ws then return () else putStrLn ("unknown words: " ++ unwords ws)
|
if null ws then return () else putStrLn ("unknown words: " ++ unwords ws)
|
||||||
return (fromString ("wrote " ++ file')),
|
return (fromString ("wrote " ++ file')),
|
||||||
needsTypeCheck = False
|
needsTypeCheck = False
|
||||||
}),
|
}),
|
||||||
|
|
||||||
("gr", emptyCommandInfo {
|
("gr", emptyCommandInfo {
|
||||||
longname = "generate_random",
|
longname = "generate_random",
|
||||||
synopsis = "generate random trees in the current abstract syntax",
|
synopsis = "generate random trees in the current abstract syntax",
|
||||||
@@ -173,54 +169,53 @@ pgfCommands = Map.fromList [
|
|||||||
explanation = unlines [
|
explanation = unlines [
|
||||||
"Generates a list of random trees, by default one tree.",
|
"Generates a list of random trees, by default one tree.",
|
||||||
"If a tree argument is given, the command completes the Tree with values to",
|
"If a tree argument is given, the command completes the Tree with values to",
|
||||||
"all metavariables in the tree. The generation can be biased by probabilities,",
|
"all metavariables in the tree. The generation can be biased by probabilities",
|
||||||
"given in a file in the -probs flag."
|
"if the grammar was compiled with option -probs"
|
||||||
|
],
|
||||||
|
options = [
|
||||||
|
("show_probs", "show the probability of each result")
|
||||||
],
|
],
|
||||||
flags = [
|
flags = [
|
||||||
("cat","generation category"),
|
("cat","generation category"),
|
||||||
("lang","uses only functions that have linearizations in all these languages"),
|
("lang","uses only functions that have linearizations in all these languages"),
|
||||||
("number","number of trees generated"),
|
("number","number of trees generated")
|
||||||
("depth","the maximum generation depth"),
|
|
||||||
("probs", "file with biased probabilities (format 'f 0.4' one by line)")
|
|
||||||
],
|
],
|
||||||
exec = getEnv $ \ opts arg (Env pgf mos) -> do
|
exec = needPGF $ \opts arg pgf -> do
|
||||||
pgf <- optProbs opts (optRestricted opts pgf)
|
|
||||||
gen <- newStdGen
|
gen <- newStdGen
|
||||||
let dp = valIntOpts "depth" 4 opts
|
|
||||||
let ts = case mexp (toExprs arg) of
|
let ts = case mexp (toExprs arg) of
|
||||||
Just ex -> generateRandomFromDepth gen pgf ex (Just dp)
|
Just ex -> generateRandomFrom gen pgf ex
|
||||||
Nothing -> generateRandomDepth gen pgf (optType pgf opts) (Just dp)
|
Nothing -> generateRandom gen pgf (optType pgf opts)
|
||||||
returnFromExprs $ take (optNum opts) ts
|
returnFromExprs (isOpt "show_probs" opts) $ take (optNum opts) ts
|
||||||
}),
|
}),
|
||||||
|
|
||||||
("gt", emptyCommandInfo {
|
("gt", emptyCommandInfo {
|
||||||
longname = "generate_trees",
|
longname = "generate_trees",
|
||||||
synopsis = "generates a list of trees, by default exhaustive",
|
synopsis = "generates a list of trees, by default exhaustive",
|
||||||
explanation = unlines [
|
explanation = unlines [
|
||||||
"Generates all trees of a given category. By default, ",
|
"Generates all trees of a given category.",
|
||||||
"the depth is limited to 4, but this can be changed by a flag.",
|
|
||||||
"If a Tree argument is given, the command completes the Tree with values",
|
"If a Tree argument is given, the command completes the Tree with values",
|
||||||
"to all metavariables in the tree."
|
"to all metavariables in the tree."
|
||||||
],
|
],
|
||||||
|
options = [
|
||||||
|
("show_probs", "show the probability of each result")
|
||||||
|
],
|
||||||
flags = [
|
flags = [
|
||||||
("cat","the generation category"),
|
("cat","the generation category"),
|
||||||
("depth","the maximum generation depth"),
|
|
||||||
("lang","excludes functions that have no linearization in this language"),
|
("lang","excludes functions that have no linearization in this language"),
|
||||||
("number","the number of trees generated")
|
("number","the number of trees generated")
|
||||||
],
|
],
|
||||||
examples = [
|
examples = [
|
||||||
mkEx "gt -- all trees in the startcat, to depth 4",
|
mkEx "gt -- all trees in the startcat",
|
||||||
mkEx "gt -cat=NP -number=16 -- 16 trees in the category NP",
|
mkEx "gt -cat=NP -number=16 -- 16 trees in the category NP",
|
||||||
mkEx "gt -cat=NP -depth=2 -- trees in the category NP to depth 2",
|
|
||||||
mkEx "gt (AdjCN ? (UseN ?)) -- trees of form (AdjCN ? (UseN ?))"
|
mkEx "gt (AdjCN ? (UseN ?)) -- trees of form (AdjCN ? (UseN ?))"
|
||||||
],
|
],
|
||||||
exec = getEnv $ \ opts arg (Env pgf mos) -> do
|
exec = needPGF $ \opts arg pgf -> do
|
||||||
let pgfr = optRestricted opts pgf
|
let es = case mexp (toExprs arg) of
|
||||||
let dp = valIntOpts "depth" 4 opts
|
Just ex -> generateAllFrom pgf ex
|
||||||
let ts = case mexp (toExprs arg) of
|
Nothing -> generateAll pgf (optType pgf opts)
|
||||||
Just ex -> generateFromDepth pgfr ex (Just dp)
|
returnFromExprs (isOpt "show_probs" opts) $ takeOptNum opts es
|
||||||
Nothing -> generateAllDepth pgfr (optType pgf opts) (Just dp)
|
|
||||||
returnFromExprs $ take (optNumInf opts) ts
|
|
||||||
}),
|
}),
|
||||||
|
|
||||||
("i", emptyCommandInfo {
|
("i", emptyCommandInfo {
|
||||||
longname = "import",
|
longname = "import",
|
||||||
synopsis = "import a grammar from source code or compiled .pgf file",
|
synopsis = "import a grammar from source code or compiled .pgf file",
|
||||||
@@ -241,33 +236,28 @@ pgfCommands = Map.fromList [
|
|||||||
("probs","file with biased probabilities for generation")
|
("probs","file with biased probabilities for generation")
|
||||||
],
|
],
|
||||||
options = [
|
options = [
|
||||||
-- ["gfo", "src", "no-cpu", "cpu", "quiet", "verbose"]
|
|
||||||
("retain","retain operations (used for cc command)"),
|
("retain","retain operations (used for cc command)"),
|
||||||
("src", "force compilation from source"),
|
("src", "force compilation from source"),
|
||||||
("v", "be verbose - show intermediate status information")
|
("v", "be verbose - show intermediate status information")
|
||||||
],
|
],
|
||||||
needsTypeCheck = False
|
needsTypeCheck = False
|
||||||
}),
|
}),
|
||||||
|
|
||||||
("l", emptyCommandInfo {
|
("l", emptyCommandInfo {
|
||||||
longname = "linearize",
|
longname = "linearize",
|
||||||
synopsis = "convert an abstract syntax expression to string",
|
synopsis = "convert an abstract syntax expression to string",
|
||||||
explanation = unlines [
|
explanation = unlines [
|
||||||
"Shows the linearization of a Tree by the grammars in scope.",
|
"Shows the linearization of a tree by the grammars in scope.",
|
||||||
"The -lang flag can be used to restrict this to fewer languages.",
|
"The -lang flag can be used to restrict this to fewer languages.",
|
||||||
"A sequence of string operations (see command ps) can be given",
|
"A sequence of string operations (see command ps) can be given",
|
||||||
"as options, and works then like a pipe to the ps command, except",
|
"as options, and works then like a pipe to the ps command, except",
|
||||||
"that it only affect the strings, not e.g. the table labels.",
|
"that it only affect the strings, not e.g. the table labels."
|
||||||
"These can be given separately to each language with the unlexer flag",
|
|
||||||
"whose results are prepended to the other lexer flags. The value of the",
|
|
||||||
"unlexer flag is a space-separated list of comma-separated string operation",
|
|
||||||
"sequences; see example."
|
|
||||||
],
|
],
|
||||||
examples = [
|
examples = [
|
||||||
mkEx "l -lang=LangSwe,LangNor no_Utt -- linearize tree to LangSwe and LangNor",
|
mkEx "l -lang=LangSwe,LangNor no_Utt -- linearize tree to LangSwe and LangNor",
|
||||||
mkEx "gr -lang=LangHin -cat=Cl | l -table -to_devanagari -- hindi table",
|
mkEx "gr -lang=LangHin -cat=Cl | l -table -to_devanagari -- hindi table"
|
||||||
mkEx "l -unlexer=\"LangAra=to_arabic LangHin=to_devanagari\" -- different unlexers"
|
|
||||||
],
|
],
|
||||||
exec = getEnv $ \ opts ts (Env pgf mos) -> return . fromStrings . optLins pgf opts $ toExprs ts,
|
exec = needPGF $ \ opts ts pgf -> return . fromStrings . optLins pgf opts $ toExprs ts,
|
||||||
options = [
|
options = [
|
||||||
("all", "show all forms and variants, one by line (cf. l -list)"),
|
("all", "show all forms and variants, one by line (cf. l -list)"),
|
||||||
("bracket","show tree structure with brackets and paths to nodes"),
|
("bracket","show tree structure with brackets and paths to nodes"),
|
||||||
@@ -275,33 +265,13 @@ pgfCommands = Map.fromList [
|
|||||||
("list","show all forms and variants, comma-separated on one line (cf. l -all)"),
|
("list","show all forms and variants, comma-separated on one line (cf. l -all)"),
|
||||||
("multi","linearize to all languages (default)"),
|
("multi","linearize to all languages (default)"),
|
||||||
("table","show all forms labelled by parameters"),
|
("table","show all forms labelled by parameters"),
|
||||||
("tabtreebank","show the tree and its linearizations on a tab-separated line"),
|
|
||||||
("treebank","show the tree and tag linearizations with language names")
|
|
||||||
] ++ stringOpOptions,
|
|
||||||
flags = [
|
|
||||||
("lang","the languages of linearization (comma-separated, no spaces)"),
|
|
||||||
("unlexer","set unlexers separately to each language (space-separated)")
|
|
||||||
]
|
|
||||||
}),
|
|
||||||
("lc", emptyCommandInfo {
|
|
||||||
longname = "linearize_chunks",
|
|
||||||
synopsis = "linearize a tree that has metavariables in maximal chunks without them",
|
|
||||||
explanation = unlines [
|
|
||||||
"A hopefully temporary command, intended to work around the type checker that fails",
|
|
||||||
"trees where a function node is a metavariable."
|
|
||||||
],
|
|
||||||
examples = [
|
|
||||||
mkEx "l -lang=LangSwe,LangNor -chunks ? a b (? c d)"
|
|
||||||
],
|
|
||||||
exec = getEnv $ \ opts ts (Env pgf mos) -> return . fromStrings $ optLins pgf (opts ++ [OOpt "chunks"]) (toExprs ts),
|
|
||||||
options = [
|
|
||||||
("treebank","show the tree and tag linearizations with language names")
|
("treebank","show the tree and tag linearizations with language names")
|
||||||
] ++ stringOpOptions,
|
] ++ stringOpOptions,
|
||||||
flags = [
|
flags = [
|
||||||
("lang","the languages of linearization (comma-separated, no spaces)")
|
("lang","the languages of linearization (comma-separated, no spaces)")
|
||||||
],
|
]
|
||||||
needsTypeCheck = False
|
|
||||||
}),
|
}),
|
||||||
|
|
||||||
("ma", emptyCommandInfo {
|
("ma", emptyCommandInfo {
|
||||||
longname = "morpho_analyse",
|
longname = "morpho_analyse",
|
||||||
synopsis = "print the morphological analyses of all words in the string",
|
synopsis = "print the morphological analyses of all words in the string",
|
||||||
@@ -309,18 +279,20 @@ pgfCommands = Map.fromList [
|
|||||||
"Prints all the analyses of space-separated words in the input string,",
|
"Prints all the analyses of space-separated words in the input string,",
|
||||||
"using the morphological analyser of the actual grammar (see command pg)"
|
"using the morphological analyser of the actual grammar (see command pg)"
|
||||||
],
|
],
|
||||||
exec = getEnv $ \opts ts env -> case opts of
|
exec = needPGF $ \opts ts pgf -> do
|
||||||
_ | isOpt "missing" opts ->
|
concr <- optLang pgf opts
|
||||||
return . fromString . unwords .
|
case opts of
|
||||||
morphoMissing (optMorpho env opts) .
|
_ | isOpt "missing" opts ->
|
||||||
concatMap words $ toStrings ts
|
return . fromString . unwords .
|
||||||
_ | isOpt "known" opts ->
|
morphoMissing concr .
|
||||||
return . fromString . unwords .
|
concatMap words $ toStrings ts
|
||||||
morphoKnown (optMorpho env opts) .
|
_ | isOpt "known" opts ->
|
||||||
concatMap words $ toStrings ts
|
return . fromString . unwords .
|
||||||
_ -> return . fromString . unlines .
|
morphoKnown concr .
|
||||||
map prMorphoAnalysis . concatMap (morphos env opts) .
|
concatMap words $ toStrings ts
|
||||||
concatMap words $ toStrings ts,
|
_ -> return . fromString . unlines .
|
||||||
|
map prMorphoAnalysis . concatMap (morphos pgf opts) .
|
||||||
|
concatMap words $ toStrings ts,
|
||||||
flags = [
|
flags = [
|
||||||
("lang","the languages of analysis (comma-separated, no spaces)")
|
("lang","the languages of analysis (comma-separated, no spaces)")
|
||||||
],
|
],
|
||||||
@@ -334,18 +306,16 @@ pgfCommands = Map.fromList [
|
|||||||
longname = "morpho_quiz",
|
longname = "morpho_quiz",
|
||||||
synopsis = "start a morphology quiz",
|
synopsis = "start a morphology quiz",
|
||||||
syntax = "mq (-cat=CAT)? (-probs=FILE)? TREE?",
|
syntax = "mq (-cat=CAT)? (-probs=FILE)? TREE?",
|
||||||
exec = getEnv $ \ opts arg (Env pgf mos) -> do
|
exec = needPGF $ \ opts arg pgf -> do
|
||||||
let lang = optLang pgf opts
|
lang <- optLang pgf opts
|
||||||
let typ = optType pgf opts
|
let typ = optType pgf opts
|
||||||
pgf <- optProbs opts pgf
|
|
||||||
let mt = mexp (toExprs arg)
|
let mt = mexp (toExprs arg)
|
||||||
restricted $ morphologyQuiz mt pgf lang typ
|
restricted $ morphologyQuiz mt pgf lang typ
|
||||||
return void,
|
return void,
|
||||||
flags = [
|
flags = [
|
||||||
("lang","language of the quiz"),
|
("lang","language of the quiz"),
|
||||||
("cat","category of the quiz"),
|
("cat","category of the quiz"),
|
||||||
("number","maximum number of questions"),
|
("number","maximum number of questions")
|
||||||
("probs","file with biased probabilities for generation")
|
|
||||||
]
|
]
|
||||||
}),
|
}),
|
||||||
|
|
||||||
@@ -356,24 +326,25 @@ pgfCommands = Map.fromList [
|
|||||||
"Shows all trees returned by parsing a string in the grammars in scope.",
|
"Shows all trees returned by parsing a string in the grammars in scope.",
|
||||||
"The -lang flag can be used to restrict this to fewer languages.",
|
"The -lang flag can be used to restrict this to fewer languages.",
|
||||||
"The default start category can be overridden by the -cat flag.",
|
"The default start category can be overridden by the -cat flag.",
|
||||||
"See also the ps command for lexing and character encoding.",
|
"See also the ps command for lexing and character encoding."
|
||||||
"",
|
],
|
||||||
"The -openclass flag is experimental and allows some robustness in ",
|
exec = needPGF $ \opts ts pgf ->
|
||||||
"the parser. For example if -openclass=\"A,N,V\" is given, the parser",
|
return $
|
||||||
"will accept unknown adjectives, nouns and verbs with the resource grammar."
|
foldr (joinPiped . fromParse1 opts) void
|
||||||
|
(concat [
|
||||||
|
[(s,parse concr (optType pgf opts) s) |
|
||||||
|
concr <- optLangs pgf opts]
|
||||||
|
| s <- toStrings ts]),
|
||||||
|
options = [
|
||||||
|
("show_probs", "show the probability of each result")
|
||||||
],
|
],
|
||||||
exec = getEnv $ \ opts ts (Env pgf mos) ->
|
|
||||||
return $ fromParse opts (concat [map ((,) s) (par pgf opts s) | s <- toStrings ts]),
|
|
||||||
flags = [
|
flags = [
|
||||||
("cat","target category of parsing"),
|
("cat","target category of parsing"),
|
||||||
("lang","the languages of parsing (comma-separated, no spaces)"),
|
("lang","the languages of parsing (comma-separated, no spaces)"),
|
||||||
("openclass","list of open-class categories for robust parsing"),
|
("number","limit the results to the top N trees")
|
||||||
("depth","maximal depth for proof search if the abstract syntax tree has meta variables")
|
|
||||||
],
|
|
||||||
options = [
|
|
||||||
("bracket","prints the bracketed string from the parser")
|
|
||||||
]
|
]
|
||||||
}),
|
}),
|
||||||
|
|
||||||
("pg", emptyCommandInfo { -----
|
("pg", emptyCommandInfo { -----
|
||||||
longname = "print_grammar",
|
longname = "print_grammar",
|
||||||
synopsis = "print the actual grammar with the given printer",
|
synopsis = "print the actual grammar with the given printer",
|
||||||
@@ -393,9 +364,8 @@ pgfCommands = Map.fromList [
|
|||||||
" " ++ opt ++ "\t\t" ++ expl |
|
" " ++ opt ++ "\t\t" ++ expl |
|
||||||
((opt,_),expl) <- outputFormatsExpl, take 1 expl /= "*"
|
((opt,_),expl) <- outputFormatsExpl, take 1 expl /= "*"
|
||||||
]),
|
]),
|
||||||
exec = getEnv $ \opts _ env -> prGrammar env opts,
|
exec = needPGF $ \opts _ pgf -> prGrammar pgf opts,
|
||||||
flags = [
|
flags = [
|
||||||
--"cat",
|
|
||||||
("file", "set the file name when printing with -pgf option"),
|
("file", "set the file name when printing with -pgf option"),
|
||||||
("lang", "select languages for the some options (default all languages)"),
|
("lang", "select languages for the some options (default all languages)"),
|
||||||
("printer","select the printing format (see flag values above)")
|
("printer","select the printing format (see flag values above)")
|
||||||
@@ -415,6 +385,7 @@ pgfCommands = Map.fromList [
|
|||||||
mkEx ("pg -funs | ? grep \" S ;\" -- show functions with value cat S")
|
mkEx ("pg -funs | ? grep \" S ;\" -- show functions with value cat S")
|
||||||
]
|
]
|
||||||
}),
|
}),
|
||||||
|
|
||||||
("pt", emptyCommandInfo {
|
("pt", emptyCommandInfo {
|
||||||
longname = "put_tree",
|
longname = "put_tree",
|
||||||
syntax = "pt OPT? TREE",
|
syntax = "pt OPT? TREE",
|
||||||
@@ -428,11 +399,12 @@ pgfCommands = Map.fromList [
|
|||||||
examples = [
|
examples = [
|
||||||
mkEx "pt -compute (plus one two) -- compute value"
|
mkEx "pt -compute (plus one two) -- compute value"
|
||||||
],
|
],
|
||||||
exec = getEnv $ \ opts arg (Env pgf mos) ->
|
exec = needPGF $ \opts arg pgf ->
|
||||||
returnFromExprs . takeOptNum opts . treeOps pgf opts $ toExprs arg,
|
returnFromExprs False . takeOptNum opts . map (flip (,) 0) . treeOps pgf opts $ toExprs arg,
|
||||||
options = treeOpOptions undefined{-pgf-},
|
options = treeOpOptions undefined{-pgf-},
|
||||||
flags = [("number","take at most this many trees")] ++ treeOpFlags undefined{-pgf-}
|
flags = [("number","take at most this many trees")] ++ treeOpFlags undefined{-pgf-}
|
||||||
}),
|
}),
|
||||||
|
|
||||||
("rf", emptyCommandInfo {
|
("rf", emptyCommandInfo {
|
||||||
longname = "read_file",
|
longname = "read_file",
|
||||||
synopsis = "read string or tree input from a file",
|
synopsis = "read string or tree input from a file",
|
||||||
@@ -445,10 +417,9 @@ pgfCommands = Map.fromList [
|
|||||||
],
|
],
|
||||||
options = [
|
options = [
|
||||||
("lines","return the list of lines, instead of the singleton of all contents"),
|
("lines","return the list of lines, instead of the singleton of all contents"),
|
||||||
("paragraphs","return the list of paragraphs, as separated by empty lines"),
|
|
||||||
("tree","convert strings into trees")
|
("tree","convert strings into trees")
|
||||||
],
|
],
|
||||||
exec = getEnv $ \ opts _ (Env pgf mos) -> do
|
exec = needPGF $ \ opts _ pgf -> do
|
||||||
let file = valStrOpts "file" "_gftmp" opts
|
let file = valStrOpts "file" "_gftmp" opts
|
||||||
let exprs [] = ([],empty)
|
let exprs [] = ([],empty)
|
||||||
exprs ((n,s):ls) | null s
|
exprs ((n,s):ls) | null s
|
||||||
@@ -457,12 +428,12 @@ pgfCommands = Map.fromList [
|
|||||||
Just e -> let (es,err) = exprs ls
|
Just e -> let (es,err) = exprs ls
|
||||||
in case inferExpr pgf e of
|
in case inferExpr pgf e of
|
||||||
Right (e,t) -> (e:es,err)
|
Right (e,t) -> (e:es,err)
|
||||||
Left tcerr -> (es,"on line" <+> n <> ':' $$ nest 2 (ppTcError tcerr) $$ err)
|
Left err -> (es,"on line" <+> n <> ':' $$ nest 2 err $$ err)
|
||||||
Nothing -> let (es,err) = exprs ls
|
Nothing -> let (es,err) = exprs ls
|
||||||
in (es,"on line" <+> n <> ':' <+> "parse error" $$ err)
|
in (es,"on line" <+> n <> ':' <+> "parse error" $$ err)
|
||||||
returnFromLines ls = case exprs ls of
|
returnFromLines ls = case exprs ls of
|
||||||
(es, err) | null es -> return $ pipeMessage $ render (err $$ "no trees found")
|
(es, err) | null es -> return $ pipeMessage $ render (err $$ "no trees found")
|
||||||
| otherwise -> return $ pipeWithMessage es (render err)
|
| otherwise -> return $ pipeWithMessage (map (flip (,) 0) es) (render err)
|
||||||
|
|
||||||
s <- restricted $ readFile file
|
s <- restricted $ readFile file
|
||||||
case opts of
|
case opts of
|
||||||
@@ -471,56 +442,26 @@ pgfCommands = Map.fromList [
|
|||||||
_ | isOpt "tree" opts ->
|
_ | isOpt "tree" opts ->
|
||||||
returnFromLines [(1::Int,s)]
|
returnFromLines [(1::Int,s)]
|
||||||
_ | isOpt "lines" opts -> return (fromStrings $ lines s)
|
_ | isOpt "lines" opts -> return (fromStrings $ lines s)
|
||||||
_ | isOpt "paragraphs" opts -> return (fromStrings $ toParagraphs $ lines s)
|
|
||||||
_ -> return (fromString s),
|
_ -> return (fromString s),
|
||||||
flags = [("file","the input file name")]
|
flags = [("file","the input file name")]
|
||||||
}),
|
}),
|
||||||
("rt", emptyCommandInfo {
|
|
||||||
longname = "rank_trees",
|
|
||||||
synopsis = "show trees in an order of decreasing probability",
|
|
||||||
explanation = unlines [
|
|
||||||
"Order trees from the most to the least probable, using either",
|
|
||||||
"even distribution in each category (default) or biased as specified",
|
|
||||||
"by the file given by flag -probs=FILE, where each line has the form",
|
|
||||||
"'function probability', e.g. 'youPol_Pron 0.01'."
|
|
||||||
],
|
|
||||||
exec = getEnv $ \ opts arg (Env pgf mos) -> do
|
|
||||||
let ts = toExprs arg
|
|
||||||
pgf <- optProbs opts pgf
|
|
||||||
let tds = rankTreesByProbs pgf ts
|
|
||||||
if isOpt "v" opts
|
|
||||||
then putStrLn $
|
|
||||||
unlines [showExpr [] t ++ "\t--" ++ show d | (t,d) <- tds]
|
|
||||||
else return ()
|
|
||||||
returnFromExprs $ map fst tds,
|
|
||||||
flags = [
|
|
||||||
("probs","probabilities from this file (format 'f 0.6' per line)")
|
|
||||||
],
|
|
||||||
options = [
|
|
||||||
("v","show all trees with their probability scores")
|
|
||||||
],
|
|
||||||
examples = [
|
|
||||||
mkEx "p \"you are here\" | rt -probs=probs | pt -number=1 -- most probable result"
|
|
||||||
]
|
|
||||||
}),
|
|
||||||
("tq", emptyCommandInfo {
|
("tq", emptyCommandInfo {
|
||||||
longname = "translation_quiz",
|
longname = "translation_quiz",
|
||||||
syntax = "tq -from=LANG -to=LANG (-cat=CAT)? (-probs=FILE)? TREE?",
|
syntax = "tq -from=LANG -to=LANG (-cat=CAT)? (-probs=FILE)? TREE?",
|
||||||
synopsis = "start a translation quiz",
|
synopsis = "start a translation quiz",
|
||||||
exec = getEnv $ \ opts arg (Env pgf mos) -> do
|
exec = needPGF $ \ opts arg pgf -> do
|
||||||
let from = optLangFlag "from" pgf opts
|
from <- optLangFlag "from" pgf opts
|
||||||
let to = optLangFlag "to" pgf opts
|
to <- optLangFlag "to" pgf opts
|
||||||
let typ = optType pgf opts
|
let typ = optType pgf opts
|
||||||
let mt = mexp (toExprs arg)
|
let mt = mexp (toExprs arg)
|
||||||
pgf <- optProbs opts pgf
|
|
||||||
restricted $ translationQuiz mt pgf from to typ
|
restricted $ translationQuiz mt pgf from to typ
|
||||||
return void,
|
return void,
|
||||||
flags = [
|
flags = [
|
||||||
("from","translate from this language"),
|
("from","translate from this language"),
|
||||||
("to","translate to this language"),
|
("to","translate to this language"),
|
||||||
("cat","translate in this category"),
|
("cat","translate in this category"),
|
||||||
("number","the maximum number of questions"),
|
("number","the maximum number of questions")
|
||||||
("probs","file with biased probabilities for generation")
|
|
||||||
],
|
],
|
||||||
examples = [
|
examples = [
|
||||||
mkEx ("tq -from=Eng -to=Swe -- any trees in startcat"),
|
mkEx ("tq -from=Eng -to=Swe -- any trees in startcat"),
|
||||||
@@ -528,7 +469,6 @@ pgfCommands = Map.fromList [
|
|||||||
]
|
]
|
||||||
}),
|
}),
|
||||||
|
|
||||||
|
|
||||||
("vd", emptyCommandInfo {
|
("vd", emptyCommandInfo {
|
||||||
longname = "visualize_dependency",
|
longname = "visualize_dependency",
|
||||||
synopsis = "show word dependency tree graphically",
|
synopsis = "show word dependency tree graphically",
|
||||||
@@ -546,7 +486,7 @@ pgfCommands = Map.fromList [
|
|||||||
"flag -format. Results from multiple trees are combined to pdf with convert (ImageMagick).",
|
"flag -format. Results from multiple trees are combined to pdf with convert (ImageMagick).",
|
||||||
"See also 'vp -showdep' for another visualization of dependencies."
|
"See also 'vp -showdep' for another visualization of dependencies."
|
||||||
],
|
],
|
||||||
exec = getEnv $ \ opts arg (Env pgf mos) -> do
|
exec = needPGF $ \ opts arg pgf -> do
|
||||||
let absname = abstractName pgf
|
let absname = abstractName pgf
|
||||||
let es = toExprs arg
|
let es = toExprs arg
|
||||||
let debug = isOpt "v" opts
|
let debug = isOpt "v" opts
|
||||||
@@ -559,8 +499,8 @@ pgfCommands = Map.fromList [
|
|||||||
mclab <- case cnclabels of
|
mclab <- case cnclabels of
|
||||||
"" -> return Nothing
|
"" -> return Nothing
|
||||||
_ -> (Just . getCncDepLabels) `fmap` restricted (readFile cnclabels)
|
_ -> (Just . getCncDepLabels) `fmap` restricted (readFile cnclabels)
|
||||||
let lang = optLang pgf opts
|
concr <- optLang pgf opts
|
||||||
let grphs = map (graphvizDependencyTree outp debug mlab mclab pgf lang) es
|
let grphs = map (graphvizDependencyTree outp debug mlab mclab concr) es
|
||||||
if isOpt "conll2latex" opts
|
if isOpt "conll2latex" opts
|
||||||
then return $ fromString $ conlls2latexDoc $ stanzas $ unlines $ toStrings arg
|
then return $ fromString $ conlls2latexDoc $ stanzas $ unlines $ toStrings arg
|
||||||
else if isFlag "view" opts && valStrOpts "output" "" opts == "latex"
|
else if isFlag "view" opts && valStrOpts "output" "" opts == "latex"
|
||||||
@@ -595,7 +535,6 @@ pgfCommands = Map.fromList [
|
|||||||
]
|
]
|
||||||
}),
|
}),
|
||||||
|
|
||||||
|
|
||||||
("vp", emptyCommandInfo {
|
("vp", emptyCommandInfo {
|
||||||
longname = "visualize_parse",
|
longname = "visualize_parse",
|
||||||
synopsis = "show parse tree graphically",
|
synopsis = "show parse tree graphically",
|
||||||
@@ -607,9 +546,8 @@ pgfCommands = Map.fromList [
|
|||||||
"by the view flag. The target format is png, unless overridden by the",
|
"by the view flag. The target format is png, unless overridden by the",
|
||||||
"flag -format. Results from multiple trees are combined to pdf with convert (ImageMagick)."
|
"flag -format. Results from multiple trees are combined to pdf with convert (ImageMagick)."
|
||||||
],
|
],
|
||||||
exec = getEnv $ \ opts arg (Env pgf mos) -> do
|
exec = needPGF $ \opts arg pgf -> do
|
||||||
let es = toExprs arg
|
let es = toExprs arg
|
||||||
let lang = optLang pgf opts
|
|
||||||
let gvOptions = GraphvizOptions {noLeaves = isOpt "noleaves" opts && not (isOpt "showleaves" opts),
|
let gvOptions = GraphvizOptions {noLeaves = isOpt "noleaves" opts && not (isOpt "showleaves" opts),
|
||||||
noFun = isOpt "nofun" opts || not (isOpt "showfun" opts),
|
noFun = isOpt "nofun" opts || not (isOpt "showfun" opts),
|
||||||
noCat = isOpt "nocat" opts && not (isOpt "showcat" opts),
|
noCat = isOpt "nocat" opts && not (isOpt "showcat" opts),
|
||||||
@@ -622,10 +560,11 @@ pgfCommands = Map.fromList [
|
|||||||
leafEdgeStyle = valStrOpts "leafedgestyle" "dashed" opts
|
leafEdgeStyle = valStrOpts "leafedgestyle" "dashed" opts
|
||||||
}
|
}
|
||||||
let depfile = valStrOpts "file" "" opts
|
let depfile = valStrOpts "file" "" opts
|
||||||
|
concr <- optLang pgf opts
|
||||||
mlab <- case depfile of
|
mlab <- case depfile of
|
||||||
"" -> return Nothing
|
"" -> return Nothing
|
||||||
_ -> (Just . getDepLabels) `fmap` restricted (readFile depfile)
|
_ -> (Just . getDepLabels) `fmap` restricted (readFile depfile)
|
||||||
let grphs = map (graphvizParseTreeDep mlab pgf lang gvOptions) es
|
let grphs = map (graphvizDependencyTree "dot" False mlab Nothing concr) es
|
||||||
if isFlag "view" opts || isFlag "format" opts
|
if isFlag "view" opts || isFlag "format" opts
|
||||||
then do
|
then do
|
||||||
let view = optViewGraph opts
|
let view = optViewGraph opts
|
||||||
@@ -660,7 +599,6 @@ pgfCommands = Map.fromList [
|
|||||||
]
|
]
|
||||||
}),
|
}),
|
||||||
|
|
||||||
|
|
||||||
("vt", emptyCommandInfo {
|
("vt", emptyCommandInfo {
|
||||||
longname = "visualize_tree",
|
longname = "visualize_tree",
|
||||||
synopsis = "show a set of trees graphically",
|
synopsis = "show a set of trees graphically",
|
||||||
@@ -673,7 +611,7 @@ pgfCommands = Map.fromList [
|
|||||||
"flag -format. Results from multiple trees are combined to pdf with convert (ImageMagick).",
|
"flag -format. Results from multiple trees are combined to pdf with convert (ImageMagick).",
|
||||||
"With option -mk, use for showing library style function names of form 'mkC'."
|
"With option -mk, use for showing library style function names of form 'mkC'."
|
||||||
],
|
],
|
||||||
exec = getEnv $ \ opts arg (Env pgf mos) ->
|
exec = needPGF $ \opts arg pgf ->
|
||||||
let es = toExprs arg in
|
let es = toExprs arg in
|
||||||
if isOpt "mk" opts
|
if isOpt "mk" opts
|
||||||
then return $ fromString $ unlines $ map (tree2mk pgf) es
|
then return $ fromString $ unlines $ map (tree2mk pgf) es
|
||||||
@@ -685,7 +623,7 @@ pgfCommands = Map.fromList [
|
|||||||
else do
|
else do
|
||||||
let funs = not (isOpt "nofun" opts)
|
let funs = not (isOpt "nofun" opts)
|
||||||
let cats = not (isOpt "nocat" opts)
|
let cats = not (isOpt "nocat" opts)
|
||||||
let grphs = map (graphvizAbstractTree pgf (funs,cats)) es
|
let grphs = map (graphvizAbstractTree pgf (graphvizDefaults{noFun=funs,noCat=cats})) es
|
||||||
if isFlag "view" opts || isFlag "format" opts
|
if isFlag "view" opts || isFlag "format" opts
|
||||||
then do
|
then do
|
||||||
let view = optViewGraph opts
|
let view = optViewGraph opts
|
||||||
@@ -707,6 +645,7 @@ pgfCommands = Map.fromList [
|
|||||||
("view","program to open the resulting file (default \"open\")")
|
("view","program to open the resulting file (default \"open\")")
|
||||||
]
|
]
|
||||||
}),
|
}),
|
||||||
|
|
||||||
("ai", emptyCommandInfo {
|
("ai", emptyCommandInfo {
|
||||||
longname = "abstract_info",
|
longname = "abstract_info",
|
||||||
syntax = "ai IDENTIFIER or ai EXPR",
|
syntax = "ai IDENTIFIER or ai EXPR",
|
||||||
@@ -719,205 +658,156 @@ pgfCommands = Map.fromList [
|
|||||||
"If a whole expression is given it prints the expression with refined",
|
"If a whole expression is given it prints the expression with refined",
|
||||||
"metavariables and the type of the expression."
|
"metavariables and the type of the expression."
|
||||||
],
|
],
|
||||||
exec = getEnv $ \ opts arg (Env pgf mos) -> do
|
exec = needPGF $ \opts arg pgf -> do
|
||||||
case toExprs arg of
|
case toExprs arg of
|
||||||
[EFun id] -> case Map.lookup id (funs (abstract pgf)) of
|
[e] -> case unApp e of
|
||||||
Just fd -> do putStrLn $ render (ppFun id fd)
|
Just (id, []) -> case functionType pgf id of
|
||||||
let (_,_,_,prob) = fd
|
Just ty -> do putStrLn (showFun pgf id ty)
|
||||||
putStrLn ("Probability: "++show prob)
|
putStrLn ("Probability: "++show (treeProbability pgf e))
|
||||||
return void
|
return void
|
||||||
Nothing -> case Map.lookup id (cats (abstract pgf)) of
|
Nothing -> case categoryContext pgf id of
|
||||||
Just cd -> do putStrLn $
|
Just hypos -> do putStrLn ("cat "++id++if null hypos then "" else ' ':showContext [] hypos)
|
||||||
render (ppCat id cd $$
|
let ls = [showFun pgf fn ty | fn <- functionsByCat pgf id, Just ty <- [functionType pgf fn]]
|
||||||
if null (functionsToCat pgf id)
|
if null ls
|
||||||
then empty
|
then return ()
|
||||||
else ' ' $$
|
else putStrLn (unlines ("":ls))
|
||||||
vcat [ppFun fid (ty,0,Just ([],[]),0) | (fid,ty) <- functionsToCat pgf id] $$
|
putStrLn ("Probability: "++show (categoryProbability pgf id))
|
||||||
' ')
|
return void
|
||||||
let (_,_,prob) = cd
|
Nothing -> do putStrLn ("unknown category of function identifier "++show id)
|
||||||
putStrLn ("Probability: "++show prob)
|
return void
|
||||||
return void
|
_ -> case inferExpr pgf e of
|
||||||
Nothing -> do putStrLn ("unknown category of function identifier "++show id)
|
Left err -> error err
|
||||||
return void
|
Right (e,ty) -> do putStrLn ("Expression: "++showExpr [] e)
|
||||||
[e] -> case inferExpr pgf e of
|
putStrLn ("Type: "++showType [] ty)
|
||||||
Left tcErr -> error $ render (ppTcError tcErr)
|
putStrLn ("Probability: "++show (treeProbability pgf e))
|
||||||
Right (e,ty) -> do putStrLn ("Expression: "++showExpr [] e)
|
return void
|
||||||
putStrLn ("Type: "++showType [] ty)
|
|
||||||
putStrLn ("Probability: "++show (probTree pgf e))
|
|
||||||
return void
|
|
||||||
_ -> do putStrLn "a single identifier or expression is expected from the command"
|
_ -> do putStrLn "a single identifier or expression is expected from the command"
|
||||||
return void,
|
return void,
|
||||||
needsTypeCheck = False
|
needsTypeCheck = False
|
||||||
})
|
})
|
||||||
]
|
]
|
||||||
where
|
where
|
||||||
getEnv exec opts ts = liftSIO . exec opts ts =<< getPGFEnv
|
needPGF exec opts ts = do
|
||||||
|
mb_pgf <- getPGF
|
||||||
par pgf opts s = case optOpenTypes opts of
|
case mb_pgf of
|
||||||
[] -> [parse_ pgf lang (optType pgf opts) (Just dp) s | lang <- optLangs pgf opts]
|
Just pgf -> liftSIO $ exec opts ts pgf
|
||||||
open_typs -> [parseWithRecovery pgf lang (optType pgf opts) open_typs (Just dp) s | lang <- optLangs pgf opts]
|
_ -> fail "Import a grammar before using this command"
|
||||||
where
|
|
||||||
dp = valIntOpts "depth" 4 opts
|
|
||||||
|
|
||||||
fromParse opts = foldr (joinPiped . fromParse1 opts) void
|
|
||||||
|
|
||||||
joinPiped (Piped (es1,ms1)) (Piped (es2,ms2)) = Piped (jA es1 es2,ms1+++-ms2)
|
joinPiped (Piped (es1,ms1)) (Piped (es2,ms2)) = Piped (jA es1 es2,ms1+++-ms2)
|
||||||
where
|
where
|
||||||
jA (Exprs es1) (Exprs es2) = Exprs (es1++es2)
|
jA (Exprs es1) (Exprs es2) = Exprs (es1++es2)
|
||||||
-- ^ fromParse1 always output Exprs
|
|
||||||
|
|
||||||
fromParse1 opts (s,(po,bs))
|
fromParse1 opts (s,po) =
|
||||||
| isOpt "bracket" opts = pipeMessage (showBracketedString bs)
|
case po of
|
||||||
| otherwise =
|
ParseOk ts -> fromExprs (isOpt "show_probs" opts) (takeOptNum opts ts)
|
||||||
case po of
|
ParseFailed i t -> pipeMessage $ "The parser failed at token "
|
||||||
ParseOk ts -> fromExprs ts
|
++ show i ++": "
|
||||||
ParseFailed i -> pipeMessage $ "The parser failed at token "
|
++ show t
|
||||||
++ show i ++": "
|
ParseIncomplete -> pipeMessage "The sentence is not complete"
|
||||||
++ show (words s !! max 0 (i-1))
|
|
||||||
-- ++ " in " ++ show s
|
|
||||||
ParseIncomplete -> pipeMessage "The sentence is not complete"
|
|
||||||
TypeError errs ->
|
|
||||||
pipeMessage . render $
|
|
||||||
"The parsing is successful but the type checking failed with error(s):"
|
|
||||||
$$ nest 2 (vcat (map (ppTcError . snd) errs))
|
|
||||||
|
|
||||||
optLins pgf opts ts = case opts of
|
optLins pgf opts ts = concatMap (optLin pgf opts) ts
|
||||||
_ | isOpt "groups" opts ->
|
|
||||||
concatMap snd $ groupResults
|
|
||||||
[[(lang, s) | lang <- optLangs pgf opts,s <- linear pgf opts lang t] | t <- ts]
|
|
||||||
_ -> concatMap (optLin pgf opts) ts
|
|
||||||
optLin pgf opts t =
|
optLin pgf opts t =
|
||||||
case opts of
|
case opts of
|
||||||
_ | isOpt "treebank" opts && isOpt "chunks" opts ->
|
_ | isOpt "treebank" opts && isOpt "chunks" opts ->
|
||||||
(showCId (abstractName pgf) ++ ": " ++ showExpr [] t) :
|
(abstractName pgf ++ ": " ++ showExpr [] t) :
|
||||||
[showCId lang ++ ": " ++ li | (lang,li) <- linChunks pgf opts t] --linear pgf opts lang t | lang <- optLangs pgf opts]
|
[lang ++ ": " ++ li | (lang,li) <- linChunks pgf opts t] --linear pgf opts lang t | lang <- optLangs pgf opts]
|
||||||
_ | isOpt "treebank" opts ->
|
_ | isOpt "treebank" opts ->
|
||||||
(showCId (abstractName pgf) ++ ": " ++ showExpr [] t) :
|
(abstractName pgf ++ ": " ++ showExpr [] t) :
|
||||||
[showCId lang ++ ": " ++ s | lang <- optLangs pgf opts, s<-linear pgf opts lang t]
|
[concreteName concr ++ ": " ++ s | concr <- optLangs pgf opts, s<-linear opts concr t]
|
||||||
_ | isOpt "tabtreebank" opts ->
|
|
||||||
return $ concat $ intersperse "\t" $ (showExpr [] t) :
|
|
||||||
[s | lang <- optLangs pgf opts, s <- linear pgf opts lang t]
|
|
||||||
_ | isOpt "chunks" opts -> map snd $ linChunks pgf opts t
|
_ | isOpt "chunks" opts -> map snd $ linChunks pgf opts t
|
||||||
_ -> [s | lang <- optLangs pgf opts, s<-linear pgf opts lang t]
|
_ -> [s | concr <- optLangs pgf opts, s <- linear opts concr t]
|
||||||
linChunks pgf opts t =
|
linChunks pgf opts t =
|
||||||
[(lang, unwords (intersperse "<+>" (map (unlines . linear pgf opts lang) (treeChunks t)))) | lang <- optLangs pgf opts]
|
[(concreteName concr, unwords (intersperse "<+>" (map (unlines . linear opts concr) (treeChunks t)))) | concr <- optLangs pgf opts]
|
||||||
|
|
||||||
linear :: PGF -> [Option] -> CId -> Expr -> [String]
|
linear :: [Option] -> Concr -> Expr -> [String]
|
||||||
linear pgf opts lang = let unl = unlex opts lang in case opts of
|
linear opts concr = case opts of
|
||||||
_ | isOpt "all" opts -> concat . -- intersperse [[]] .
|
_ | isOpt "all" opts -> concat .
|
||||||
map (map (unl . snd)) . tabularLinearizes pgf lang
|
map (map snd) . tabularLinearizeAll concr
|
||||||
_ | isOpt "list" opts -> (:[]) . commaList . concat .
|
_ | isOpt "list" opts -> (:[]) . commaList . concat .
|
||||||
map (map (unl . snd)) . tabularLinearizes pgf lang
|
map (map snd) . tabularLinearizeAll concr
|
||||||
_ | isOpt "table" opts -> concat . -- intersperse [[]] .
|
_ | isOpt "table" opts -> concat .
|
||||||
map (map (\(p,v) -> p+++":"+++unl v)) . tabularLinearizes pgf lang
|
map (map (\(p,v) -> p+++":"+++v)) . tabularLinearizeAll concr
|
||||||
_ | isOpt "bracket" opts -> (:[]) . unwords . map showBracketedString . bracketedLinearize pgf lang
|
_ | isOpt "bracket" opts -> (:[]) . unwords . map showBracketedString . bracketedLinearize concr
|
||||||
_ -> (:[]) . unl . linearize pgf lang
|
_ -> (:[]) . linearize concr
|
||||||
|
|
||||||
-- replace each non-atomic constructor with mkC, where C is the val cat
|
-- replace each non-atomic constructor with mkC, where C is the val cat
|
||||||
tree2mk pgf = showExpr [] . t2m where
|
tree2mk pgf = showExpr [] . t2m where
|
||||||
t2m t = case unApp t of
|
t2m t = case unApp t of
|
||||||
Just (cid,ts@(_:_)) -> mkApp (mk cid) (map t2m ts)
|
Just (cid,ts@(_:_)) -> mkApp (mk cid) (map t2m ts)
|
||||||
_ -> t
|
_ -> t
|
||||||
mk = mkCId . ("mk" ++) . showCId . lookValCat (abstract pgf)
|
mk f = case functionType pgf f of
|
||||||
|
Just ty -> let (_,cat,_) = unType ty
|
||||||
unlex opts lang = stringOps Nothing (getUnlex opts lang ++ map prOpt opts) ----
|
in "mk" ++ cat
|
||||||
|
Nothing -> f
|
||||||
getUnlex opts lang = case words (valStrOpts "unlexer" "" opts) of
|
|
||||||
lexs -> case lookup lang
|
|
||||||
[(mkCId la,tail le) | lex <- lexs, let (la,le) = span (/='=') lex, not (null le)] of
|
|
||||||
Just le -> chunks ',' le
|
|
||||||
_ -> []
|
|
||||||
|
|
||||||
commaList [] = []
|
commaList [] = []
|
||||||
commaList ws = concat $ head ws : map (", " ++) (tail ws)
|
commaList ws = concat $ head ws : map (", " ++) (tail ws)
|
||||||
|
|
||||||
-- Proposed logic of coding in unlexing:
|
|
||||||
-- - If lang has no coding flag, or -to_utf8 is not in opts, just opts are used.
|
|
||||||
-- - If lang has flag coding=utf8, -to_utf8 is ignored.
|
|
||||||
-- - If lang has coding=other, and -to_utf8 is in opts, from_other is applied first.
|
|
||||||
-- THIS DOES NOT WORK UNFORTUNATELY - can't use the grammar flag properly
|
|
||||||
{-
|
|
||||||
unlexx pgf opts lang = {- trace (unwords optsC) $ -} stringOps Nothing optsC where ----
|
|
||||||
optsC = case lookConcrFlag pgf (mkCId lang) (mkCId "coding") of
|
|
||||||
Just (LStr "utf8") -> filter (/="to_utf8") $ map prOpt opts
|
|
||||||
Just (LStr other) | isOpt "to_utf8" opts ->
|
|
||||||
let cod = ("from_" ++ other)
|
|
||||||
in cod : filter (/=cod) (map prOpt opts)
|
|
||||||
_ -> map prOpt opts
|
|
||||||
-}
|
|
||||||
optRestricted opts pgf =
|
|
||||||
restrictPGF (\f -> and [hasLin pgf la f | la <- optLangs pgf opts]) pgf
|
|
||||||
|
|
||||||
optLang = optLangFlag "lang"
|
optLang = optLangFlag "lang"
|
||||||
optLangs = optLangsFlag "lang"
|
optLangs = optLangsFlag "lang"
|
||||||
|
|
||||||
optLangsFlag f pgf opts = case valStrOpts f "" opts of
|
optLangFlag flag pgf opts =
|
||||||
"" -> languages pgf
|
case optLangsFlag flag pgf opts of
|
||||||
lang -> map (completeLang pgf) (chunks ',' lang)
|
[] -> fail "no language specified"
|
||||||
completeLang pgf la = let cla = (mkCId la) in
|
(l:ls) -> return l
|
||||||
if elem cla (languages pgf)
|
|
||||||
then cla
|
|
||||||
else (mkCId (showCId (abstractName pgf) ++ la))
|
|
||||||
|
|
||||||
optLangFlag f pgf opts = head $ optLangsFlag f pgf opts ++ [wildCId]
|
optLangsFlag flag pgf opts =
|
||||||
|
case valStrOpts flag "" opts of
|
||||||
|
"" -> Map.elems langs
|
||||||
|
str -> mapMaybe (completeLang pgf) (chunks ',' str)
|
||||||
|
where
|
||||||
|
langs = languages pgf
|
||||||
|
|
||||||
optOpenTypes opts = case valStrOpts "openclass" "" opts of
|
completeLang pgf la =
|
||||||
"" -> []
|
mplus (Map.lookup la langs)
|
||||||
cats -> mapMaybe readType (chunks ',' cats)
|
(Map.lookup (abstractName pgf ++ la) langs)
|
||||||
|
|
||||||
optProbs opts pgf = case valStrOpts "probs" "" opts of
|
|
||||||
"" -> return pgf
|
|
||||||
file -> do
|
|
||||||
probs <- restricted $ readProbabilitiesFromFile file pgf
|
|
||||||
return (setProbabilities probs pgf)
|
|
||||||
|
|
||||||
optFile opts = valStrOpts "file" "_gftmp" opts
|
optFile opts = valStrOpts "file" "_gftmp" opts
|
||||||
|
|
||||||
optType pgf opts =
|
optType pgf opts =
|
||||||
let str = valStrOpts "cat" (showCId $ lookStartCat pgf) opts
|
let readOpt str = case readType str of
|
||||||
in case readType str of
|
Just ty -> case checkType pgf ty of
|
||||||
Just ty -> case checkType pgf ty of
|
Left err -> error err
|
||||||
Left tcErr -> error $ render (ppTcError tcErr)
|
Right ty -> ty
|
||||||
Right ty -> ty
|
Nothing -> error ("Can't parse '"++str++"' as a type")
|
||||||
Nothing -> error ("Can't parse '"++str++"' as a type")
|
in maybeStrOpts "cat" (startCat pgf) readOpt opts
|
||||||
optViewFormat opts = valStrOpts "format" "png" opts
|
optViewFormat opts = valStrOpts "format" "png" opts
|
||||||
optViewGraph opts = valStrOpts "view" "open" opts
|
optViewGraph opts = valStrOpts "view" "open" opts
|
||||||
optNum opts = valIntOpts "number" 1 opts
|
optNum opts = valIntOpts "number" 1 opts
|
||||||
optNumInf opts = valIntOpts "number" 1000000000 opts ---- 10^9
|
optNumInf opts = valIntOpts "number" 1000000000 opts ---- 10^9
|
||||||
takeOptNum opts = take (optNumInf opts)
|
takeOptNum opts = take (optNumInf opts)
|
||||||
|
|
||||||
returnFromExprs es = return $ case es of
|
returnFromExprs show_p es =
|
||||||
[] -> pipeMessage "no trees found"
|
return $
|
||||||
_ -> fromExprs es
|
case es of
|
||||||
|
[] -> pipeMessage "no trees found"
|
||||||
|
_ -> fromExprs show_p es
|
||||||
|
|
||||||
prGrammar (Env pgf mos) opts
|
prGrammar pgf opts
|
||||||
| isOpt "pgf" opts = do
|
| isOpt "pgf" opts = do
|
||||||
let pgf1 = if isOpt "opt" opts then optimizePGF pgf else pgf
|
let outfile = valStrOpts "file" (abstractName pgf ++ ".pgf") opts
|
||||||
let outfile = valStrOpts "file" (showCId (abstractName pgf) ++ ".pgf") opts
|
restricted $ writePGF outfile pgf
|
||||||
restricted $ encodeFile outfile pgf1
|
|
||||||
putStrLn $ "wrote file " ++ outfile
|
putStrLn $ "wrote file " ++ outfile
|
||||||
return void
|
return void
|
||||||
| isOpt "cats" opts = return $ fromString $ unwords $ map showCId $ categories pgf
|
| isOpt "cats" opts = return $ fromString $ unwords $ categories pgf
|
||||||
| isOpt "funs" opts = return $ fromString $ unlines $ map showFun $ funsigs pgf
|
| isOpt "funs" opts = return $ fromString $ unlines [showFun pgf f ty | f <- functions pgf, Just ty <- [functionType pgf f]]
|
||||||
| isOpt "fullform" opts = return $ fromString $ concatMap (morpho mos "" prFullFormLexicon) $ optLangs pgf opts
|
| isOpt "fullform" opts = return $ fromString $ concatMap prFullFormLexicon $ optLangs pgf opts
|
||||||
| isOpt "langs" opts = return $ fromString $ unwords $ map showCId $ languages pgf
|
| isOpt "langs" opts = return $ fromString $ unwords $ Map.keys $ languages pgf
|
||||||
|
|
||||||
| isOpt "lexc" opts = return $ fromString $ concatMap (morpho mos "" prLexcLexicon) $ optLangs pgf opts
|
| isOpt "lexc" opts = return $ fromString $ concatMap prLexcLexicon $ optLangs pgf opts
|
||||||
| isOpt "missing" opts = return $ fromString $ unlines $ [unwords (showCId la:":": map showCId cs) |
|
| isOpt "missing" opts = return $ fromString $ unlines $ [unwords (concreteName concr:":":[f | f <- functions pgf, not (hasLinearization concr f)]) |
|
||||||
la <- optLangs pgf opts, let cs = missingLins pgf la]
|
concr <- optLangs pgf opts]
|
||||||
| isOpt "words" opts = return $ fromString $ concatMap (morpho mos "" prAllWords) $ optLangs pgf opts
|
| isOpt "words" opts = return $ fromString $ concatMap prAllWords $ optLangs pgf opts
|
||||||
| otherwise = do fmt <- readOutputFormat (valStrOpts "printer" "pgf_pretty" opts)
|
| otherwise = do fmt <- readOutputFormat (valStrOpts "printer" "pgf_pretty" opts)
|
||||||
return $ fromString $ concatMap snd $ exportPGF noOptions fmt pgf
|
return $ fromString $ concatMap snd $ exportPGF noOptions fmt pgf
|
||||||
|
|
||||||
funsigs pgf = [(f,ty) | (f,(ty,_,_,_)) <- Map.assocs (funs (abstract pgf))]
|
showFun pgf id ty = kwd++" "++ id ++ " : " ++ showType [] ty
|
||||||
showFun (f,ty) = showCId f ++ " : " ++ showType [] ty ++ " ;"
|
where
|
||||||
|
kwd | functionIsDataCon pgf id = "data"
|
||||||
|
| otherwise = "fun"
|
||||||
|
|
||||||
morphos (Env pgf mos) opts s =
|
morphos pgf opts s =
|
||||||
[(s,morpho mos [] (\mo -> lookupMorpho mo s) la) | la <- optLangs pgf opts]
|
[(s,lookupMorpho concr s) | concr <- optLangs pgf opts]
|
||||||
|
|
||||||
morpho mos z f la = maybe z f $ Map.lookup la mos
|
|
||||||
|
|
||||||
optMorpho (Env pgf mos) opts = morpho mos (error "no morpho") id (head (optLangs pgf opts))
|
|
||||||
|
|
||||||
optClitics opts = case valStrOpts "clitics" "" opts of
|
optClitics opts = case valStrOpts "clitics" "" opts of
|
||||||
"" -> []
|
"" -> []
|
||||||
@@ -930,18 +820,28 @@ pgfCommands = Map.fromList [
|
|||||||
-- ps -f -g s returns g (f s)
|
-- ps -f -g s returns g (f s)
|
||||||
treeOps pgf opts s = foldr app s (reverse opts) where
|
treeOps pgf opts s = foldr app s (reverse opts) where
|
||||||
app (OOpt op) | Just (Left f) <- treeOp pgf op = f
|
app (OOpt op) | Just (Left f) <- treeOp pgf op = f
|
||||||
app (OFlag op (VId x)) | Just (Right f) <- treeOp pgf op = f (mkCId x)
|
app (OFlag op (VId x)) | Just (Right f) <- treeOp pgf op = f x
|
||||||
app _ = id
|
app _ = id
|
||||||
|
|
||||||
|
morphoMissing :: Concr -> [String] -> [String]
|
||||||
|
morphoMissing = morphoClassify False
|
||||||
|
|
||||||
|
morphoKnown :: Concr -> [String] -> [String]
|
||||||
|
morphoKnown = morphoClassify True
|
||||||
|
|
||||||
|
morphoClassify :: Bool -> Concr -> [String] -> [String]
|
||||||
|
morphoClassify k concr ws = [w | w <- ws, k /= null (lookupMorpho concr w), notLiteral w] where
|
||||||
|
notLiteral w = not (all isDigit w)
|
||||||
|
|
||||||
treeOpOptions pgf = [(op,expl) | (op,(expl,Left _)) <- allTreeOps pgf]
|
treeOpOptions pgf = [(op,expl) | (op,(expl,Left _)) <- allTreeOps pgf]
|
||||||
treeOpFlags pgf = [(op,expl) | (op,(expl,Right _)) <- allTreeOps pgf]
|
treeOpFlags pgf = [(op,expl) | (op,(expl,Right _)) <- allTreeOps pgf]
|
||||||
|
|
||||||
translationQuiz :: Maybe Expr -> PGF -> Language -> Language -> Type -> IO ()
|
translationQuiz :: Maybe Expr -> PGF -> Concr -> Concr -> Type -> IO ()
|
||||||
translationQuiz mex pgf ig og typ = do
|
translationQuiz mex pgf ig og typ = do
|
||||||
tts <- translationList mex pgf ig og typ infinity
|
tts <- translationList mex pgf ig og typ infinity
|
||||||
mkQuiz "Welcome to GF Translation Quiz." tts
|
mkQuiz "Welcome to GF Translation Quiz." tts
|
||||||
|
|
||||||
morphologyQuiz :: Maybe Expr -> PGF -> Language -> Type -> IO ()
|
morphologyQuiz :: Maybe Expr -> PGF -> Concr -> Type -> IO ()
|
||||||
morphologyQuiz mex pgf ig typ = do
|
morphologyQuiz mex pgf ig typ = do
|
||||||
tts <- morphologyList mex pgf ig typ infinity
|
tts <- morphologyList mex pgf ig typ infinity
|
||||||
mkQuiz "Welcome to GF Morphology Quiz." tts
|
mkQuiz "Welcome to GF Morphology Quiz." tts
|
||||||
@@ -950,30 +850,28 @@ morphologyQuiz mex pgf ig typ = do
|
|||||||
infinity :: Int
|
infinity :: Int
|
||||||
infinity = 256
|
infinity = 256
|
||||||
|
|
||||||
prLexcLexicon :: Morpho -> String
|
prLexcLexicon :: Concr -> String
|
||||||
prLexcLexicon mo =
|
prLexcLexicon concr =
|
||||||
unlines $ "Multichar_Symbols":multichars:"":"LEXICON Root" : [prLexc l p ++ ":" ++ w ++ " # ;" | (w,lps) <- morpho, (l,p) <- lps] ++ ["END"]
|
unlines $ "Multichar_Symbols":multichars:"":"LEXICON Root" : [prLexc l p ++ ":" ++ w ++ " # ;" | (w,lps) <- morpho, (l,p,_) <- lps] ++ ["END"]
|
||||||
where
|
where
|
||||||
morpho = fullFormLexicon mo
|
morpho = fullFormLexicon concr
|
||||||
prLexc l p = showCId l ++ concat (mkTags (words p))
|
prLexc l p = l ++ concat (mkTags (words p))
|
||||||
mkTags p = case p of
|
mkTags p = case p of
|
||||||
"s":ws -> mkTags ws --- remove record field
|
"s":ws -> mkTags ws --- remove record field
|
||||||
ws -> map ('+':) ws
|
ws -> map ('+':) ws
|
||||||
|
|
||||||
multichars = unwords $ nub $ concat [mkTags (words p) | (w,lps) <- morpho, (l,p) <- lps]
|
multichars = unwords $ nub $ concat [mkTags (words p) | (w,lps) <- morpho, (l,p,_) <- lps]
|
||||||
-- thick_A+(AAdj+Posit+Gen):thick's # ;
|
|
||||||
|
|
||||||
prFullFormLexicon :: Morpho -> String
|
prFullFormLexicon :: Concr -> String
|
||||||
prFullFormLexicon mo =
|
prFullFormLexicon concr =
|
||||||
unlines (map prMorphoAnalysis (fullFormLexicon mo))
|
unlines (map prMorphoAnalysis (fullFormLexicon concr))
|
||||||
|
|
||||||
prAllWords :: Morpho -> String
|
prAllWords :: Concr -> String
|
||||||
prAllWords mo =
|
prAllWords concr =
|
||||||
unwords [w | (w,_) <- fullFormLexicon mo]
|
unwords [w | (w,_) <- fullFormLexicon concr]
|
||||||
|
|
||||||
prMorphoAnalysis :: (String,[(Lemma,Analysis)]) -> String
|
|
||||||
prMorphoAnalysis (w,lps) =
|
prMorphoAnalysis (w,lps) =
|
||||||
unlines (w:[showCId l ++ " : " ++ p | (l,p) <- lps])
|
unlines (w:[l ++ " : " ++ p ++ show prob | (l,p,prob) <- lps])
|
||||||
|
|
||||||
viewGraphviz :: String -> String -> String -> [String] -> SIO CommandOutput
|
viewGraphviz :: String -> String -> String -> [String] -> SIO CommandOutput
|
||||||
viewGraphviz view format name grphs = do
|
viewGraphviz view format name grphs = do
|
||||||
|
|||||||
@@ -1,822 +0,0 @@
|
|||||||
{-# LANGUAGE FlexibleInstances, UndecidableInstances #-}
|
|
||||||
module GF.Command.Commands2 (
|
|
||||||
PGFEnv,HasPGFEnv(..),pgf,concs,pgfEnv,emptyPGFEnv,pgfCommands,
|
|
||||||
options, flags,
|
|
||||||
) where
|
|
||||||
import Prelude hiding (putStrLn,(<>)) -- GHC 8.4.1 clash with Text.PrettyPrint
|
|
||||||
|
|
||||||
import PGF2
|
|
||||||
import qualified PGF as H
|
|
||||||
import GF.Compile.ToAPI(exprToAPI)
|
|
||||||
import GF.Infra.UseIO(writeUTF8File)
|
|
||||||
import GF.Infra.SIO(MonadSIO,liftSIO,putStrLn,restricted,restrictedSystem)
|
|
||||||
import GF.Command.Abstract
|
|
||||||
import GF.Command.CommandInfo
|
|
||||||
import GF.Data.Operations
|
|
||||||
import Data.List(intersperse,intersect,nub,sortBy)
|
|
||||||
import Data.Maybe
|
|
||||||
import qualified Data.Map as Map
|
|
||||||
import GF.Text.Pretty
|
|
||||||
import Control.Monad(mplus)
|
|
||||||
|
|
||||||
|
|
||||||
data PGFEnv = Env {pgf::Maybe PGF,concs::Map.Map ConcName Concr}
|
|
||||||
|
|
||||||
pgfEnv pgf = Env (Just pgf) (languages pgf)
|
|
||||||
emptyPGFEnv = Env Nothing Map.empty
|
|
||||||
|
|
||||||
class (Monad m,MonadSIO m) => HasPGFEnv m where getPGFEnv :: m PGFEnv
|
|
||||||
|
|
||||||
instance (Monad m,HasPGFEnv m) => TypeCheckArg m where
|
|
||||||
typeCheckArg e = do env <- getPGFEnv
|
|
||||||
case pgf env of
|
|
||||||
Just gr -> either fail
|
|
||||||
(return . hsExpr . fst)
|
|
||||||
(inferExpr gr (cExpr e))
|
|
||||||
Nothing -> fail "Import a grammar before using this command"
|
|
||||||
|
|
||||||
pgfCommands :: HasPGFEnv m => Map.Map String (CommandInfo m)
|
|
||||||
pgfCommands = Map.fromList [
|
|
||||||
("aw", emptyCommandInfo {
|
|
||||||
longname = "align_words",
|
|
||||||
synopsis = "show word alignments between languages graphically",
|
|
||||||
explanation = unlines [
|
|
||||||
"Prints a set of strings in the .dot format (the graphviz format).",
|
|
||||||
"The graph can be saved in a file by the wf command as usual.",
|
|
||||||
"If the -view flag is defined, the graph is saved in a temporary file",
|
|
||||||
"which is processed by graphviz and displayed by the program indicated",
|
|
||||||
"by the flag. The target format is postscript, unless overridden by the",
|
|
||||||
"flag -format."
|
|
||||||
],
|
|
||||||
exec = needPGF $ \opts es env -> do
|
|
||||||
let cncs = optConcs env opts
|
|
||||||
if isOpt "giza" opts
|
|
||||||
then if length cncs == 2
|
|
||||||
then let giz = map (gizaAlignment pgf (snd (cncs !! 0)) (snd (cncs !! 1)) . cExpr) (toExprs es)
|
|
||||||
lsrc = unlines $ map (\(x,_,_) -> x) giz
|
|
||||||
ltrg = unlines $ map (\(_,x,_) -> x) giz
|
|
||||||
align = unlines $ map (\(_,_,x) -> x) giz
|
|
||||||
grph = if null (toExprs es) then [] else lsrc ++ "\n--end_source--\n\n"++ltrg++"\n-end_target--\n\n"++align
|
|
||||||
in return (fromString grph)
|
|
||||||
else error "For giza alignment you need exactly two languages"
|
|
||||||
else let gvOptions=graphvizDefaults{leafFont = valStrOpts "font" "" opts,
|
|
||||||
leafColor = valStrOpts "color" "" opts,
|
|
||||||
leafEdgeStyle = valStrOpts "edgestyle" "" opts
|
|
||||||
}
|
|
||||||
grph = if null (toExprs es) then [] else graphvizWordAlignment (map snd cncs) gvOptions (cExpr (head (toExprs es)))
|
|
||||||
in if isFlag "view" opts || isFlag "format" opts
|
|
||||||
then do let file s = "_grph." ++ s
|
|
||||||
let view = optViewGraph opts
|
|
||||||
let format = optViewFormat opts
|
|
||||||
restricted $ writeUTF8File (file "dot") grph
|
|
||||||
restrictedSystem $ "dot -T" ++ format ++ " " ++ file "dot" ++ " > " ++ file format
|
|
||||||
restrictedSystem $ view ++ " " ++ file format
|
|
||||||
return void
|
|
||||||
else return (fromString grph),
|
|
||||||
examples = [
|
|
||||||
("gr | aw" , "generate a tree and show word alignment as graph script"),
|
|
||||||
("gr | aw -view=\"open\"" , "generate a tree and display alignment on Mac"),
|
|
||||||
("gr | aw -view=\"eog\"" , "generate a tree and display alignment on Ubuntu"),
|
|
||||||
("gt | aw -giza | wf -file=aligns" , "generate trees, send giza alignments to file")
|
|
||||||
],
|
|
||||||
options = [
|
|
||||||
("giza", "show alignments in the Giza format; the first two languages")
|
|
||||||
],
|
|
||||||
flags = [
|
|
||||||
("format","format of the visualization file (default \"png\")"),
|
|
||||||
("lang", "alignments for this list of languages (default: all)"),
|
|
||||||
("view", "program to open the resulting file"),
|
|
||||||
("font", "font for the words"),
|
|
||||||
("color", "color for the words"),
|
|
||||||
("edgestyle", "the style for links between words")
|
|
||||||
]
|
|
||||||
}),
|
|
||||||
{-
|
|
||||||
("eb", emptyCommandInfo {
|
|
||||||
longname = "example_based",
|
|
||||||
syntax = "eb (-probs=FILE | -lang=LANG)* -file=FILE.gfe",
|
|
||||||
synopsis = "converts .gfe files to .gf files by parsing examples to trees",
|
|
||||||
explanation = unlines [
|
|
||||||
"Reads FILE.gfe and writes FILE.gf. Each expression of form",
|
|
||||||
"'%ex CAT QUOTEDSTRING' in FILE.gfe is replaced by a syntax tree.",
|
|
||||||
"This tree is the first one returned by the parser; a biased ranking",
|
|
||||||
"can be used to regulate the order. If there are more than one parses",
|
|
||||||
"the rest are shown in comments, with probabilities if the order is biased.",
|
|
||||||
"The probabilities flag and configuration file is similar to the commands",
|
|
||||||
"gr and rt. Notice that the command doesn't change the environment,",
|
|
||||||
"but the resulting .gf file must be imported separately."
|
|
||||||
],
|
|
||||||
options = [
|
|
||||||
("api","convert trees to overloaded API expressions (using Syntax not Lang)")
|
|
||||||
],
|
|
||||||
flags = [
|
|
||||||
("file","the file to be converted (suffix .gfe must be given)"),
|
|
||||||
("lang","the language in which to parse"),
|
|
||||||
("probs","file with probabilities to rank the parses")
|
|
||||||
],
|
|
||||||
exec = \env@(pgf, mos) opts _ -> do
|
|
||||||
let file = optFile opts
|
|
||||||
pgf <- optProbs opts pgf
|
|
||||||
let printer = if (isOpt "api" opts) then exprToAPI else (H.showExpr [])
|
|
||||||
let conf = configureExBased pgf (optMorpho env opts) (optLang pgf opts) printer
|
|
||||||
(file',ws) <- restricted $ parseExamplesInGrammar conf file
|
|
||||||
if null ws then return () else putStrLn ("unknown words: " ++ unwords ws)
|
|
||||||
return (fromString ("wrote " ++ file')),
|
|
||||||
needsTypeCheck = False
|
|
||||||
}),
|
|
||||||
-}
|
|
||||||
{-
|
|
||||||
("gr", emptyCommandInfo {
|
|
||||||
longname = "generate_random",
|
|
||||||
synopsis = "generate random trees in the current abstract syntax",
|
|
||||||
syntax = "gr [-cat=CAT] [-number=INT]",
|
|
||||||
examples = [
|
|
||||||
mkEx "gr -- one tree in the startcat of the current grammar",
|
|
||||||
mkEx "gr -cat=NP -number=16 -- 16 trees in the category NP",
|
|
||||||
mkEx "gr -lang=LangHin,LangTha -cat=Cl -- Cl, both in LangHin and LangTha",
|
|
||||||
mkEx "gr -probs=FILE -- generate with bias",
|
|
||||||
mkEx "gr (AdjCN ? (UseN ?)) -- generate trees of form (AdjCN ? (UseN ?))"
|
|
||||||
],
|
|
||||||
explanation = unlines [
|
|
||||||
"Generates a list of random trees, by default one tree.",
|
|
||||||
"If a tree argument is given, the command completes the Tree with values to",
|
|
||||||
"all metavariables in the tree. The generation can be biased by probabilities,",
|
|
||||||
"given in a file in the -probs flag."
|
|
||||||
],
|
|
||||||
flags = [
|
|
||||||
("cat","generation category"),
|
|
||||||
("lang","uses only functions that have linearizations in all these languages"),
|
|
||||||
("number","number of trees generated"),
|
|
||||||
("depth","the maximum generation depth"),
|
|
||||||
("probs", "file with biased probabilities (format 'f 0.4' one by line)")
|
|
||||||
],
|
|
||||||
exec = \env@(pgf, mos) opts xs -> do
|
|
||||||
pgf <- optProbs opts (optRestricted opts pgf)
|
|
||||||
gen <- newStdGen
|
|
||||||
let dp = valIntOpts "depth" 4 opts
|
|
||||||
let ts = case mexp xs of
|
|
||||||
Just ex -> H.generateRandomFromDepth gen pgf ex (Just dp)
|
|
||||||
Nothing -> H.generateRandomDepth gen pgf (optType pgf opts) (Just dp)
|
|
||||||
returnFromExprs $ take (optNum opts) ts
|
|
||||||
}),
|
|
||||||
-}
|
|
||||||
("gt", emptyCommandInfo {
|
|
||||||
longname = "generate_trees",
|
|
||||||
synopsis = "generates a list of trees, by default exhaustive",
|
|
||||||
flags = [("cat","the generation category"),
|
|
||||||
("number","the number of trees generated")],
|
|
||||||
examples = [
|
|
||||||
mkEx "gt -- all trees in the startcat",
|
|
||||||
mkEx "gt -cat=NP -number=16 -- 16 trees in the category NP"],
|
|
||||||
exec = needPGF $ \ opts _ env@(pgf,_) ->
|
|
||||||
let ts = map fst (generateAll pgf cat)
|
|
||||||
cat = optType pgf opts
|
|
||||||
in returnFromCExprs (takeOptNum opts ts),
|
|
||||||
needsTypeCheck = False
|
|
||||||
}),
|
|
||||||
("i", emptyCommandInfo {
|
|
||||||
longname = "import",
|
|
||||||
synopsis = "import a grammar from a compiled .pgf file",
|
|
||||||
explanation = unlines [
|
|
||||||
"Reads a grammar from a compiled .pgf file.",
|
|
||||||
"Old modules are discarded.",
|
|
||||||
{-
|
|
||||||
"The grammar parser depends on the file name suffix:",
|
|
||||||
|
|
||||||
" .cf context-free (labelled BNF) source",
|
|
||||||
" .ebnf extended BNF source",
|
|
||||||
" .gfm multi-module GF source",
|
|
||||||
" .gf normal GF source",
|
|
||||||
" .gfo compiled GF source",
|
|
||||||
-}
|
|
||||||
" .pgf precompiled grammar in Portable Grammar Format"
|
|
||||||
],
|
|
||||||
flags = [
|
|
||||||
-- ("probs","file with biased probabilities for generation")
|
|
||||||
],
|
|
||||||
options = [
|
|
||||||
-- ["gfo", "src", "no-cpu", "cpu", "quiet", "verbose"]
|
|
||||||
-- ("retain","retain operations (used for cc command)"),
|
|
||||||
-- ("src", "force compilation from source"),
|
|
||||||
-- ("v", "be verbose - show intermediate status information")
|
|
||||||
],
|
|
||||||
needsTypeCheck = False
|
|
||||||
}),
|
|
||||||
("l", emptyCommandInfo {
|
|
||||||
longname = "linearize",
|
|
||||||
synopsis = "convert an abstract syntax expression to string",
|
|
||||||
explanation = unlines [
|
|
||||||
"Shows the linearization of a Tree by the grammars in scope.",
|
|
||||||
"The -lang flag can be used to restrict this to fewer languages.",
|
|
||||||
"A sequence of string operations (see command ps) can be given",
|
|
||||||
"as options, and works then like a pipe to the ps command, except",
|
|
||||||
"that it only affect the strings, not e.g. the table labels.",
|
|
||||||
"These can be given separately to each language with the unlexer flag",
|
|
||||||
"whose results are prepended to the other lexer flags. The value of the",
|
|
||||||
"unlexer flag is a space-separated list of comma-separated string operation",
|
|
||||||
"sequences; see example."
|
|
||||||
],
|
|
||||||
examples = [
|
|
||||||
mkEx "l -lang=LangSwe,LangNor no_Utt -- linearize a tree to LangSwe and LangNor",
|
|
||||||
mkEx "gr -lang=LangHin -cat=Cl | l -table -to_devanagari -- hindi table",
|
|
||||||
mkEx "l -unlexer=\"LangAra=to_arabic LangHin=to_devanagari\" -- different unlexers"
|
|
||||||
],
|
|
||||||
exec = needPGF $ \ opts arg env ->
|
|
||||||
return . fromStrings . optLins env opts . map cExpr $ toExprs arg,
|
|
||||||
options = [
|
|
||||||
("all", "show all forms and variants, one by line (cf. l -list)"),
|
|
||||||
("bracket","show tree structure with brackets and paths to nodes"),
|
|
||||||
("groups", "all languages, grouped by lang, remove duplicate strings"),
|
|
||||||
("list","show all forms and variants, comma-separated on one line (cf. l -all)"),
|
|
||||||
("multi","linearize to all languages (default)"),
|
|
||||||
("table","show all forms labelled by parameters"),
|
|
||||||
("treebank","show the tree and tag linearizations with language names")
|
|
||||||
],
|
|
||||||
flags = [
|
|
||||||
("lang","the languages of linearization (comma-separated, no spaces)")
|
|
||||||
]
|
|
||||||
}),
|
|
||||||
("ma", emptyCommandInfo {
|
|
||||||
longname = "morpho_analyse",
|
|
||||||
synopsis = "print the morphological analyses of the (multiword) expression in the string",
|
|
||||||
explanation = unlines [
|
|
||||||
"Prints all the analyses of the (multiword) expression in the input string,",
|
|
||||||
"using the morphological analyser of the actual grammar (see command pg)"
|
|
||||||
],
|
|
||||||
exec = needPGF $ \opts args env ->
|
|
||||||
return ((fromString . unlines .
|
|
||||||
map prMorphoAnalysis . concatMap (morphos env opts) . toStrings) args),
|
|
||||||
flags = [
|
|
||||||
("lang","the languages of analysis (comma-separated, no spaces)")
|
|
||||||
]
|
|
||||||
}),
|
|
||||||
{-
|
|
||||||
("mq", emptyCommandInfo {
|
|
||||||
longname = "morpho_quiz",
|
|
||||||
synopsis = "start a morphology quiz",
|
|
||||||
syntax = "mq (-cat=CAT)? (-probs=FILE)? TREE?",
|
|
||||||
exec = \env@(pgf, mos) opts xs -> do
|
|
||||||
let lang = optLang pgf opts
|
|
||||||
let typ = optType pgf opts
|
|
||||||
pgf <- optProbs opts pgf
|
|
||||||
let mt = mexp xs
|
|
||||||
restricted $ morphologyQuiz mt pgf lang typ
|
|
||||||
return void,
|
|
||||||
flags = [
|
|
||||||
("lang","language of the quiz"),
|
|
||||||
("cat","category of the quiz"),
|
|
||||||
("number","maximum number of questions"),
|
|
||||||
("probs","file with biased probabilities for generation")
|
|
||||||
]
|
|
||||||
}),
|
|
||||||
-}
|
|
||||||
("p", emptyCommandInfo {
|
|
||||||
longname = "parse",
|
|
||||||
synopsis = "parse a string to abstract syntax expression",
|
|
||||||
explanation = unlines [
|
|
||||||
"Shows all trees returned by parsing a string in the grammars in scope.",
|
|
||||||
"The -lang flag can be used to restrict this to fewer languages.",
|
|
||||||
"The default start category can be overridden by the -cat flag.",
|
|
||||||
"See also the ps command for lexing and character encoding."
|
|
||||||
],
|
|
||||||
flags = [
|
|
||||||
("cat","target category of parsing"),
|
|
||||||
("lang","the languages of parsing (comma-separated, no spaces)"),
|
|
||||||
("number","maximum number of trees returned")
|
|
||||||
],
|
|
||||||
examples = [
|
|
||||||
mkEx "p \"this fish is fresh\" | l -lang=Swe -- try parsing with all languages and translate the successful parses to Swedish"
|
|
||||||
],
|
|
||||||
exec = needPGF $ \ opts ts env -> return . cParse env opts $ toStrings ts
|
|
||||||
}),
|
|
||||||
("pg", emptyCommandInfo {
|
|
||||||
longname = "print_grammar",
|
|
||||||
synopsis = "prints different information about the grammar",
|
|
||||||
exec = needPGF $ \opts _ env -> prGrammar env opts,
|
|
||||||
options = [
|
|
||||||
("cats", "show just the names of abstract syntax categories"),
|
|
||||||
("fullform", "print the fullform lexicon"),
|
|
||||||
("funs", "show just the names and types of abstract syntax functions"),
|
|
||||||
("langs", "show just the names of top concrete syntax modules"),
|
|
||||||
("lexc", "print the lexicon in Xerox LEXC format"),
|
|
||||||
("missing","show just the names of functions that have no linearization"),
|
|
||||||
("words", "print the list of words")
|
|
||||||
],
|
|
||||||
flags = [
|
|
||||||
("lang","the languages that need to be printed")
|
|
||||||
],
|
|
||||||
examples = [
|
|
||||||
mkEx "pg -langs -- show the names of top concrete syntax modules",
|
|
||||||
mkEx "pg -funs | ? grep \" S ;\" -- show functions with value cat S"
|
|
||||||
]
|
|
||||||
}),
|
|
||||||
|
|
||||||
{-
|
|
||||||
("pt", emptyCommandInfo {
|
|
||||||
longname = "put_tree",
|
|
||||||
syntax = "pt OPT? TREE",
|
|
||||||
synopsis = "return a tree, possibly processed with a function",
|
|
||||||
explanation = unlines [
|
|
||||||
"Returns a tree obtained from its argument tree by applying",
|
|
||||||
"tree processing functions in the order given in the command line",
|
|
||||||
"option list. Thus 'pt -f -g s' returns g (f s). Typical tree processors",
|
|
||||||
"are type checking and semantic computation."
|
|
||||||
],
|
|
||||||
examples = [
|
|
||||||
mkEx "pt -compute (plus one two) -- compute value",
|
|
||||||
mkEx "p \"4 dogs love 5 cats\" | pt -transfer=digits2numeral | l -- four...five..."
|
|
||||||
],
|
|
||||||
exec = \env@(pgf, mos) opts ->
|
|
||||||
returnFromExprs . takeOptNum opts . treeOps pgf opts,
|
|
||||||
options = treeOpOptions undefined{-pgf-},
|
|
||||||
flags = [("number","take at most this many trees")] ++ treeOpFlags undefined{-pgf-}
|
|
||||||
}),
|
|
||||||
-}
|
|
||||||
("rf", emptyCommandInfo {
|
|
||||||
longname = "read_file",
|
|
||||||
synopsis = "read string or tree input from a file",
|
|
||||||
explanation = unlines [
|
|
||||||
"Reads input from file. The filename must be in double quotes.",
|
|
||||||
"The input is interpreted as a string by default, and can hence be",
|
|
||||||
"piped e.g. to the parse command. The option -tree interprets the",
|
|
||||||
"input as a tree, which can be given e.g. to the linearize command.",
|
|
||||||
"The option -lines will result in a list of strings or trees, one by line."
|
|
||||||
],
|
|
||||||
options = [
|
|
||||||
("lines","return the list of lines, instead of the singleton of all contents"),
|
|
||||||
("tree","convert strings into trees")
|
|
||||||
],
|
|
||||||
exec = needPGF $ \opts _ env@(pgf, mos) -> do
|
|
||||||
let file = optFile opts
|
|
||||||
let exprs [] = ([],empty)
|
|
||||||
exprs ((n,s):ls) | null s
|
|
||||||
= exprs ls
|
|
||||||
exprs ((n,s):ls) = case readExpr s of
|
|
||||||
Just e -> let (es,err) = exprs ls
|
|
||||||
in case inferExpr pgf e of
|
|
||||||
Right (e,t) -> (e:es,err)
|
|
||||||
Left msg -> (es,"on line" <+> n <> ':' $$ msg $$ err)
|
|
||||||
Nothing -> let (es,err) = exprs ls
|
|
||||||
in (es,"on line" <+> n <> ':' <+> "parse error" $$ err)
|
|
||||||
returnFromLines ls = case exprs ls of
|
|
||||||
(es, err) | null es -> return $ pipeMessage $ render (err $$ "no trees found")
|
|
||||||
| otherwise -> return $ pipeWithMessage (map hsExpr es) (render err)
|
|
||||||
|
|
||||||
s <- restricted $ readFile file
|
|
||||||
case opts of
|
|
||||||
_ | isOpt "lines" opts && isOpt "tree" opts ->
|
|
||||||
returnFromLines (zip [1::Int ..] (lines s))
|
|
||||||
_ | isOpt "tree" opts ->
|
|
||||||
returnFromLines [(1::Int,s)]
|
|
||||||
_ | isOpt "lines" opts -> return (fromStrings $ lines s)
|
|
||||||
_ -> return (fromString s),
|
|
||||||
flags = [("file","the input file name")]
|
|
||||||
}),
|
|
||||||
("rt", emptyCommandInfo {
|
|
||||||
longname = "rank_trees",
|
|
||||||
synopsis = "show trees in an order of decreasing probability",
|
|
||||||
explanation = unlines [
|
|
||||||
"Order trees from the most to the least probable, using either",
|
|
||||||
"even distribution in each category (default) or biased as specified",
|
|
||||||
"by the file given by flag -probs=FILE, where each line has the form",
|
|
||||||
"'function probability', e.g. 'youPol_Pron 0.01'."
|
|
||||||
],
|
|
||||||
exec = needPGF $ \opts es env@(pgf, _) -> do
|
|
||||||
let tds = sortBy (\(_,p) (_,q) -> compare p q)
|
|
||||||
[(t, treeProbability pgf t) | t <- map cExpr (toExprs es)]
|
|
||||||
if isOpt "v" opts
|
|
||||||
then putStrLn $
|
|
||||||
unlines [PGF2.showExpr [] t ++ "\t--" ++ show d | (t,d) <- tds]
|
|
||||||
else return ()
|
|
||||||
returnFromExprs $ map (hsExpr . fst) tds,
|
|
||||||
flags = [
|
|
||||||
("probs","probabilities from this file (format 'f 0.6' per line)")
|
|
||||||
],
|
|
||||||
options = [
|
|
||||||
("v","show all trees with their probability scores")
|
|
||||||
],
|
|
||||||
examples = [
|
|
||||||
mkEx "p \"you are here\" | rt -probs=probs | pt -number=1 -- most probable result"
|
|
||||||
]
|
|
||||||
}),
|
|
||||||
{-
|
|
||||||
("tq", emptyCommandInfo {
|
|
||||||
longname = "translation_quiz",
|
|
||||||
syntax = "tq -from=LANG -to=LANG (-cat=CAT)? (-probs=FILE)? TREE?",
|
|
||||||
synopsis = "start a translation quiz",
|
|
||||||
exec = \env@(pgf, mos) opts xs -> do
|
|
||||||
let from = optLangFlag "from" pgf opts
|
|
||||||
let to = optLangFlag "to" pgf opts
|
|
||||||
let typ = optType pgf opts
|
|
||||||
let mt = mexp xs
|
|
||||||
pgf <- optProbs opts pgf
|
|
||||||
restricted $ translationQuiz mt pgf from to typ
|
|
||||||
return void,
|
|
||||||
flags = [
|
|
||||||
("from","translate from this language"),
|
|
||||||
("to","translate to this language"),
|
|
||||||
("cat","translate in this category"),
|
|
||||||
("number","the maximum number of questions"),
|
|
||||||
("probs","file with biased probabilities for generation")
|
|
||||||
],
|
|
||||||
examples = [
|
|
||||||
mkEx ("tq -from=Eng -to=Swe -- any trees in startcat"),
|
|
||||||
mkEx ("tq -from=Eng -to=Swe (AdjCN (PositA ?2) (UseN ?)) -- only trees of this form")
|
|
||||||
]
|
|
||||||
}),
|
|
||||||
("vd", emptyCommandInfo {
|
|
||||||
longname = "visualize_dependency",
|
|
||||||
synopsis = "show word dependency tree graphically",
|
|
||||||
explanation = unlines [
|
|
||||||
"Prints a dependency tree in the .dot format (the graphviz format, default)",
|
|
||||||
"or the CoNLL/MaltParser format (flag -output=conll for training, malt_input",
|
|
||||||
"for unanalysed input).",
|
|
||||||
"By default, the last argument is the head of every abstract syntax",
|
|
||||||
"function; moreover, the head depends on the head of the function above.",
|
|
||||||
"The graph can be saved in a file by the wf command as usual.",
|
|
||||||
"If the -view flag is defined, the graph is saved in a temporary file",
|
|
||||||
"which is processed by graphviz and displayed by the program indicated",
|
|
||||||
"by the flag. The target format is png, unless overridden by the",
|
|
||||||
"flag -format."
|
|
||||||
],
|
|
||||||
exec = \env@(pgf, mos) opts es -> do
|
|
||||||
let debug = isOpt "v" opts
|
|
||||||
let file = valStrOpts "file" "" opts
|
|
||||||
let outp = valStrOpts "output" "dot" opts
|
|
||||||
mlab <- case file of
|
|
||||||
"" -> return Nothing
|
|
||||||
_ -> (Just . H.getDepLabels . lines) `fmap` restricted (readFile file)
|
|
||||||
let lang = optLang pgf opts
|
|
||||||
let grphs = unlines $ map (H.graphvizDependencyTree outp debug mlab Nothing pgf lang) es
|
|
||||||
if isFlag "view" opts || isFlag "format" opts then do
|
|
||||||
let file s = "_grphd." ++ s
|
|
||||||
let view = optViewGraph opts
|
|
||||||
let format = optViewFormat opts
|
|
||||||
restricted $ writeUTF8File (file "dot") grphs
|
|
||||||
restrictedSystem $ "dot -T" ++ format ++ " " ++ file "dot" ++ " > " ++ file format
|
|
||||||
restrictedSystem $ view ++ " " ++ file format
|
|
||||||
return void
|
|
||||||
else return $ fromString grphs,
|
|
||||||
examples = [
|
|
||||||
mkEx "gr | vd -- generate a tree and show dependency tree in .dot",
|
|
||||||
mkEx "gr | vd -view=open -- generate a tree and display dependency tree on a Mac",
|
|
||||||
mkEx "gr -number=1000 | vd -file=dep.labels -output=malt -- generate training treebank",
|
|
||||||
mkEx "gr -number=100 | vd -file=dep.labels -output=malt_input -- generate test sentences"
|
|
||||||
],
|
|
||||||
options = [
|
|
||||||
("v","show extra information")
|
|
||||||
],
|
|
||||||
flags = [
|
|
||||||
("file","configuration file for labels per fun, format 'fun l1 ... label ... l2'"),
|
|
||||||
("format","format of the visualization file (default \"png\")"),
|
|
||||||
("output","output format of graph source (default \"dot\")"),
|
|
||||||
("view","program to open the resulting file (default \"open\")"),
|
|
||||||
("lang","the language of analysis")
|
|
||||||
]
|
|
||||||
}),
|
|
||||||
-}
|
|
||||||
|
|
||||||
("vp", emptyCommandInfo {
|
|
||||||
longname = "visualize_parse",
|
|
||||||
synopsis = "show parse tree graphically",
|
|
||||||
explanation = unlines [
|
|
||||||
"Prints a parse tree in the .dot format (the graphviz format).",
|
|
||||||
"The graph can be saved in a file by the wf command as usual.",
|
|
||||||
"If the -view flag is defined, the graph is saved in a temporary file",
|
|
||||||
"which is processed by graphviz and displayed by the program indicated",
|
|
||||||
"by the flag. The target format is png, unless overridden by the",
|
|
||||||
"flag -format."
|
|
||||||
],
|
|
||||||
exec = needPGF $ \opts arg env@(pgf, concs) ->
|
|
||||||
do let es = toExprs arg
|
|
||||||
let concs = optConcs env opts
|
|
||||||
|
|
||||||
let gvOptions=graphvizDefaults{noLeaves = isOpt "noleaves" opts && not (isOpt "showleaves" opts),
|
|
||||||
noFun = isOpt "nofun" opts || not (isOpt "showfun" opts),
|
|
||||||
noCat = isOpt "nocat" opts && not (isOpt "showcat" opts),
|
|
||||||
nodeFont = valStrOpts "nodefont" "" opts,
|
|
||||||
leafFont = valStrOpts "leaffont" "" opts,
|
|
||||||
nodeColor = valStrOpts "nodecolor" "" opts,
|
|
||||||
leafColor = valStrOpts "leafcolor" "" opts,
|
|
||||||
nodeEdgeStyle = valStrOpts "nodeedgestyle" "solid" opts,
|
|
||||||
leafEdgeStyle = valStrOpts "leafedgestyle" "dashed" opts
|
|
||||||
}
|
|
||||||
|
|
||||||
let grph= if null es || null concs
|
|
||||||
then []
|
|
||||||
else graphvizParseTree (snd (head concs)) gvOptions (cExpr (head es))
|
|
||||||
if isFlag "view" opts || isFlag "format" opts then do
|
|
||||||
let file s = "_grph." ++ s
|
|
||||||
let view = optViewGraph opts
|
|
||||||
let format = optViewFormat opts
|
|
||||||
restricted $ writeUTF8File (file "dot") grph
|
|
||||||
restrictedSystem $ "dot -T" ++ format ++ " " ++ file "dot" ++ " > " ++ file format
|
|
||||||
restrictedSystem $ view ++ " " ++ file format
|
|
||||||
return void
|
|
||||||
else return $ fromString grph,
|
|
||||||
examples = [
|
|
||||||
mkEx "p -lang=Eng \"John walks\" | vp -- generate a tree and show parse tree as .dot script",
|
|
||||||
mkEx "gr | vp -view=\"open\" -- generate a tree and display parse tree on a Mac"
|
|
||||||
],
|
|
||||||
options = [
|
|
||||||
("showcat","show categories in the tree nodes (default)"),
|
|
||||||
("nocat","don't show categories"),
|
|
||||||
("showfun","show function names in the tree nodes"),
|
|
||||||
("nofun","don't show function names (default)"),
|
|
||||||
("showleaves","show the leaves of the tree (default)"),
|
|
||||||
("noleaves","don't show the leaves of the tree (i.e., only the abstract tree)")
|
|
||||||
],
|
|
||||||
flags = [
|
|
||||||
("lang","the language to visualize"),
|
|
||||||
("format","format of the visualization file (default \"png\")"),
|
|
||||||
("view","program to open the resulting file (default \"open\")"),
|
|
||||||
("nodefont","font for tree nodes (default: Times -- graphviz standard font)"),
|
|
||||||
("leaffont","font for tree leaves (default: nodefont)"),
|
|
||||||
("nodecolor","color for tree nodes (default: black -- graphviz standard color)"),
|
|
||||||
("leafcolor","color for tree leaves (default: nodecolor)"),
|
|
||||||
("nodeedgestyle","edge style between tree nodes (solid/dashed/dotted/bold, default: solid)"),
|
|
||||||
("leafedgestyle","edge style for links to leaves (solid/dashed/dotted/bold, default: dashed)")
|
|
||||||
]
|
|
||||||
}),
|
|
||||||
|
|
||||||
("vt", emptyCommandInfo {
|
|
||||||
longname = "visualize_tree",
|
|
||||||
synopsis = "show a set of trees graphically",
|
|
||||||
explanation = unlines [
|
|
||||||
"Prints a set of trees in the .dot format (the graphviz format).",
|
|
||||||
"The graph can be saved in a file by the wf command as usual.",
|
|
||||||
"If the -view flag is defined, the graph is saved in a temporary file",
|
|
||||||
"which is processed by graphviz and displayed by the program indicated",
|
|
||||||
"by the flag. The target format is postscript, unless overridden by the",
|
|
||||||
"flag -format."
|
|
||||||
],
|
|
||||||
exec = needPGF $ \opts arg env@(pgf, _) ->
|
|
||||||
let es = toExprs arg in
|
|
||||||
if isOpt "api" opts
|
|
||||||
then do
|
|
||||||
mapM_ (putStrLn . exprToAPI) es
|
|
||||||
return void
|
|
||||||
else do
|
|
||||||
let gvOptions=graphvizDefaults{noFun = isOpt "nofun" opts,
|
|
||||||
noCat = isOpt "nocat" opts,
|
|
||||||
nodeFont = valStrOpts "nodefont" "" opts,
|
|
||||||
nodeColor = valStrOpts "nodecolor" "" opts,
|
|
||||||
nodeEdgeStyle = valStrOpts "nodeedgestyle" "solid" opts
|
|
||||||
}
|
|
||||||
let grph = unlines (map (graphvizAbstractTree pgf gvOptions . cExpr) es)
|
|
||||||
if isFlag "view" opts || isFlag "format" opts then do
|
|
||||||
let file s = "_grph." ++ s
|
|
||||||
let view = optViewGraph opts
|
|
||||||
let format = optViewFormat opts
|
|
||||||
restricted $ writeUTF8File (file "dot") grph
|
|
||||||
restrictedSystem $ "dot -T" ++ format ++ " " ++ file "dot" ++ " > " ++ file format
|
|
||||||
restrictedSystem $ view ++ " " ++ file format
|
|
||||||
return void
|
|
||||||
else return $ fromString grph,
|
|
||||||
examples = [
|
|
||||||
mkEx "p \"hello\" | vt -- parse a string and show trees as graph script",
|
|
||||||
mkEx "p \"hello\" | vt -view=\"open\" -- parse a string and display trees on a Mac"
|
|
||||||
],
|
|
||||||
options = [
|
|
||||||
("api", "show the tree with function names converted to 'mkC' with value cats C"),
|
|
||||||
("nofun","don't show functions but only categories"),
|
|
||||||
("nocat","don't show categories but only functions")
|
|
||||||
],
|
|
||||||
flags = [
|
|
||||||
("format","format of the visualization file (default \"png\")"),
|
|
||||||
("view","program to open the resulting file (default \"open\")"),
|
|
||||||
("nodefont","font for tree nodes (default: Times -- graphviz standard font)"),
|
|
||||||
("nodecolor","color for tree nodes (default: black -- graphviz standard color)"),
|
|
||||||
("nodeedgestyle","edge style between tree nodes (solid/dashed/dotted/bold, default: solid)")
|
|
||||||
]
|
|
||||||
}),
|
|
||||||
|
|
||||||
("ai", emptyCommandInfo {
|
|
||||||
longname = "abstract_info",
|
|
||||||
syntax = "ai IDENTIFIER or ai EXPR",
|
|
||||||
synopsis = "Provides an information about a function, an expression or a category from the abstract syntax",
|
|
||||||
explanation = unlines [
|
|
||||||
"The command has one argument which is either function, expression or",
|
|
||||||
"a category defined in the abstract syntax of the current grammar. ",
|
|
||||||
"If the argument is a function then its type is printed out.",
|
|
||||||
"If it is a category then the category definition is printed.",
|
|
||||||
"If a whole expression is given it prints the expression with refined",
|
|
||||||
"metavariables and the type of the expression."
|
|
||||||
],
|
|
||||||
exec = needPGF $ \opts args env@(pgf,cncs) ->
|
|
||||||
case map cExpr (toExprs args) of
|
|
||||||
[e] -> case unApp e of
|
|
||||||
Just (id,[]) -> return (fromString
|
|
||||||
(case functionType pgf id of
|
|
||||||
Just ty -> showFun id ty
|
|
||||||
Nothing -> let funs = functionsByCat pgf id
|
|
||||||
in showCat id funs))
|
|
||||||
where
|
|
||||||
showCat c funs = "cat "++c++
|
|
||||||
" ;\n\n"++
|
|
||||||
unlines [showFun f ty| f<-funs,
|
|
||||||
Just ty <- [functionType pgf f]]
|
|
||||||
showFun f ty = "fun "++f++" : "++showType [] ty++" ;"
|
|
||||||
_ -> case inferExpr pgf e of
|
|
||||||
Left msg -> error msg
|
|
||||||
Right (e,ty) -> do putStrLn ("Expression: "++PGF2.showExpr [] e)
|
|
||||||
putStrLn ("Type: "++PGF2.showType [] ty)
|
|
||||||
putStrLn ("Probability: "++show (treeProbability pgf e))
|
|
||||||
return void
|
|
||||||
_ -> do putStrLn "a single function name or category name is expected"
|
|
||||||
return void,
|
|
||||||
needsTypeCheck = False
|
|
||||||
})
|
|
||||||
]
|
|
||||||
where
|
|
||||||
cParse env@(pgf,_) opts ss =
|
|
||||||
parsed [ parse cnc cat s | s<-ss,(lang,cnc)<-cncs]
|
|
||||||
where
|
|
||||||
cat = optType pgf opts
|
|
||||||
cncs = optConcs env opts
|
|
||||||
parsed rs = Piped (Exprs ts,unlines msgs)
|
|
||||||
where
|
|
||||||
ts = [hsExpr t|ParseOk ts<-rs,(t,p)<-takeOptNum opts ts]
|
|
||||||
msgs = concatMap mkMsg rs
|
|
||||||
|
|
||||||
mkMsg (ParseOk ts) = (map (PGF2.showExpr [] . fst).takeOptNum opts) ts
|
|
||||||
mkMsg (ParseFailed _ tok) = ["Parse failed: "++tok]
|
|
||||||
mkMsg (ParseIncomplete) = ["The sentence is incomplete"]
|
|
||||||
|
|
||||||
optLins env opts ts = case opts of
|
|
||||||
_ | isOpt "groups" opts ->
|
|
||||||
concatMap snd $ groupResults
|
|
||||||
[[(lang, s) | (lang,concr) <- optConcs env opts,s <- linear opts lang concr t] | t <- ts]
|
|
||||||
_ -> concatMap (optLin env opts) ts
|
|
||||||
optLin env@(pgf,_) opts t =
|
|
||||||
case opts of
|
|
||||||
_ | isOpt "treebank" opts ->
|
|
||||||
(abstractName pgf ++ ": " ++ PGF2.showExpr [] t) :
|
|
||||||
[lang ++ ": " ++ s | (lang,concr) <- optConcs env opts, s<-linear opts lang concr t]
|
|
||||||
_ -> [s | (lang,concr) <- optConcs env opts, s<-linear opts lang concr t]
|
|
||||||
|
|
||||||
linear :: [Option] -> ConcName -> Concr -> PGF2.Expr -> [String]
|
|
||||||
linear opts lang concr = case opts of
|
|
||||||
_ | isOpt "all" opts -> concat . map (map snd) . tabularLinearizeAll concr
|
|
||||||
_ | isOpt "list" opts -> (:[]) . commaList .
|
|
||||||
concatMap (map snd) . tabularLinearizeAll concr
|
|
||||||
_ | isOpt "table" opts -> concatMap (map (\(p,v) -> p+++":"+++v)) . tabularLinearizeAll concr
|
|
||||||
_ | isOpt "bracket" opts -> (:[]) . unwords . map showBracketedString . bracketedLinearize concr
|
|
||||||
_ -> (:[]) . linearize concr
|
|
||||||
|
|
||||||
groupResults :: [[(ConcName,String)]] -> [(ConcName,[String])]
|
|
||||||
groupResults = Map.toList . foldr more Map.empty . start . concat
|
|
||||||
where
|
|
||||||
start ls = [(l,[s]) | (l,s) <- ls]
|
|
||||||
more (l,s) =
|
|
||||||
Map.insertWith (\ [x] xs -> if elem x xs then xs else (x : xs)) l s
|
|
||||||
|
|
||||||
optConcs = optConcsFlag "lang"
|
|
||||||
|
|
||||||
optConcsFlag f (pgf,cncs) opts =
|
|
||||||
case valStrOpts f "" opts of
|
|
||||||
"" -> Map.toList cncs
|
|
||||||
lang -> mapMaybe pickLang (chunks ',' lang)
|
|
||||||
where
|
|
||||||
pickLang l = pick l `mplus` pick fl
|
|
||||||
where
|
|
||||||
fl = abstractName pgf++l
|
|
||||||
pick l = (,) l `fmap` Map.lookup l cncs
|
|
||||||
|
|
||||||
{-
|
|
||||||
-- replace each non-atomic constructor with mkC, where C is the val cat
|
|
||||||
tree2mk pgf = H.showExpr [] . t2m where
|
|
||||||
t2m t = case H.unApp t of
|
|
||||||
Just (cid,ts@(_:_)) -> H.mkApp (mk cid) (map t2m ts)
|
|
||||||
_ -> t
|
|
||||||
mk = H.mkCId . ("mk" ++) . H.showCId . H.lookValCat (H.abstract pgf)
|
|
||||||
|
|
||||||
unlex opts lang = stringOps Nothing (getUnlex opts lang ++ map prOpt opts) ----
|
|
||||||
|
|
||||||
getUnlex opts lang = case words (valStrOpts "unlexer" "" opts) of
|
|
||||||
lexs -> case lookup lang
|
|
||||||
[(H.mkCId la,tail le) | lex <- lexs, let (la,le) = span (/='=') lex, not (null le)] of
|
|
||||||
Just le -> chunks ',' le
|
|
||||||
_ -> []
|
|
||||||
-}
|
|
||||||
commaList [] = []
|
|
||||||
commaList ws = concat $ head ws : map (", " ++) (tail ws)
|
|
||||||
|
|
||||||
optFile opts = valStrOpts "file" "_gftmp" opts
|
|
||||||
|
|
||||||
optType pgf opts =
|
|
||||||
case listFlags "cat" opts of
|
|
||||||
v:_ -> let str = valueString v
|
|
||||||
in case readType str of
|
|
||||||
Just ty -> case checkType pgf ty of
|
|
||||||
Left msg -> error msg
|
|
||||||
Right ty -> ty
|
|
||||||
Nothing -> error ("Can't parse '"++str++"' as a type")
|
|
||||||
_ -> startCat pgf
|
|
||||||
|
|
||||||
optViewFormat opts = valStrOpts "format" "png" opts
|
|
||||||
optViewGraph opts = valStrOpts "view" "open" opts
|
|
||||||
{-
|
|
||||||
optNum opts = valIntOpts "number" 1 opts
|
|
||||||
-}
|
|
||||||
optNumInf opts = valIntOpts "number" 1000000000 opts ---- 10^9
|
|
||||||
takeOptNum opts = take (optNumInf opts)
|
|
||||||
|
|
||||||
returnFromCExprs = returnFromExprs . map hsExpr
|
|
||||||
returnFromExprs es =
|
|
||||||
return $ case es of
|
|
||||||
[] -> pipeMessage "no trees found"
|
|
||||||
_ -> fromExprs es
|
|
||||||
|
|
||||||
prGrammar env@(pgf,cncs) opts
|
|
||||||
| isOpt "langs" opts = return . fromString . unwords $ (map fst (optConcs env opts))
|
|
||||||
| isOpt "cats" opts = return . fromString . unwords $ categories pgf
|
|
||||||
| isOpt "funs" opts = return . fromString . unwords $ functions pgf
|
|
||||||
| isOpt "missing" opts = return . fromString . unwords $
|
|
||||||
[f | f <- functions pgf, not (and [hasLinearization concr f | (_,concr) <- optConcs env opts])]
|
|
||||||
| isOpt "fullform" opts = return $ fromString $ concatMap (prFullFormLexicon . snd) $ optConcs env opts
|
|
||||||
| isOpt "words" opts = return $ fromString $ concatMap (prAllWords . snd) $ optConcs env opts
|
|
||||||
| isOpt "lexc" opts = return $ fromString $ concatMap (prLexcLexicon . snd) $ optConcs env opts
|
|
||||||
| otherwise = return void
|
|
||||||
|
|
||||||
gizaAlignment pgf src_cnc tgt_cnc e =
|
|
||||||
let src_res = alignWords src_cnc e
|
|
||||||
tgt_res = alignWords tgt_cnc e
|
|
||||||
alignment = [show i++"-"++show j | (i,(_,src_fids)) <- zip [0..] src_res, (j,(_,tgt_fids)) <- zip [0..] tgt_res, not (null (intersect src_fids tgt_fids))]
|
|
||||||
in (unwords (map fst src_res), unwords (map fst tgt_res), unwords alignment)
|
|
||||||
|
|
||||||
morphos env opts s =
|
|
||||||
[(s,res) | (lang,concr) <- optConcs env opts, let res = lookupMorpho concr s, not (null res)]
|
|
||||||
{-
|
|
||||||
mexp xs = case xs of
|
|
||||||
t:_ -> Just t
|
|
||||||
_ -> Nothing
|
|
||||||
-}
|
|
||||||
-- ps -f -g s returns g (f s)
|
|
||||||
{-
|
|
||||||
treeOps pgf opts s = foldr app s (reverse opts) where
|
|
||||||
app (OOpt op) | Just (Left f) <- treeOp pgf op = f
|
|
||||||
app (OFlag op (VId x)) | Just (Right f) <- treeOp pgf op = f (H.mkCId x)
|
|
||||||
app _ = id
|
|
||||||
|
|
||||||
treeOpOptions pgf = [(op,expl) | (op,(expl,Left _)) <- allTreeOps pgf]
|
|
||||||
treeOpFlags pgf = [(op,expl) | (op,(expl,Right _)) <- allTreeOps pgf]
|
|
||||||
|
|
||||||
translationQuiz :: Maybe H.Expr -> H.PGF -> H.Language -> H.Language -> H.Type -> IO ()
|
|
||||||
translationQuiz mex pgf ig og typ = do
|
|
||||||
tts <- translationList mex pgf ig og typ infinity
|
|
||||||
mkQuiz "Welcome to GF Translation Quiz." tts
|
|
||||||
|
|
||||||
morphologyQuiz :: Maybe H.Expr -> H.PGF -> H.Language -> H.Type -> IO ()
|
|
||||||
morphologyQuiz mex pgf ig typ = do
|
|
||||||
tts <- morphologyList mex pgf ig typ infinity
|
|
||||||
mkQuiz "Welcome to GF Morphology Quiz." tts
|
|
||||||
|
|
||||||
-- | the maximal number of precompiled quiz problems
|
|
||||||
infinity :: Int
|
|
||||||
infinity = 256
|
|
||||||
-}
|
|
||||||
prLexcLexicon :: Concr -> String
|
|
||||||
prLexcLexicon concr =
|
|
||||||
unlines $ "Multichar_Symbols":multichars:"":"LEXICON Root" : [prLexc l p ++ ":" ++ w ++ " # ;" | (w,lps) <- morpho, (l,p,_) <- lps] ++ ["END"]
|
|
||||||
where
|
|
||||||
morpho = fullFormLexicon concr
|
|
||||||
prLexc l p = l ++ concat (mkTags (words p))
|
|
||||||
mkTags p = case p of
|
|
||||||
"s":ws -> mkTags ws --- remove record field
|
|
||||||
ws -> map ('+':) ws
|
|
||||||
|
|
||||||
multichars = unwords $ nub $ concat [mkTags (words p) | (w,lps) <- morpho, (l,p,_) <- lps]
|
|
||||||
-- thick_A+(AAdj+Posit+Gen):thick's # ;
|
|
||||||
|
|
||||||
prFullFormLexicon :: Concr -> String
|
|
||||||
prFullFormLexicon concr =
|
|
||||||
unlines (map prMorphoAnalysis (fullFormLexicon concr))
|
|
||||||
|
|
||||||
prAllWords :: Concr -> String
|
|
||||||
prAllWords concr =
|
|
||||||
unwords [w | (w,_) <- fullFormLexicon concr]
|
|
||||||
|
|
||||||
prMorphoAnalysis :: (String,[MorphoAnalysis]) -> String
|
|
||||||
prMorphoAnalysis (w,lps) =
|
|
||||||
unlines (w:[fun ++ " : " ++ cat | (fun,cat,p) <- lps])
|
|
||||||
|
|
||||||
hsExpr c =
|
|
||||||
case unApp c of
|
|
||||||
Just (f,cs) -> H.mkApp (H.mkCId f) (map hsExpr cs)
|
|
||||||
_ -> case unStr c of
|
|
||||||
Just str -> H.mkStr str
|
|
||||||
_ -> error $ "GF.Command.Commands2.hsExpr "++show c
|
|
||||||
|
|
||||||
cExpr e =
|
|
||||||
case H.unApp e of
|
|
||||||
Just (f,es) -> mkApp (H.showCId f) (map cExpr es)
|
|
||||||
_ -> case H.unStr e of
|
|
||||||
Just str -> mkStr str
|
|
||||||
_ -> error $ "GF.Command.Commands2.cExpr "++show e
|
|
||||||
|
|
||||||
needPGF exec opts ts =
|
|
||||||
do Env mb_pgf cncs <- getPGFEnv
|
|
||||||
case mb_pgf of
|
|
||||||
Just pgf -> liftSIO $ exec opts ts (pgf,cncs)
|
|
||||||
_ -> fail "Import a grammar before using this command"
|
|
||||||
@@ -3,7 +3,6 @@
|
|||||||
-- elsewhere
|
-- elsewhere
|
||||||
module GF.Command.CommonCommands where
|
module GF.Command.CommonCommands where
|
||||||
import Data.List(sort)
|
import Data.List(sort)
|
||||||
import Data.Char (isSpace)
|
|
||||||
import GF.Command.CommandInfo
|
import GF.Command.CommandInfo
|
||||||
import qualified Data.Map as Map
|
import qualified Data.Map as Map
|
||||||
import GF.Infra.SIO
|
import GF.Infra.SIO
|
||||||
@@ -16,7 +15,7 @@ import GF.Text.Pretty
|
|||||||
import GF.Text.Transliterations
|
import GF.Text.Transliterations
|
||||||
import GF.Text.Lexing(stringOp,opInEnv)
|
import GF.Text.Lexing(stringOp,opInEnv)
|
||||||
|
|
||||||
import qualified PGF as H(showCId,showExpr,toATree,toTrie,Trie(..))
|
import PGF2(showExpr)
|
||||||
|
|
||||||
extend old new = Map.union (Map.fromList new) old -- Map.union is left-biased
|
extend old new = Map.union (Map.fromList new) old -- Map.union is left-biased
|
||||||
|
|
||||||
@@ -102,9 +101,7 @@ commonCommands = fmap (mapCommandExec liftSIO) $ Map.fromList [
|
|||||||
"To see transliteration tables, use command ut."
|
"To see transliteration tables, use command ut."
|
||||||
],
|
],
|
||||||
examples = [
|
examples = [
|
||||||
-- mkEx "l (EAdd 3 4) | ps -code -- linearize code-like output",
|
|
||||||
mkEx "l (EAdd 3 4) | ps -unlexcode -- linearize code-like output",
|
mkEx "l (EAdd 3 4) | ps -unlexcode -- linearize code-like output",
|
||||||
-- mkEx "ps -lexer=code | p -cat=Exp -- parse code-like input",
|
|
||||||
mkEx "ps -lexcode | p -cat=Exp -- parse code-like input",
|
mkEx "ps -lexcode | p -cat=Exp -- parse code-like input",
|
||||||
mkEx "gr -cat=QCl | l | ps -bind -- linearization output from LangFin",
|
mkEx "gr -cat=QCl | l | ps -bind -- linearization output from LangFin",
|
||||||
mkEx "ps -to_devanagari \"A-p\" -- show Devanagari in UTF8 terminal",
|
mkEx "ps -to_devanagari \"A-p\" -- show Devanagari in UTF8 terminal",
|
||||||
@@ -117,13 +114,11 @@ commonCommands = fmap (mapCommandExec liftSIO) $ Map.fromList [
|
|||||||
let (os,fs) = optsAndFlags opts
|
let (os,fs) = optsAndFlags opts
|
||||||
trans <- optTranslit opts
|
trans <- optTranslit opts
|
||||||
|
|
||||||
case opts of
|
if isOpt "lines" opts
|
||||||
_ | isOpt "lines" opts -> return $ fromStrings $ map (trans . stringOps (envFlag fs) (map prOpt os)) $ toStrings x
|
then return $ fromStrings $ map (trans . stringOps (envFlag fs) (map prOpt os)) $ toStrings x
|
||||||
_ | isOpt "paragraphs" opts -> return $ fromStrings $ map (trans . stringOps (envFlag fs) (map prOpt os)) $ toParagraphs $ toStrings x
|
else return ((fromString . trans . stringOps (envFlag fs) (map prOpt os) . toString) x),
|
||||||
_ -> return ((fromString . trans . stringOps (envFlag fs) (map prOpt os) . toString) x),
|
|
||||||
options = [
|
options = [
|
||||||
("lines","apply the operation separately to each input line, returning a list of lines"),
|
("lines","apply the operation separately to each input line, returning a list of lines")
|
||||||
("paragraphs","apply separately to each input paragraph (as separated by empty lines), returning a list of lines")
|
|
||||||
] ++
|
] ++
|
||||||
stringOpOptions,
|
stringOpOptions,
|
||||||
flags = [
|
flags = [
|
||||||
@@ -178,12 +173,6 @@ commonCommands = fmap (mapCommandExec liftSIO) $ Map.fromList [
|
|||||||
mkEx "gt | l | ? wc -- generate trees, linearize, and count words"
|
mkEx "gt | l | ? wc -- generate trees, linearize, and count words"
|
||||||
]
|
]
|
||||||
}),
|
}),
|
||||||
("tt", emptyCommandInfo {
|
|
||||||
longname = "to_trie",
|
|
||||||
syntax = "to_trie",
|
|
||||||
synopsis = "combine a list of trees into a trie",
|
|
||||||
exec = \ _ -> return . fromString . trie . toExprs
|
|
||||||
}),
|
|
||||||
("ut", emptyCommandInfo {
|
("ut", emptyCommandInfo {
|
||||||
longname = "unicode_table",
|
longname = "unicode_table",
|
||||||
synopsis = "show a transliteration table for a unicode character set",
|
synopsis = "show a transliteration table for a unicode character set",
|
||||||
@@ -231,7 +220,6 @@ envFlag fs =
|
|||||||
_ -> Nothing
|
_ -> Nothing
|
||||||
|
|
||||||
stringOpOptions = sort $ [
|
stringOpOptions = sort $ [
|
||||||
("bind","bind tokens separated by Prelude.BIND, i.e. &+"),
|
|
||||||
("chars","lexer that makes every non-space character a token"),
|
("chars","lexer that makes every non-space character a token"),
|
||||||
("from_cp1251","decode from cp1251 (Cyrillic used in Bulgarian resource)"),
|
("from_cp1251","decode from cp1251 (Cyrillic used in Bulgarian resource)"),
|
||||||
("from_utf8","decode from utf8 (default)"),
|
("from_utf8","decode from utf8 (default)"),
|
||||||
@@ -256,27 +244,6 @@ stringOpOptions = sort $ [
|
|||||||
("to_" ++ p, "from GF " ++ n ++ " transliteration to unicode")] |
|
("to_" ++ p, "from GF " ++ n ++ " transliteration to unicode")] |
|
||||||
(p,n) <- transliterationPrintNames]
|
(p,n) <- transliterationPrintNames]
|
||||||
|
|
||||||
trie = render . pptss . H.toTrie . map H.toATree
|
|
||||||
where
|
|
||||||
pptss [ts] = "*"<+>nest 2 (ppts ts)
|
|
||||||
pptss tss = vcat [i<+>nest 2 (ppts ts)|(i,ts)<-zip [(1::Int)..] tss]
|
|
||||||
|
|
||||||
ppts = vcat . map ppt
|
|
||||||
|
|
||||||
ppt t =
|
|
||||||
case t of
|
|
||||||
H.Oth e -> pp (H.showExpr [] e)
|
|
||||||
H.Ap f [[]] -> pp (H.showCId f)
|
|
||||||
H.Ap f tss -> H.showCId f $$ nest 2 (pptss tss)
|
|
||||||
|
|
||||||
-- ** Converting command input
|
-- ** Converting command input
|
||||||
toString = unwords . toStrings
|
toString = unwords . toStrings
|
||||||
toLines = unlines . toStrings
|
toLines = unlines . toStrings
|
||||||
|
|
||||||
toParagraphs = map (unwords . words) . toParas
|
|
||||||
where
|
|
||||||
toParas ls = case break (all isSpace) ls of
|
|
||||||
([],[]) -> []
|
|
||||||
([],_:ll) -> toParas ll
|
|
||||||
(l, []) -> [unwords l]
|
|
||||||
(l, _:ll) -> unwords l : toParas ll
|
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
module GF.Command.Importing (importGrammar, importSource) where
|
module GF.Command.Importing (importGrammar, importSource) where
|
||||||
|
|
||||||
import PGF
|
import PGF2
|
||||||
import PGF.Internal(optimizePGF,unionPGF,msgUnionPGF)
|
import PGF2.Internal(unionPGF)
|
||||||
|
|
||||||
import GF.Compile
|
import GF.Compile
|
||||||
import GF.Compile.Multi (readMulti)
|
import GF.Compile.Multi (readMulti)
|
||||||
@@ -17,14 +17,16 @@ import GF.Data.ErrM
|
|||||||
|
|
||||||
import System.FilePath
|
import System.FilePath
|
||||||
import qualified Data.Set as Set
|
import qualified Data.Set as Set
|
||||||
|
import qualified Data.Map as Map
|
||||||
|
import Control.Monad(foldM)
|
||||||
|
|
||||||
-- import a grammar in an environment where it extends an existing grammar
|
-- import a grammar in an environment where it extends an existing grammar
|
||||||
importGrammar :: PGF -> Options -> [FilePath] -> IO PGF
|
importGrammar :: Maybe PGF -> Options -> [FilePath] -> IO (Maybe PGF)
|
||||||
importGrammar pgf0 _ [] = return pgf0
|
importGrammar pgf0 _ [] = return pgf0
|
||||||
importGrammar pgf0 opts files =
|
importGrammar pgf0 opts files =
|
||||||
case takeExtensions (last files) of
|
case takeExtensions (last files) of
|
||||||
".cf" -> importCF opts files getBNFCRules bnfc2cf
|
".cf" -> fmap Just $ importCF opts files getBNFCRules bnfc2cf
|
||||||
".ebnf" -> importCF opts files getEBNFRules ebnf2cf
|
".ebnf" -> fmap Just $ importCF opts files getEBNFRules ebnf2cf
|
||||||
".gfm" -> do
|
".gfm" -> do
|
||||||
ascss <- mapM readMulti files
|
ascss <- mapM readMulti files
|
||||||
let cs = concatMap snd ascss
|
let cs = concatMap snd ascss
|
||||||
@@ -36,14 +38,15 @@ importGrammar pgf0 opts files =
|
|||||||
Bad msg -> do putStrLn ('\n':'\n':msg)
|
Bad msg -> do putStrLn ('\n':'\n':msg)
|
||||||
return pgf0
|
return pgf0
|
||||||
".pgf" -> do
|
".pgf" -> do
|
||||||
pgf2 <- mapM readPGF files >>= return . foldl1 unionPGF
|
mapM readPGF files >>= foldM ioUnionPGF pgf0
|
||||||
ioUnionPGF pgf0 pgf2
|
|
||||||
ext -> die $ "Unknown filename extension: " ++ show ext
|
ext -> die $ "Unknown filename extension: " ++ show ext
|
||||||
|
|
||||||
ioUnionPGF :: PGF -> PGF -> IO PGF
|
ioUnionPGF :: Maybe PGF -> PGF -> IO (Maybe PGF)
|
||||||
ioUnionPGF one two = case msgUnionPGF one two of
|
ioUnionPGF Nothing two = return (Just two)
|
||||||
(pgf, Just msg) -> putStrLn msg >> return pgf
|
ioUnionPGF (Just one) two =
|
||||||
(pgf,_) -> return pgf
|
case unionPGF one two of
|
||||||
|
Nothing -> putStrLn "Abstract changed, previous concretes discarded." >> return (Just two)
|
||||||
|
Just pgf -> return (Just pgf)
|
||||||
|
|
||||||
importSource :: Options -> [FilePath] -> IO SourceGrammar
|
importSource :: Options -> [FilePath] -> IO SourceGrammar
|
||||||
importSource opts files = fmap (snd.snd) (batchCompile opts files)
|
importSource opts files = fmap (snd.snd) (batchCompile opts files)
|
||||||
@@ -56,7 +59,6 @@ importCF opts files get convert = impCF
|
|||||||
startCat <- case rules of
|
startCat <- case rules of
|
||||||
(Rule cat _ _ : _) -> return cat
|
(Rule cat _ _ : _) -> return cat
|
||||||
_ -> fail "empty CFG"
|
_ -> fail "empty CFG"
|
||||||
let pgf = cf2pgf (last files) (mkCFG startCat Set.empty rules)
|
probs <- maybe (return Map.empty) readProbabilitiesFromFile (flag optProbsFile opts)
|
||||||
probs <- maybe (return . defaultProbabilities) readProbabilitiesFromFile (flag optProbsFile opts) pgf
|
let pgf = cf2pgf opts (last files) (mkCFG startCat Set.empty rules) probs
|
||||||
return $ setProbabilities probs
|
return pgf
|
||||||
$ if flag optOptimizePGF opts then optimizePGF pgf else pgf
|
|
||||||
|
|||||||
@@ -6,8 +6,8 @@ module GF.Command.Interpreter (
|
|||||||
import GF.Command.CommandInfo
|
import GF.Command.CommandInfo
|
||||||
import GF.Command.Abstract
|
import GF.Command.Abstract
|
||||||
import GF.Command.Parse
|
import GF.Command.Parse
|
||||||
import PGF.Internal(Expr(..))
|
|
||||||
import GF.Infra.UseIO(putStrLnE)
|
import GF.Infra.UseIO(putStrLnE)
|
||||||
|
import PGF2
|
||||||
|
|
||||||
import Control.Monad(when)
|
import Control.Monad(when)
|
||||||
import qualified Data.Map as Map
|
import qualified Data.Map as Map
|
||||||
@@ -53,17 +53,8 @@ interpretPipe env cs = do
|
|||||||
-- | macro definition applications: replace ?i by (exps !! i)
|
-- | macro definition applications: replace ?i by (exps !! i)
|
||||||
appCommand :: CommandArguments -> Command -> Command
|
appCommand :: CommandArguments -> Command -> Command
|
||||||
appCommand args c@(Command i os arg) = case arg of
|
appCommand args c@(Command i os arg) = case arg of
|
||||||
AExpr e -> Command i os (AExpr (app e))
|
AExpr e -> Command i os (AExpr (exprSubstitute e (toExprs args)))
|
||||||
_ -> c
|
_ -> c
|
||||||
where
|
|
||||||
xs = toExprs args
|
|
||||||
|
|
||||||
app e = case e of
|
|
||||||
EAbs b x e -> EAbs b x (app e)
|
|
||||||
EApp e1 e2 -> EApp (app e1) (app e2)
|
|
||||||
ELit l -> ELit l
|
|
||||||
EMeta i -> xs !! i
|
|
||||||
EFun x -> EFun x
|
|
||||||
|
|
||||||
-- | return the trees to be sent in pipe, and the output possibly printed
|
-- | return the trees to be sent in pipe, and the output possibly printed
|
||||||
--interpret :: CommandEnv -> [Expr] -> Command -> SIO CommandOutput
|
--interpret :: CommandEnv -> [Expr] -> Command -> SIO CommandOutput
|
||||||
@@ -110,4 +101,4 @@ getCommandTrees env needsTypeCheck a args =
|
|||||||
ATerm t -> return (Term t)
|
ATerm t -> return (Term t)
|
||||||
ANoArg -> return args -- use piped
|
ANoArg -> return args -- use piped
|
||||||
where
|
where
|
||||||
one e = return (Exprs [e]) -- ignore piped
|
one e = return (Exprs [(e,0)]) -- ignore piped
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
module GF.Command.Parse(readCommandLine, pCommand) where
|
module GF.Command.Parse(readCommandLine, pCommand) where
|
||||||
|
|
||||||
import PGF(pExpr,pIdent)
|
import PGF2(pExpr,pIdent)
|
||||||
import GF.Grammar.Parser(runPartial,pTerm)
|
import GF.Grammar.Parser(runPartial,pTerm)
|
||||||
import GF.Command.Abstract
|
import GF.Command.Abstract
|
||||||
|
|
||||||
@@ -22,7 +22,7 @@ pCommandLine =
|
|||||||
pPipe = sepBy1 (skipSpaces >> pCommand) (skipSpaces >> char '|')
|
pPipe = sepBy1 (skipSpaces >> pCommand) (skipSpaces >> char '|')
|
||||||
|
|
||||||
pCommand = (do
|
pCommand = (do
|
||||||
cmd <- pIdent <++ (char '%' >> fmap ('%':) pIdent)
|
cmd <- readS_to_P pIdent <++ (char '%' >> fmap ('%':) (readS_to_P pIdent))
|
||||||
skipSpaces
|
skipSpaces
|
||||||
opts <- sepBy pOption skipSpaces
|
opts <- sepBy pOption skipSpaces
|
||||||
arg <- if getCommandOp cmd == "cc" then pArgTerm else pArgument
|
arg <- if getCommandOp cmd == "cc" then pArgTerm else pArgument
|
||||||
@@ -37,7 +37,7 @@ pCommand = (do
|
|||||||
|
|
||||||
pOption = do
|
pOption = do
|
||||||
char '-'
|
char '-'
|
||||||
flg <- pIdent
|
flg <- readS_to_P pIdent
|
||||||
option (OOpt flg) (fmap (OFlag flg) (char '=' >> pValue))
|
option (OOpt flg) (fmap (OFlag flg) (char '=' >> pValue))
|
||||||
|
|
||||||
pValue = do
|
pValue = do
|
||||||
@@ -52,9 +52,9 @@ pFilename = liftM2 (:) (satisfy isFileFirst) (munch (not . isSpace)) where
|
|||||||
|
|
||||||
pArgument =
|
pArgument =
|
||||||
option ANoArg
|
option ANoArg
|
||||||
(fmap AExpr pExpr
|
(fmap AExpr (readS_to_P pExpr)
|
||||||
<++
|
<++
|
||||||
(skipSpaces >> char '%' >> fmap AMacro pIdent))
|
(skipSpaces >> char '%' >> fmap AMacro (readS_to_P pIdent)))
|
||||||
|
|
||||||
pArgTerm = ATerm `fmap` readS_to_P sTerm
|
pArgTerm = ATerm `fmap` readS_to_P sTerm
|
||||||
where
|
where
|
||||||
|
|||||||
@@ -4,15 +4,15 @@ module GF.Command.TreeOperations (
|
|||||||
treeChunks
|
treeChunks
|
||||||
) where
|
) where
|
||||||
|
|
||||||
import PGF(Expr,PGF,CId,compute,mkApp,unApp,unapply,unMeta,exprSize,exprFunctions)
|
import PGF2(Expr,PGF,Fun,compute,mkApp,unApp,unapply,unMeta,exprSize,exprFunctions)
|
||||||
import Data.List
|
import Data.List
|
||||||
|
|
||||||
type TreeOp = [Expr] -> [Expr]
|
type TreeOp = [Expr] -> [Expr]
|
||||||
|
|
||||||
treeOp :: PGF -> String -> Maybe (Either TreeOp (CId -> TreeOp))
|
treeOp :: PGF -> String -> Maybe (Either TreeOp (Fun -> TreeOp))
|
||||||
treeOp pgf f = fmap snd $ lookup f $ allTreeOps pgf
|
treeOp pgf f = fmap snd $ lookup f $ allTreeOps pgf
|
||||||
|
|
||||||
allTreeOps :: PGF -> [(String,(String,Either TreeOp (CId -> TreeOp)))]
|
allTreeOps :: PGF -> [(String,(String,Either TreeOp (Fun -> TreeOp)))]
|
||||||
allTreeOps pgf = [
|
allTreeOps pgf = [
|
||||||
("compute",("compute by using semantic definitions (def)",
|
("compute",("compute by using semantic definitions (def)",
|
||||||
Left $ map (compute pgf))),
|
Left $ map (compute pgf))),
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
module GF.Compile (compileToPGF, link, batchCompile, srcAbsName) where
|
module GF.Compile (compileToPGF, link, batchCompile, srcAbsName) where
|
||||||
|
|
||||||
import GF.Compile.GrammarToPGF(mkCanon2pgf)
|
import GF.Compile.GrammarToPGF(grammar2PGF)
|
||||||
import GF.Compile.ReadFiles(ModEnv,getOptionsFromFile,getAllFiles,
|
import GF.Compile.ReadFiles(ModEnv,getOptionsFromFile,getAllFiles,
|
||||||
importsOfModule)
|
importsOfModule)
|
||||||
import GF.CompileOne(compileOne)
|
import GF.CompileOne(compileOne)
|
||||||
@@ -14,7 +14,7 @@ import GF.Infra.UseIO(IOE,FullPath,liftIO,getLibraryDirectory,putIfVerb,
|
|||||||
justModuleName,extendPathEnv,putStrE,putPointE)
|
justModuleName,extendPathEnv,putStrE,putPointE)
|
||||||
import GF.Data.Operations(raise,(+++),err)
|
import GF.Data.Operations(raise,(+++),err)
|
||||||
|
|
||||||
import Control.Monad(foldM,when,(<=<),filterM,liftM)
|
import Control.Monad(foldM,when,(<=<))
|
||||||
import GF.System.Directory(doesFileExist,getModificationTime)
|
import GF.System.Directory(doesFileExist,getModificationTime)
|
||||||
import System.FilePath((</>),isRelative,dropFileName)
|
import System.FilePath((</>),isRelative,dropFileName)
|
||||||
import qualified Data.Map as Map(empty,insert,elems) --lookup
|
import qualified Data.Map as Map(empty,insert,elems) --lookup
|
||||||
@@ -22,8 +22,7 @@ import Data.List(nub)
|
|||||||
import Data.Time(UTCTime)
|
import Data.Time(UTCTime)
|
||||||
import GF.Text.Pretty(render,($$),(<+>),nest)
|
import GF.Text.Pretty(render,($$),(<+>),nest)
|
||||||
|
|
||||||
import PGF.Internal(optimizePGF)
|
import PGF2(PGF,readProbabilitiesFromFile)
|
||||||
import PGF(PGF,defaultProbabilities,setProbabilities,readProbabilitiesFromFile)
|
|
||||||
|
|
||||||
-- | Compiles a number of source files and builds a 'PGF' structure for them.
|
-- | Compiles a number of source files and builds a 'PGF' structure for them.
|
||||||
-- This is a composition of 'link' and 'batchCompile'.
|
-- This is a composition of 'link' and 'batchCompile'.
|
||||||
@@ -36,11 +35,10 @@ link :: Options -> (ModuleName,Grammar) -> IOE PGF
|
|||||||
link opts (cnc,gr) =
|
link opts (cnc,gr) =
|
||||||
putPointE Normal opts "linking ... " $ do
|
putPointE Normal opts "linking ... " $ do
|
||||||
let abs = srcAbsName gr cnc
|
let abs = srcAbsName gr cnc
|
||||||
pgf <- mkCanon2pgf opts gr abs
|
probs <- liftIO (maybe (return Map.empty) readProbabilitiesFromFile (flag optProbsFile opts))
|
||||||
probs <- liftIO (maybe (return . defaultProbabilities) readProbabilitiesFromFile (flag optProbsFile opts) pgf)
|
pgf <- grammar2PGF opts gr abs probs
|
||||||
when (verbAtLeast opts Normal) $ putStrE "OK"
|
when (verbAtLeast opts Normal) $ putStrE "OK"
|
||||||
return $ setProbabilities probs
|
return pgf
|
||||||
$ if flag optOptimizePGF opts then optimizePGF pgf else pgf
|
|
||||||
|
|
||||||
-- | Returns the name of the abstract syntax corresponding to the named concrete syntax
|
-- | Returns the name of the abstract syntax corresponding to the named concrete syntax
|
||||||
srcAbsName gr cnc = err (const cnc) id $ abstractOfConcrete gr cnc
|
srcAbsName gr cnc = err (const cnc) id $ abstractOfConcrete gr cnc
|
||||||
@@ -78,14 +76,10 @@ compileModule opts1 env@(_,rfs) file =
|
|||||||
do file <- getRealFile file
|
do file <- getRealFile file
|
||||||
opts0 <- getOptionsFromFile file
|
opts0 <- getOptionsFromFile file
|
||||||
let curr_dir = dropFileName file
|
let curr_dir = dropFileName file
|
||||||
lib_dirs <- getLibraryDirectory (addOptions opts0 opts1)
|
lib_dir <- getLibraryDirectory (addOptions opts0 opts1)
|
||||||
let opts = addOptions (fixRelativeLibPaths curr_dir lib_dirs opts0) opts1
|
let opts = addOptions (fixRelativeLibPaths curr_dir lib_dir opts0) opts1
|
||||||
-- putIfVerb opts $ "curr_dir:" +++ show curr_dir ----
|
|
||||||
-- putIfVerb opts $ "lib_dir:" +++ show lib_dirs ----
|
|
||||||
ps0 <- extendPathEnv opts
|
ps0 <- extendPathEnv opts
|
||||||
let ps = nub (curr_dir : ps0)
|
let ps = nub (curr_dir : ps0)
|
||||||
-- putIfVerb opts $ "options from file: " ++ show opts0
|
|
||||||
-- putIfVerb opts $ "augmented options: " ++ show opts
|
|
||||||
putIfVerb opts $ "module search path:" +++ show ps ----
|
putIfVerb opts $ "module search path:" +++ show ps ----
|
||||||
files <- getAllFiles opts ps rfs file
|
files <- getAllFiles opts ps rfs file
|
||||||
putIfVerb opts $ "files to read:" +++ show files ----
|
putIfVerb opts $ "files to read:" +++ show files ----
|
||||||
@@ -98,17 +92,13 @@ compileModule opts1 env@(_,rfs) file =
|
|||||||
if exists
|
if exists
|
||||||
then return file
|
then return file
|
||||||
else if isRelative file
|
else if isRelative file
|
||||||
then do
|
then do lib_dir <- getLibraryDirectory opts1
|
||||||
lib_dirs <- getLibraryDirectory opts1
|
let file1 = lib_dir </> file
|
||||||
let candidates = [ lib_dir </> file | lib_dir <- lib_dirs ]
|
exists <- doesFileExist file1
|
||||||
putIfVerb opts1 (render ("looking for: " $$ nest 2 candidates))
|
if exists
|
||||||
file1s <- filterM doesFileExist candidates
|
then return file1
|
||||||
case length file1s of
|
else raise (render ("None of these files exists:" $$ nest 2 (file $$ file1)))
|
||||||
0 -> raise (render ("Unable to find: " $$ nest 2 candidates))
|
else raise (render ("File" <+> file <+> "does not exist."))
|
||||||
1 -> do return $ head file1s
|
|
||||||
_ -> do putIfVerb opts1 ("matched multiple candidates: " +++ show file1s)
|
|
||||||
return $ head file1s
|
|
||||||
else raise (render ("File" <+> file <+> "does not exist"))
|
|
||||||
|
|
||||||
compileOne' :: Options -> CompileEnv -> FullPath -> IOE CompileEnv
|
compileOne' :: Options -> CompileEnv -> FullPath -> IOE CompileEnv
|
||||||
compileOne' opts env@(gr,_) = extendCompileEnv env <=< compileOne opts gr
|
compileOne' opts env@(gr,_) = extendCompileEnv env <=< compileOne opts gr
|
||||||
|
|||||||
@@ -1,99 +1,110 @@
|
|||||||
{-# LANGUAGE FlexibleContexts #-}
|
{-# LANGUAGE FlexibleContexts, ImplicitParams #-}
|
||||||
module GF.Compile.CFGtoPGF (cf2pgf) where
|
module GF.Compile.CFGtoPGF (cf2pgf) where
|
||||||
|
|
||||||
import GF.Grammar.CFG
|
import GF.Grammar.CFG
|
||||||
import GF.Infra.UseIO
|
import GF.Infra.UseIO
|
||||||
|
import GF.Infra.Option
|
||||||
|
import GF.Compile.OptimizePGF
|
||||||
|
|
||||||
import PGF
|
import PGF2
|
||||||
import PGF.Internal
|
import PGF2.Internal
|
||||||
|
|
||||||
import qualified Data.Set as Set
|
import qualified Data.Set as Set
|
||||||
import qualified Data.Map as Map
|
import qualified Data.Map as Map
|
||||||
import qualified Data.IntMap as IntMap
|
import qualified Data.IntMap as IntMap
|
||||||
import Data.Array.IArray
|
import Data.Array.IArray
|
||||||
import Data.List
|
import Data.List
|
||||||
|
import Data.Maybe(fromMaybe)
|
||||||
|
|
||||||
--------------------------
|
--------------------------
|
||||||
-- the compiler ----------
|
-- the compiler ----------
|
||||||
--------------------------
|
--------------------------
|
||||||
|
|
||||||
cf2pgf :: FilePath -> ParamCFG -> PGF
|
cf2pgf :: Options -> FilePath -> ParamCFG -> Map.Map Fun Double -> PGF
|
||||||
cf2pgf fpath cf =
|
cf2pgf opts fpath cf probs =
|
||||||
let pgf = PGF Map.empty aname (cf2abstr cf) (Map.singleton cname (cf2concr cf))
|
build (let abstr = cf2abstr cf probs
|
||||||
in updateProductionIndices pgf
|
in newPGF [] aname abstr [(cname, cf2concr opts abstr cf)])
|
||||||
where
|
where
|
||||||
name = justModuleName fpath
|
name = justModuleName fpath
|
||||||
aname = mkCId (name ++ "Abs")
|
aname = name ++ "Abs"
|
||||||
cname = mkCId name
|
cname = name
|
||||||
|
|
||||||
cf2abstr :: ParamCFG -> Abstr
|
cf2abstr :: (?builder :: Builder s) => ParamCFG -> Map.Map Fun Double -> B s AbstrInfo
|
||||||
cf2abstr cfg = Abstr aflags afuns acats
|
cf2abstr cfg probs = newAbstr aflags acats afuns
|
||||||
where
|
where
|
||||||
aflags = Map.singleton (mkCId "startcat") (LStr (fst (cfgStartCat cfg)))
|
aflags = [("startcat", LStr (fst (cfgStartCat cfg)))]
|
||||||
|
|
||||||
acats = Map.fromList [(cat, ([], [(0,mkRuleName rule) | rule <- rules], 0))
|
acats = [(c', [], toLogProb (fromMaybe 0 (Map.lookup c' probs))) | cat <- allCats' cfg, let c' = cat2id cat]
|
||||||
| (cat,rules) <- (Map.toList . Map.fromListWith (++))
|
afuns = [(f', dTyp [hypo Explicit "_" (dTyp [] (cat2id c) []) | NonTerminal c <- ruleRhs rule] (cat2id (ruleLhs rule)) [], 0, [], toLogProb (fromMaybe 0 (Map.lookup f' funs_probs)))
|
||||||
[(cat2id cat, catRules cfg cat) |
|
| rule <- allRules cfg
|
||||||
cat <- allCats' cfg]]
|
, let f' = mkRuleName rule]
|
||||||
afuns = Map.fromList [(mkRuleName rule, (cftype [cat2id c | NonTerminal c <- ruleRhs rule] (cat2id (ruleLhs rule)), 0, Nothing, 0))
|
|
||||||
| rule <- allRules cfg]
|
|
||||||
|
|
||||||
cat2id = mkCId . fst
|
funs_probs = (Map.fromList . concat . Map.elems . fmap pad . Map.fromListWith (++))
|
||||||
|
[(cat,[(f',Map.lookup f' probs)]) | rule <- allRules cfg,
|
||||||
|
let cat = cat2id (ruleLhs rule),
|
||||||
|
let f' = mkRuleName rule]
|
||||||
|
where
|
||||||
|
pad :: [(a,Maybe Double)] -> [(a,Double)]
|
||||||
|
pad pfs = [(f,fromMaybe deflt mb_p) | (f,mb_p) <- pfs]
|
||||||
|
where
|
||||||
|
deflt = case length [f | (f,Nothing) <- pfs] of
|
||||||
|
0 -> 0
|
||||||
|
n -> max 0 ((1 - sum [d | (f,Just d) <- pfs]) / fromIntegral n)
|
||||||
|
|
||||||
cf2concr :: ParamCFG -> Concr
|
toLogProb = realToFrac . negate . log
|
||||||
cf2concr cfg = Concr Map.empty Map.empty
|
|
||||||
cncfuns lindefsrefs lindefsrefs
|
cat2id = fst
|
||||||
sequences productions
|
|
||||||
IntMap.empty Map.empty
|
cf2concr :: (?builder :: Builder s) => Options -> B s AbstrInfo -> ParamCFG -> B s ConcrInfo
|
||||||
cnccats
|
cf2concr opts abstr cfg =
|
||||||
IntMap.empty
|
let (lindefs',linrefs',productions',cncfuns',sequences',cnccats') =
|
||||||
totalCats
|
(if flag optOptimizePGF opts then optimizePGF (fst (cfgStartCat cfg)) else id)
|
||||||
|
(lindefsrefs,lindefsrefs,IntMap.toList productions,cncfuns,sequences,cnccats)
|
||||||
|
in newConcr abstr [] []
|
||||||
|
lindefs' linrefs'
|
||||||
|
productions' cncfuns'
|
||||||
|
sequences' cnccats' totalCats
|
||||||
where
|
where
|
||||||
cats = allCats' cfg
|
cats = allCats' cfg
|
||||||
rules = allRules cfg
|
rules = allRules cfg
|
||||||
|
|
||||||
sequences0 = Set.fromList (listArray (0,0) [SymCat 0 0] :
|
idSeq = [SymCat 0 0]
|
||||||
map mkSequence rules)
|
|
||||||
sequences = listArray (0,Set.size sequences0-1) (Set.toList sequences0)
|
|
||||||
|
|
||||||
idFun = CncFun wildCId (listArray (0,0) [seqid])
|
sequences0 = Set.fromList (idSeq :
|
||||||
where
|
map mkSequence rules)
|
||||||
seq = listArray (0,0) [SymCat 0 0]
|
sequences = Set.toList sequences0
|
||||||
seqid = binSearch seq sequences (bounds sequences)
|
|
||||||
|
idFun = ("_",[Set.findIndex idSeq sequences0])
|
||||||
((fun_cnt,cncfuns0),productions0) = mapAccumL (convertRule cs) (1,[idFun]) rules
|
((fun_cnt,cncfuns0),productions0) = mapAccumL (convertRule cs) (1,[idFun]) rules
|
||||||
productions = foldl addProd IntMap.empty (concat (productions0++coercions))
|
productions = foldl addProd IntMap.empty (concat (productions0++coercions))
|
||||||
cncfuns = listArray (0,fun_cnt-1) (reverse cncfuns0)
|
cncfuns = reverse cncfuns0
|
||||||
|
|
||||||
lbls = listArray (0,0) ["s"]
|
lbls = ["s"]
|
||||||
(fid,cnccats0) = (mapAccumL mkCncCat 0 . Map.toList . Map.fromListWith max)
|
(fid,cnccats) = (mapAccumL mkCncCat 0 . Map.toList . Map.fromListWith max)
|
||||||
[(c,p) | (c,ps) <- cats, p <- ps]
|
[(c,p) | (c,ps) <- cats, p <- ps]
|
||||||
((totalCats,cs), coercions) = mapAccumL mkCoercions (fid,Map.empty) cats
|
((totalCats,cs), coercions) = mapAccumL mkCoercions (fid,Map.empty) cats
|
||||||
cnccats = Map.fromList cnccats0
|
|
||||||
|
|
||||||
lindefsrefs =
|
lindefsrefs = map mkLinDefRef cats
|
||||||
IntMap.fromList (map mkLinDefRef cats)
|
|
||||||
|
|
||||||
convertRule cs (funid,funs) rule =
|
convertRule cs (funid,funs) rule =
|
||||||
let args = [PArg [] (cat2arg c) | NonTerminal c <- ruleRhs rule]
|
let args = [PArg [] (cat2arg c) | NonTerminal c <- ruleRhs rule]
|
||||||
prod = PApply funid args
|
prod = PApply funid args
|
||||||
seqid = binSearch (mkSequence rule) sequences (bounds sequences)
|
seqid = Set.findIndex (mkSequence rule) sequences0
|
||||||
fun = CncFun (mkRuleName rule) (listArray (0,0) [seqid])
|
fun = (mkRuleName rule, [seqid])
|
||||||
funid' = funid+1
|
funid' = funid+1
|
||||||
in funid' `seq` ((funid',fun:funs),let (c,ps) = ruleLhs rule in [(cat2fid c p, prod) | p <- ps])
|
in funid' `seq` ((funid',fun:funs),let (c,ps) = ruleLhs rule in [(cat2fid c p, prod) | p <- ps])
|
||||||
|
|
||||||
mkSequence rule = listArray (0,length syms-1) syms
|
mkSequence rule = snd $ mapAccumL convertSymbol 0 (ruleRhs rule)
|
||||||
where
|
where
|
||||||
syms = snd $ mapAccumL convertSymbol 0 (ruleRhs rule)
|
|
||||||
|
|
||||||
convertSymbol d (NonTerminal (c,_)) = (d+1,if c `elem` ["Int","Float","String"] then SymLit d 0 else SymCat d 0)
|
convertSymbol d (NonTerminal (c,_)) = (d+1,if c `elem` ["Int","Float","String"] then SymLit d 0 else SymCat d 0)
|
||||||
convertSymbol d (Terminal t) = (d, SymKS t)
|
convertSymbol d (Terminal t) = (d, SymKS t)
|
||||||
|
|
||||||
mkCncCat fid (cat,n)
|
mkCncCat fid (cat,n)
|
||||||
| cat == "Int" = (fid, (mkCId cat, CncCat fidInt fidInt lbls))
|
| cat == "Int" = (fid, (cat, fidInt, fidInt, lbls))
|
||||||
| cat == "Float" = (fid, (mkCId cat, CncCat fidFloat fidFloat lbls))
|
| cat == "Float" = (fid, (cat, fidFloat, fidFloat, lbls))
|
||||||
| cat == "String" = (fid, (mkCId cat, CncCat fidString fidString lbls))
|
| cat == "String" = (fid, (cat, fidString, fidString, lbls))
|
||||||
| otherwise = let fid' = fid+n+1
|
| otherwise = let fid' = fid+n+1
|
||||||
in fid' `seq` (fid', (mkCId cat,CncCat fid (fid+n) lbls))
|
in fid' `seq` (fid', (cat, fid, fid+n, lbls))
|
||||||
|
|
||||||
mkCoercions (fid,cs) c@(cat,[p]) = ((fid,cs),[])
|
mkCoercions (fid,cs) c@(cat,[p]) = ((fid,cs),[])
|
||||||
mkCoercions (fid,cs) c@(cat,ps ) =
|
mkCoercions (fid,cs) c@(cat,ps ) =
|
||||||
@@ -102,25 +113,16 @@ cf2concr cfg = Concr Map.empty Map.empty
|
|||||||
|
|
||||||
mkLinDefRef (cat,_) =
|
mkLinDefRef (cat,_) =
|
||||||
(cat2fid cat 0,[0])
|
(cat2fid cat 0,[0])
|
||||||
|
|
||||||
addProd prods (fid,prod) =
|
addProd prods (fid,prod) =
|
||||||
case IntMap.lookup fid prods of
|
case IntMap.lookup fid prods of
|
||||||
Just set -> IntMap.insert fid (Set.insert prod set) prods
|
Just set -> IntMap.insert fid (prod:set) prods
|
||||||
Nothing -> IntMap.insert fid (Set.singleton prod) prods
|
Nothing -> IntMap.insert fid [prod] prods
|
||||||
|
|
||||||
binSearch v arr (i,j)
|
|
||||||
| i <= j = case compare v (arr ! k) of
|
|
||||||
LT -> binSearch v arr (i,k-1)
|
|
||||||
EQ -> k
|
|
||||||
GT -> binSearch v arr (k+1,j)
|
|
||||||
| otherwise = error "binSearch"
|
|
||||||
where
|
|
||||||
k = (i+j) `div` 2
|
|
||||||
|
|
||||||
cat2fid cat p =
|
cat2fid cat p =
|
||||||
case Map.lookup (mkCId cat) cnccats of
|
case [start | (cat',start,_,_) <- cnccats, cat == cat'] of
|
||||||
Just (CncCat fid _ _) -> fid+p
|
(start:_) -> fid+p
|
||||||
_ -> error "cat2fid"
|
_ -> error "cat2fid"
|
||||||
|
|
||||||
cat2arg c@(cat,[p]) = cat2fid cat p
|
cat2arg c@(cat,[p]) = cat2fid cat p
|
||||||
cat2arg c@(cat,ps ) =
|
cat2arg c@(cat,ps ) =
|
||||||
@@ -131,4 +133,5 @@ cf2concr cfg = Concr Map.empty Map.empty
|
|||||||
mkRuleName rule =
|
mkRuleName rule =
|
||||||
case ruleName rule of
|
case ruleName rule of
|
||||||
CFObj n _ -> n
|
CFObj n _ -> n
|
||||||
_ -> wildCId
|
_ -> "_"
|
||||||
|
|
||||||
|
|||||||
@@ -21,7 +21,6 @@
|
|||||||
-----------------------------------------------------------------------------
|
-----------------------------------------------------------------------------
|
||||||
|
|
||||||
module GF.Compile.CheckGrammar(checkModule) where
|
module GF.Compile.CheckGrammar(checkModule) where
|
||||||
import Prelude hiding ((<>)) -- GHC 8.4.1 clash with Text.PrettyPrint
|
|
||||||
|
|
||||||
import GF.Infra.Ident
|
import GF.Infra.Ident
|
||||||
import GF.Infra.Option
|
import GF.Infra.Option
|
||||||
@@ -34,14 +33,13 @@ import qualified GF.Compile.Compute.ConcreteNew as CN
|
|||||||
import GF.Grammar
|
import GF.Grammar
|
||||||
import GF.Grammar.Lexer
|
import GF.Grammar.Lexer
|
||||||
import GF.Grammar.Lookup
|
import GF.Grammar.Lookup
|
||||||
--import GF.Grammar.Predef
|
|
||||||
--import GF.Grammar.PatternMatch
|
|
||||||
|
|
||||||
import GF.Data.Operations
|
import GF.Data.Operations
|
||||||
import GF.Infra.CheckM
|
import GF.Infra.CheckM
|
||||||
|
|
||||||
import Data.List
|
import Data.List
|
||||||
import qualified Data.Set as Set
|
import qualified Data.Set as Set
|
||||||
|
import qualified Data.Map as Map
|
||||||
import Control.Monad
|
import Control.Monad
|
||||||
import GF.Text.Pretty
|
import GF.Text.Pretty
|
||||||
|
|
||||||
@@ -59,7 +57,7 @@ checkModule opts cwd sgr mo@(m,mi) = do
|
|||||||
where
|
where
|
||||||
updateCheckInfos mo = fmap (foldl update mo) . parallelCheck . map check
|
updateCheckInfos mo = fmap (foldl update mo) . parallelCheck . map check
|
||||||
where check (i,info) = fmap ((,) i) (checkInfo opts cwd sgr mo i info)
|
where check (i,info) = fmap ((,) i) (checkInfo opts cwd sgr mo i info)
|
||||||
update mo@(m,mi) (i,info) = (m,mi{jments=updateTree (i,info) (jments mi)})
|
update mo@(m,mi) (i,info) = (m,mi{jments=Map.insert i info (jments mi)})
|
||||||
|
|
||||||
-- check if restricted inheritance modules are still coherent
|
-- check if restricted inheritance modules are still coherent
|
||||||
-- i.e. that the defs of remaining names don't depend on omitted names
|
-- i.e. that the defs of remaining names don't depend on omitted names
|
||||||
@@ -72,7 +70,7 @@ checkRestrictedInheritance cwd sgr (name,mo) = checkInModule cwd mo NoLoc empty
|
|||||||
where
|
where
|
||||||
mos = modules sgr
|
mos = modules sgr
|
||||||
checkRem ((i,m),mi) = do
|
checkRem ((i,m),mi) = do
|
||||||
let (incl,excl) = partition (isInherited mi) (map fst (tree2list (jments m)))
|
let (incl,excl) = partition (isInherited mi) (Map.keys (jments m))
|
||||||
let incld c = Set.member c (Set.fromList incl)
|
let incld c = Set.member c (Set.fromList incl)
|
||||||
let illegal c = Set.member c (Set.fromList excl)
|
let illegal c = Set.member c (Set.fromList excl)
|
||||||
let illegals = [(f,is) |
|
let illegals = [(f,is) |
|
||||||
@@ -89,10 +87,10 @@ checkCompleteGrammar opts cwd gr (am,abs) (cm,cnc) = checkInModule cwd cnc NoLoc
|
|||||||
let jsc = jments cnc
|
let jsc = jments cnc
|
||||||
|
|
||||||
-- check that all concrete constants are in abstract; build types for all lin
|
-- check that all concrete constants are in abstract; build types for all lin
|
||||||
jsc <- foldM checkCnc emptyBinTree (tree2list jsc)
|
jsc <- foldM checkCnc Map.empty (Map.toList jsc)
|
||||||
|
|
||||||
-- check that all abstract constants are in concrete; build default lin and lincats
|
-- check that all abstract constants are in concrete; build default lin and lincats
|
||||||
jsc <- foldM checkAbs jsc (tree2list jsa)
|
jsc <- foldM checkAbs jsc (Map.toList jsa)
|
||||||
|
|
||||||
return (cm,cnc{jments=jsc})
|
return (cm,cnc{jments=jsc})
|
||||||
where
|
where
|
||||||
@@ -113,17 +111,17 @@ checkCompleteGrammar opts cwd gr (am,abs) (cm,cnc) = checkInModule cwd cnc NoLoc
|
|||||||
case lookupIdent c js of
|
case lookupIdent c js of
|
||||||
Ok (AnyInd _ _) -> return js
|
Ok (AnyInd _ _) -> return js
|
||||||
Ok (CncFun ty (Just def) mn mf) ->
|
Ok (CncFun ty (Just def) mn mf) ->
|
||||||
return $ updateTree (c,CncFun ty (Just def) mn mf) js
|
return $ Map.insert c (CncFun ty (Just def) mn mf) js
|
||||||
Ok (CncFun ty Nothing mn mf) ->
|
Ok (CncFun ty Nothing mn mf) ->
|
||||||
case mb_def of
|
case mb_def of
|
||||||
Ok def -> return $ updateTree (c,CncFun ty (Just (L NoLoc def)) mn mf) js
|
Ok def -> return $ Map.insert c (CncFun ty (Just (L NoLoc def)) mn mf) js
|
||||||
Bad _ -> do noLinOf c
|
Bad _ -> do noLinOf c
|
||||||
return js
|
return js
|
||||||
_ -> do
|
_ -> do
|
||||||
case mb_def of
|
case mb_def of
|
||||||
Ok def -> do (cont,val) <- linTypeOfType gr cm ty
|
Ok def -> do (cont,val) <- linTypeOfType gr cm ty
|
||||||
let linty = (snd (valCat ty),cont,val)
|
let linty = (snd (valCat ty),cont,val)
|
||||||
return $ updateTree (c,CncFun (Just linty) (Just (L NoLoc def)) Nothing Nothing) js
|
return $ Map.insert c (CncFun (Just linty) (Just (L NoLoc def)) Nothing Nothing) js
|
||||||
Bad _ -> do noLinOf c
|
Bad _ -> do noLinOf c
|
||||||
return js
|
return js
|
||||||
where noLinOf c = checkWarn ("no linearization of" <+> c)
|
where noLinOf c = checkWarn ("no linearization of" <+> c)
|
||||||
@@ -132,24 +130,24 @@ checkCompleteGrammar opts cwd gr (am,abs) (cm,cnc) = checkInModule cwd cnc NoLoc
|
|||||||
Ok (CncCat (Just _) _ _ _ _) -> return js
|
Ok (CncCat (Just _) _ _ _ _) -> return js
|
||||||
Ok (CncCat Nothing md mr mp mpmcfg) -> do
|
Ok (CncCat Nothing md mr mp mpmcfg) -> do
|
||||||
checkWarn ("no linearization type for" <+> c <> ", inserting default {s : Str}")
|
checkWarn ("no linearization type for" <+> c <> ", inserting default {s : Str}")
|
||||||
return $ updateTree (c,CncCat (Just (L NoLoc defLinType)) md mr mp mpmcfg) js
|
return $ Map.insert c (CncCat (Just (L NoLoc defLinType)) md mr mp mpmcfg) js
|
||||||
_ -> do
|
_ -> do
|
||||||
checkWarn ("no linearization type for" <+> c <> ", inserting default {s : Str}")
|
checkWarn ("no linearization type for" <+> c <> ", inserting default {s : Str}")
|
||||||
return $ updateTree (c,CncCat (Just (L NoLoc defLinType)) Nothing Nothing Nothing Nothing) js
|
return $ Map.insert c (CncCat (Just (L NoLoc defLinType)) Nothing Nothing Nothing Nothing) js
|
||||||
_ -> return js
|
_ -> return js
|
||||||
|
|
||||||
checkCnc js i@(c,info) =
|
checkCnc js (c,info) =
|
||||||
case info of
|
case info of
|
||||||
CncFun _ d mn mf -> case lookupOrigInfo gr (am,c) of
|
CncFun _ d mn mf -> case lookupOrigInfo gr (am,c) of
|
||||||
Ok (_,AbsFun (Just (L _ ty)) _ _ _) ->
|
Ok (_,AbsFun (Just (L _ ty)) _ _ _) ->
|
||||||
do (cont,val) <- linTypeOfType gr cm ty
|
do (cont,val) <- linTypeOfType gr cm ty
|
||||||
let linty = (snd (valCat ty),cont,val)
|
let linty = (snd (valCat ty),cont,val)
|
||||||
return $ updateTree (c,CncFun (Just linty) d mn mf) js
|
return $ Map.insert c (CncFun (Just linty) d mn mf) js
|
||||||
_ -> do checkWarn ("function" <+> c <+> "is not in abstract")
|
_ -> do checkWarn ("function" <+> c <+> "is not in abstract")
|
||||||
return js
|
return js
|
||||||
CncCat {} ->
|
CncCat {} ->
|
||||||
case lookupOrigInfo gr (am,c) of
|
case lookupOrigInfo gr (am,c) of
|
||||||
Ok (_,AbsCat _) -> return $ updateTree i js
|
Ok (_,AbsCat _) -> return $ Map.insert c info js
|
||||||
{- -- This might be too pedantic:
|
{- -- This might be too pedantic:
|
||||||
Ok (_,AbsFun {}) ->
|
Ok (_,AbsFun {}) ->
|
||||||
checkError ("lincat:"<+>c<+>"is a fun, not a cat")
|
checkError ("lincat:"<+>c<+>"is a fun, not a cat")
|
||||||
@@ -157,7 +155,7 @@ checkCompleteGrammar opts cwd gr (am,abs) (cm,cnc) = checkInModule cwd cnc NoLoc
|
|||||||
_ -> do checkWarn ("category" <+> c <+> "is not in abstract")
|
_ -> do checkWarn ("category" <+> c <+> "is not in abstract")
|
||||||
return js
|
return js
|
||||||
|
|
||||||
_ -> return $ updateTree i js
|
_ -> return $ Map.insert c info js
|
||||||
|
|
||||||
|
|
||||||
-- | General Principle: only Just-values are checked.
|
-- | General Principle: only Just-values are checked.
|
||||||
@@ -261,18 +259,30 @@ checkInfo opts cwd sgr (m,mo) c info = checkInModule cwd mo NoLoc empty $ do
|
|||||||
return (ResOverload os [(y,x) | (x,y) <- tysts'])
|
return (ResOverload os [(y,x) | (x,y) <- tysts'])
|
||||||
|
|
||||||
ResParam (Just (L loc pcs)) _ -> do
|
ResParam (Just (L loc pcs)) _ -> do
|
||||||
ts <- chIn loc "parameter type" $
|
(vs,pcs) <- chIn loc "parameter type" $
|
||||||
liftM concat $ mapM mkPar pcs
|
mkParams 0 [] pcs
|
||||||
return (ResParam (Just (L loc pcs)) (Just ts))
|
return (ResParam (Just (L loc pcs)) (Just vs))
|
||||||
|
|
||||||
|
ResValue (L loc ty) _ ->
|
||||||
|
chIn loc "operation" $ do
|
||||||
|
let (_,Cn x) = typeFormCnc ty
|
||||||
|
is = case Map.lookup x (jments mo) of
|
||||||
|
Just (ResParam (Just (L _ pcs)) _) -> [i | (f,_,i) <- pcs, f == c]
|
||||||
|
_ -> []
|
||||||
|
case is of
|
||||||
|
[i] -> return (ResValue (L loc ty) i)
|
||||||
|
_ -> checkError (pp "Failed to find the value index for parameter" <+> pp c)
|
||||||
|
|
||||||
_ -> return info
|
_ -> return info
|
||||||
where
|
where
|
||||||
gr = prependModule sgr (m,mo)
|
gr = prependModule sgr (m,mo)
|
||||||
chIn loc cat = checkInModule cwd mo loc ("Happened in" <+> cat <+> c)
|
chIn loc cat = checkInModule cwd mo loc ("Happened in" <+> cat <+> c)
|
||||||
|
|
||||||
mkPar (f,co) = do
|
mkParams i vs [] = return (vs,[])
|
||||||
vs <- liftM combinations $ mapM (\(_,_,ty) -> allParamValues gr ty) co
|
mkParams i vs ((f,co,_):pcs) = do
|
||||||
return $ map (mkApp (QC (m,f))) vs
|
vs0 <- liftM combinations $ mapM (\(_,_,ty) -> allParamValues gr ty) co
|
||||||
|
(vs,pcs) <- mkParams (i + length vs0) (vs ++ map (mkApp (QC (m,f))) vs0) pcs
|
||||||
|
return (vs,(f,co,i):pcs)
|
||||||
|
|
||||||
checkUniq xss = case xss of
|
checkUniq xss = case xss of
|
||||||
x:y:xs
|
x:y:xs
|
||||||
|
|||||||
@@ -1,64 +0,0 @@
|
|||||||
module GF.Compile.Coding where
|
|
||||||
{-
|
|
||||||
import GF.Grammar.Grammar
|
|
||||||
import GF.Grammar.Macros
|
|
||||||
import GF.Text.Coding
|
|
||||||
--import GF.Infra.Option
|
|
||||||
import GF.Data.Operations
|
|
||||||
|
|
||||||
--import Data.Char
|
|
||||||
import System.IO
|
|
||||||
import qualified Data.ByteString.Char8 as BS
|
|
||||||
|
|
||||||
encodeStringsInModule :: TextEncoding -> SourceModule -> SourceModule
|
|
||||||
encodeStringsInModule enc = codeSourceModule (BS.unpack . encodeUnicode enc)
|
|
||||||
|
|
||||||
decodeStringsInModule :: TextEncoding -> SourceModule -> SourceModule
|
|
||||||
decodeStringsInModule enc mo = codeSourceModule (decodeUnicode enc . BS.pack) mo
|
|
||||||
|
|
||||||
codeSourceModule :: (String -> String) -> SourceModule -> SourceModule
|
|
||||||
codeSourceModule co (id,mo) = (id,mo{jments = mapTree codj (jments mo)})
|
|
||||||
where
|
|
||||||
codj (c,info) = case info of
|
|
||||||
ResOper pty pt -> ResOper (codeLTerms co pty) (codeLTerms co pt)
|
|
||||||
ResOverload es tyts -> ResOverload es [(codeLTerm co ty,codeLTerm co t) | (ty,t) <- tyts]
|
|
||||||
CncCat mcat mdef mref mpr mpmcfg -> CncCat mcat (codeLTerms co mdef) (codeLTerms co mref) (codeLTerms co mpr) mpmcfg
|
|
||||||
CncFun mty mt mpr mpmcfg -> CncFun mty (codeLTerms co mt) (codeLTerms co mpr) mpmcfg
|
|
||||||
_ -> info
|
|
||||||
|
|
||||||
codeLTerms co = fmap (codeLTerm co)
|
|
||||||
|
|
||||||
codeLTerm :: (String -> String) -> L Term -> L Term
|
|
||||||
codeLTerm = fmap . codeTerm
|
|
||||||
|
|
||||||
codeTerm :: (String -> String) -> Term -> Term
|
|
||||||
codeTerm co = codt
|
|
||||||
where
|
|
||||||
codt t = case t of
|
|
||||||
K s -> K (co s)
|
|
||||||
T ty cs -> T ty [(codp p,codt v) | (p,v) <- cs]
|
|
||||||
EPatt p -> EPatt (codp p)
|
|
||||||
_ -> composSafeOp codt t
|
|
||||||
|
|
||||||
codp p = case p of --- really: composOpPatt
|
|
||||||
PR rs -> PR [(l,codp p) | (l,p) <- rs]
|
|
||||||
PString s -> PString (co s)
|
|
||||||
PChars s -> PChars (co s)
|
|
||||||
PT x p -> PT x (codp p)
|
|
||||||
PAs x p -> PAs x (codp p)
|
|
||||||
PNeg p -> PNeg (codp p)
|
|
||||||
PRep p -> PRep (codp p)
|
|
||||||
PSeq p q -> PSeq (codp p) (codp q)
|
|
||||||
PAlt p q -> PAlt (codp p) (codp q)
|
|
||||||
_ -> p
|
|
||||||
|
|
||||||
-- | Run an encoding function on all string literals within the given string.
|
|
||||||
codeStringLiterals :: (String -> String) -> String -> String
|
|
||||||
codeStringLiterals _ [] = []
|
|
||||||
codeStringLiterals co ('"':cs) = '"' : inStringLiteral cs
|
|
||||||
where inStringLiteral [] = error "codeStringLiterals: unterminated string literal"
|
|
||||||
inStringLiteral ('"':ds) = '"' : codeStringLiterals co ds
|
|
||||||
inStringLiteral ('\\':d:ds) = '\\' : co [d] ++ inStringLiteral ds
|
|
||||||
inStringLiteral (d:ds) = co [d] ++ inStringLiteral ds
|
|
||||||
codeStringLiterals co (c:cs) = c : codeStringLiterals co cs
|
|
||||||
-}
|
|
||||||
@@ -5,7 +5,6 @@ module GF.Compile.Compute.ConcreteNew
|
|||||||
normalForm,
|
normalForm,
|
||||||
Value(..), Bind(..), Env, value2term, eval, vapply
|
Value(..), Bind(..), Env, value2term, eval, vapply
|
||||||
) where
|
) where
|
||||||
import Prelude hiding ((<>)) -- GHC 8.4.1 clash with Text.PrettyPrint
|
|
||||||
|
|
||||||
import GF.Grammar hiding (Env, VGen, VApp, VRecType)
|
import GF.Grammar hiding (Env, VGen, VApp, VRecType)
|
||||||
import GF.Grammar.Lookup(lookupResDefLoc,allParamValues)
|
import GF.Grammar.Lookup(lookupResDefLoc,allParamValues)
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
module GF.Compile.Compute.Value where
|
module GF.Compile.Compute.Value where
|
||||||
import GF.Grammar.Grammar(Label,Type,MetaId,Patt,QIdent)
|
import GF.Grammar.Grammar(Label,Type,MetaId,Patt,QIdent)
|
||||||
import PGF.Internal(BindType)
|
import PGF2(BindType)
|
||||||
import GF.Infra.Ident(Ident)
|
import GF.Infra.Ident(Ident)
|
||||||
import Text.Show.Functions()
|
import Text.Show.Functions()
|
||||||
import Data.Ix(Ix)
|
import Data.Ix(Ix)
|
||||||
|
|||||||
@@ -3,11 +3,7 @@ module GF.Compile.ExampleBased (
|
|||||||
configureExBased
|
configureExBased
|
||||||
) where
|
) where
|
||||||
|
|
||||||
import PGF
|
import PGF2
|
||||||
--import PGF.Probabilistic
|
|
||||||
--import PGF.Morphology
|
|
||||||
--import GF.Compile.ToAPI
|
|
||||||
|
|
||||||
import Data.List
|
import Data.List
|
||||||
|
|
||||||
parseExamplesInGrammar :: ExConfiguration -> FilePath -> IO (FilePath,[String])
|
parseExamplesInGrammar :: ExConfiguration -> FilePath -> IO (FilePath,[String])
|
||||||
@@ -37,47 +33,38 @@ convertFile conf src file = do
|
|||||||
(ex, end) = break (=='"') (tail exend)
|
(ex, end) = break (=='"') (tail exend)
|
||||||
in ((unwords (words cat),ex), tail end) -- quotes ignored
|
in ((unwords (words cat),ex), tail end) -- quotes ignored
|
||||||
pgf = resource_pgf conf
|
pgf = resource_pgf conf
|
||||||
morpho = resource_morpho conf
|
|
||||||
lang = language conf
|
lang = language conf
|
||||||
convEx (cat,ex) = do
|
convEx (cat,ex) = do
|
||||||
appn "("
|
appn "("
|
||||||
let typ = maybe (error "no valid cat") id $ readType cat
|
let typ = maybe (error "no valid cat") id $ readType cat
|
||||||
ws <- case fst (parse_ pgf lang typ (Just 4) ex) of
|
ws <- case parse lang typ ex of
|
||||||
ParseFailed _ -> do
|
ParseFailed _ _ -> do
|
||||||
let ws = morphoMissing morpho (words ex)
|
|
||||||
appv ("WARNING: cannot parse example " ++ ex)
|
appv ("WARNING: cannot parse example " ++ ex)
|
||||||
case ws of
|
|
||||||
[] -> return ()
|
|
||||||
_ -> appv (" missing words: " ++ unwords ws)
|
|
||||||
return ws
|
|
||||||
TypeError _ ->
|
|
||||||
return []
|
return []
|
||||||
ParseIncomplete ->
|
ParseIncomplete ->
|
||||||
return []
|
return []
|
||||||
ParseOk ts ->
|
ParseOk ts ->
|
||||||
case rank ts of
|
case ts of
|
||||||
(t:tt) -> do
|
(t:tt) -> do
|
||||||
if null tt
|
if null tt
|
||||||
then return ()
|
then return ()
|
||||||
else appv ("WARNING: ambiguous example " ++ ex)
|
else appv ("WARNING: ambiguous example " ++ ex)
|
||||||
appn t
|
appn (printExp conf (fst t))
|
||||||
mapM_ (appn . (" --- " ++)) tt
|
mapM_ (appn . (" --- " ++) . printExp conf . fst) tt
|
||||||
appn ")"
|
appn ")"
|
||||||
return []
|
return []
|
||||||
return ws
|
return ws
|
||||||
rank ts = [printExp conf t ++ " -- " ++ show p | (t,p) <- rankTreesByProbs pgf ts]
|
|
||||||
appf = appendFile file
|
appf = appendFile file
|
||||||
appn s = appf s >> appf "\n"
|
appn s = appf s >> appf "\n"
|
||||||
appv s = appn ("--- " ++ s) >> putStrLn s
|
appv s = appn ("--- " ++ s) >> putStrLn s
|
||||||
|
|
||||||
data ExConfiguration = ExConf {
|
data ExConfiguration = ExConf {
|
||||||
resource_pgf :: PGF,
|
resource_pgf :: PGF,
|
||||||
resource_morpho :: Morpho,
|
|
||||||
verbose :: Bool,
|
verbose :: Bool,
|
||||||
language :: Language,
|
language :: Concr,
|
||||||
printExp :: Tree -> String
|
printExp :: Expr -> String
|
||||||
}
|
}
|
||||||
|
|
||||||
configureExBased :: PGF -> Morpho -> Language -> (Tree -> String) -> ExConfiguration
|
configureExBased :: PGF -> Concr -> (Expr -> String) -> ExConfiguration
|
||||||
configureExBased pgf morpho lang pr = ExConf pgf morpho False lang pr
|
configureExBased pgf concr pr = ExConf pgf False concr pr
|
||||||
|
|
||||||
|
|||||||
@@ -1,14 +1,10 @@
|
|||||||
module GF.Compile.Export where
|
module GF.Compile.Export where
|
||||||
|
|
||||||
import PGF
|
import PGF2
|
||||||
import PGF.Internal(ppPGF)
|
|
||||||
import GF.Compile.PGFtoHaskell
|
import GF.Compile.PGFtoHaskell
|
||||||
--import GF.Compile.PGFtoAbstract
|
--import GF.Compile.PGFtoAbstract
|
||||||
import GF.Compile.PGFtoJava
|
import GF.Compile.PGFtoJava
|
||||||
import GF.Compile.PGFtoProlog
|
|
||||||
import GF.Compile.PGFtoJS
|
|
||||||
import GF.Compile.PGFtoJSON
|
import GF.Compile.PGFtoJSON
|
||||||
import GF.Compile.PGFtoPython
|
|
||||||
import GF.Infra.Option
|
import GF.Infra.Option
|
||||||
--import GF.Speech.CFG
|
--import GF.Speech.CFG
|
||||||
import GF.Speech.PGFToCFG
|
import GF.Speech.PGFToCFG
|
||||||
@@ -22,6 +18,7 @@ import GF.Speech.SLF
|
|||||||
import GF.Speech.PrRegExp
|
import GF.Speech.PrRegExp
|
||||||
|
|
||||||
import Data.Maybe
|
import Data.Maybe
|
||||||
|
import qualified Data.Map as Map
|
||||||
import System.FilePath
|
import System.FilePath
|
||||||
import GF.Text.Pretty
|
import GF.Text.Pretty
|
||||||
|
|
||||||
@@ -35,15 +32,12 @@ exportPGF :: Options
|
|||||||
-> [(FilePath,String)] -- ^ List of recommended file names and contents.
|
-> [(FilePath,String)] -- ^ List of recommended file names and contents.
|
||||||
exportPGF opts fmt pgf =
|
exportPGF opts fmt pgf =
|
||||||
case fmt of
|
case fmt of
|
||||||
FmtPGFPretty -> multi "txt" (render . ppPGF)
|
FmtPGFPretty -> multi "txt" (showPGF)
|
||||||
FmtCanonicalGF -> [] -- canon "gf" (render80 . abstract2canonical)
|
FmtCanonicalGF -> [] -- canon "gf" (render80 . abstract2canonical)
|
||||||
FmtCanonicalJson-> []
|
FmtCanonicalJson-> []
|
||||||
FmtJavaScript -> multi "js" pgf2js
|
|
||||||
FmtJSON -> multi "json" pgf2json
|
FmtJSON -> multi "json" pgf2json
|
||||||
FmtPython -> multi "py" pgf2python
|
|
||||||
FmtHaskell -> multi "hs" (grammar2haskell opts name)
|
FmtHaskell -> multi "hs" (grammar2haskell opts name)
|
||||||
FmtJava -> multi "java" (grammar2java opts name)
|
FmtJava -> multi "java" (grammar2java opts name)
|
||||||
FmtProlog -> multi "pl" grammar2prolog
|
|
||||||
FmtBNF -> single "bnf" bnfPrinter
|
FmtBNF -> single "bnf" bnfPrinter
|
||||||
FmtEBNF -> single "ebnf" (ebnfPrinter opts)
|
FmtEBNF -> single "ebnf" (ebnfPrinter opts)
|
||||||
FmtSRGS_XML -> single "grxml" (srgsXmlPrinter opts)
|
FmtSRGS_XML -> single "grxml" (srgsXmlPrinter opts)
|
||||||
@@ -57,20 +51,13 @@ exportPGF opts fmt pgf =
|
|||||||
FmtRegExp -> single "rexp" regexpPrinter
|
FmtRegExp -> single "rexp" regexpPrinter
|
||||||
FmtFA -> single "dot" slfGraphvizPrinter
|
FmtFA -> single "dot" slfGraphvizPrinter
|
||||||
where
|
where
|
||||||
name = fromMaybe (showCId (abstractName pgf)) (flag optName opts)
|
name = fromMaybe (abstractName pgf) (flag optName opts)
|
||||||
|
|
||||||
multi :: String -> (PGF -> String) -> [(FilePath,String)]
|
multi :: String -> (PGF -> String) -> [(FilePath,String)]
|
||||||
multi ext pr = [(name <.> ext, pr pgf)]
|
multi ext pr = [(name <.> ext, pr pgf)]
|
||||||
|
|
||||||
-- canon ext pr = [("canonical"</>name<.>ext,pr pgf)]
|
-- canon ext pr = [("canonical"</>name<.>ext,pr pgf)]
|
||||||
|
|
||||||
single :: String -> (PGF -> CId -> String) -> [(FilePath,String)]
|
single :: String -> (PGF -> Concr -> String) -> [(FilePath,String)]
|
||||||
single ext pr = [(showCId cnc <.> ext, pr pgf cnc) | cnc <- languages pgf]
|
single ext pr = [(concreteName cnc <.> ext, pr pgf cnc) | cnc <- Map.elems (languages pgf)]
|
||||||
|
|
||||||
|
|
||||||
-- | Get the name of the concrete syntax to generate output from.
|
|
||||||
-- FIXME: there should be an option to change this.
|
|
||||||
outputConcr :: PGF -> CId
|
|
||||||
outputConcr pgf = case languages pgf of
|
|
||||||
[] -> error "No concrete syntax."
|
|
||||||
cnc:_ -> cnc
|
|
||||||
|
|||||||
@@ -1,10 +1,10 @@
|
|||||||
|
{-# LANGUAGE CPP #-}
|
||||||
module GF.Compile.GenerateBC(generateByteCode) where
|
module GF.Compile.GenerateBC(generateByteCode) where
|
||||||
|
|
||||||
import GF.Grammar
|
import GF.Grammar
|
||||||
import GF.Grammar.Lookup(lookupAbsDef,lookupFunType)
|
import GF.Grammar.Lookup(lookupAbsDef,lookupFunType)
|
||||||
import GF.Data.Operations
|
import GF.Data.Operations
|
||||||
import PGF(CId,utf8CId)
|
import PGF2.Internal(CodeLabel,Instr(..),IVal(..),TailInfo(..),Literal(..))
|
||||||
import PGF.Internal(CodeLabel,Instr(..),IVal(..),TailInfo(..),Literal(..))
|
|
||||||
import qualified Data.Map as Map
|
import qualified Data.Map as Map
|
||||||
import Data.List(nub,mapAccumL)
|
import Data.List(nub,mapAccumL)
|
||||||
import Data.Maybe(fromMaybe)
|
import Data.Maybe(fromMaybe)
|
||||||
@@ -63,7 +63,7 @@ compileEquations gr arity st (i:is) eqs fl bs = whilePP eqs Map.empty
|
|||||||
|
|
||||||
case_instr t =
|
case_instr t =
|
||||||
case t of
|
case t of
|
||||||
(Q (_,id)) -> CASE (i2i id)
|
(Q (_,id)) -> CASE (showIdent id)
|
||||||
(EInt n) -> CASE_LIT (LInt n)
|
(EInt n) -> CASE_LIT (LInt n)
|
||||||
(K s) -> CASE_LIT (LStr s)
|
(K s) -> CASE_LIT (LStr s)
|
||||||
(EFloat d) -> CASE_LIT (LFlt d)
|
(EFloat d) -> CASE_LIT (LFlt d)
|
||||||
@@ -105,7 +105,7 @@ compileFun gr eval st vs (App e1 e2) h0 bs args =
|
|||||||
compileFun gr eval st vs (Q (m,id)) h0 bs args =
|
compileFun gr eval st vs (Q (m,id)) h0 bs args =
|
||||||
case lookupAbsDef gr m id of
|
case lookupAbsDef gr m id of
|
||||||
Ok (_,Just _)
|
Ok (_,Just _)
|
||||||
-> (h0,bs,eval st (GLOBAL (i2i id)) args)
|
-> (h0,bs,eval st (GLOBAL (showIdent id)) args)
|
||||||
_ -> let Ok ty = lookupFunType gr m id
|
_ -> let Ok ty = lookupFunType gr m id
|
||||||
(ctxt,_,_) = typeForm ty
|
(ctxt,_,_) = typeForm ty
|
||||||
c_arity = length ctxt
|
c_arity = length ctxt
|
||||||
@@ -114,14 +114,14 @@ compileFun gr eval st vs (Q (m,id)) h0 bs args =
|
|||||||
diff = c_arity-n_args
|
diff = c_arity-n_args
|
||||||
in if diff <= 0
|
in if diff <= 0
|
||||||
then if n_args == 0
|
then if n_args == 0
|
||||||
then (h0,bs,eval st (GLOBAL (i2i id)) [])
|
then (h0,bs,eval st (GLOBAL (showIdent id)) [])
|
||||||
else let h1 = h0 + 2 + n_args
|
else let h1 = h0 + 2 + n_args
|
||||||
in (h1,bs,PUT_CONSTR (i2i id):is1++eval st (HEAP h0) [])
|
in (h1,bs,PUT_CONSTR (showIdent id):is1++eval st (HEAP h0) [])
|
||||||
else let h1 = h0 + 1 + n_args
|
else let h1 = h0 + 1 + n_args
|
||||||
is2 = [SET (FREE_VAR i) | i <- [0..n_args-1]] ++ [SET (ARG_VAR (i+1)) | i <- [0..diff-1]]
|
is2 = [SET (FREE_VAR i) | i <- [0..n_args-1]] ++ [SET (ARG_VAR (i+1)) | i <- [0..diff-1]]
|
||||||
b = CHECK_ARGS diff :
|
b = CHECK_ARGS diff :
|
||||||
ALLOC (c_arity+2) :
|
ALLOC (c_arity+2) :
|
||||||
PUT_CONSTR (i2i id) :
|
PUT_CONSTR (showIdent id) :
|
||||||
is2 ++
|
is2 ++
|
||||||
TUCK (ARG_VAR 0) diff :
|
TUCK (ARG_VAR 0) diff :
|
||||||
EVAL (HEAP h0) (TailCall diff) :
|
EVAL (HEAP h0) (TailCall diff) :
|
||||||
@@ -167,16 +167,16 @@ compileFun gr eval st vs e _ _ _ = error (show e)
|
|||||||
|
|
||||||
compileArg gr st vs (Q(m,id)) h0 bs =
|
compileArg gr st vs (Q(m,id)) h0 bs =
|
||||||
case lookupAbsDef gr m id of
|
case lookupAbsDef gr m id of
|
||||||
Ok (_,Just _) -> (h0,bs,GLOBAL (i2i id),[])
|
Ok (_,Just _) -> (h0,bs,GLOBAL (showIdent id),[])
|
||||||
_ -> let Ok ty = lookupFunType gr m id
|
_ -> let Ok ty = lookupFunType gr m id
|
||||||
(ctxt,_,_) = typeForm ty
|
(ctxt,_,_) = typeForm ty
|
||||||
c_arity = length ctxt
|
c_arity = length ctxt
|
||||||
in if c_arity == 0
|
in if c_arity == 0
|
||||||
then (h0,bs,GLOBAL (i2i id),[])
|
then (h0,bs,GLOBAL (showIdent id),[])
|
||||||
else let is2 = [SET (ARG_VAR (i+1)) | i <- [0..c_arity-1]]
|
else let is2 = [SET (ARG_VAR (i+1)) | i <- [0..c_arity-1]]
|
||||||
b = CHECK_ARGS c_arity :
|
b = CHECK_ARGS c_arity :
|
||||||
ALLOC (c_arity+2) :
|
ALLOC (c_arity+2) :
|
||||||
PUT_CONSTR (i2i id) :
|
PUT_CONSTR (showIdent id) :
|
||||||
is2 ++
|
is2 ++
|
||||||
TUCK (ARG_VAR 0) c_arity :
|
TUCK (ARG_VAR 0) c_arity :
|
||||||
EVAL (HEAP h0) (TailCall c_arity) :
|
EVAL (HEAP h0) (TailCall c_arity) :
|
||||||
@@ -224,12 +224,12 @@ compileArg gr st vs e h0 bs =
|
|||||||
diff = c_arity-n_args
|
diff = c_arity-n_args
|
||||||
in if diff <= 0
|
in if diff <= 0
|
||||||
then let h2 = h1 + 2 + n_args
|
then let h2 = h1 + 2 + n_args
|
||||||
in (h2,bs1,HEAP h1,is1 ++ (PUT_CONSTR (i2i id) : is2))
|
in (h2,bs1,HEAP h1,is1 ++ (PUT_CONSTR (showIdent id) : is2))
|
||||||
else let h2 = h1 + 1 + n_args
|
else let h2 = h1 + 1 + n_args
|
||||||
is2 = [SET (FREE_VAR i) | i <- [0..n_args-1]] ++ [SET (ARG_VAR (i+1)) | i <- [0..diff-1]]
|
is2 = [SET (FREE_VAR i) | i <- [0..n_args-1]] ++ [SET (ARG_VAR (i+1)) | i <- [0..diff-1]]
|
||||||
b = CHECK_ARGS diff :
|
b = CHECK_ARGS diff :
|
||||||
ALLOC (c_arity+2) :
|
ALLOC (c_arity+2) :
|
||||||
PUT_CONSTR (i2i id) :
|
PUT_CONSTR (showIdent id) :
|
||||||
is2 ++
|
is2 ++
|
||||||
TUCK (ARG_VAR 0) diff :
|
TUCK (ARG_VAR 0) diff :
|
||||||
EVAL (HEAP h0) (TailCall diff) :
|
EVAL (HEAP h0) (TailCall diff) :
|
||||||
@@ -298,9 +298,6 @@ freeVars xs (Vr x)
|
|||||||
| not (elem x xs) = [x]
|
| not (elem x xs) = [x]
|
||||||
freeVars xs e = collectOp (freeVars xs) e
|
freeVars xs e = collectOp (freeVars xs) e
|
||||||
|
|
||||||
i2i :: Ident -> CId
|
|
||||||
i2i = utf8CId . ident2utf8
|
|
||||||
|
|
||||||
push_is :: Int -> Int -> [IVal] -> [IVal]
|
push_is :: Int -> Int -> [IVal] -> [IVal]
|
||||||
push_is i 0 is = is
|
push_is i 0 is = is
|
||||||
push_is i n is = ARG_VAR i : push_is (i-1) (n-1) is
|
push_is i n is = ARG_VAR i : push_is (i-1) (n-1) is
|
||||||
|
|||||||
@@ -13,8 +13,9 @@ module GF.Compile.GeneratePMCFG
|
|||||||
(generatePMCFG, pgfCncCat, addPMCFG, resourceValues
|
(generatePMCFG, pgfCncCat, addPMCFG, resourceValues
|
||||||
) where
|
) where
|
||||||
|
|
||||||
--import PGF.CId
|
import qualified PGF2 as PGF2
|
||||||
import PGF.Internal as PGF(CncCat(..),Symbol(..),fidVar)
|
import qualified PGF2.Internal as PGF2
|
||||||
|
import PGF2.Internal(Symbol(..),fidVar)
|
||||||
|
|
||||||
import GF.Infra.Option
|
import GF.Infra.Option
|
||||||
import GF.Grammar hiding (Env, mkRecord, mkTable)
|
import GF.Grammar hiding (Env, mkRecord, mkTable)
|
||||||
@@ -68,7 +69,7 @@ mapAccumWithKeyM f a m = do let xs = Map.toAscList m
|
|||||||
|
|
||||||
|
|
||||||
--addPMCFG :: Options -> SourceGrammar -> GlobalEnv -> Maybe FilePath -> Ident -> Ident -> SeqSet -> Ident -> Info -> IOE (SeqSet, Info)
|
--addPMCFG :: Options -> SourceGrammar -> GlobalEnv -> Maybe FilePath -> Ident -> Ident -> SeqSet -> Ident -> Info -> IOE (SeqSet, Info)
|
||||||
addPMCFG opts gr cenv opath am cm seqs id (GF.Grammar.CncFun mty@(Just (cat,cont,val)) mlin@(Just (L loc term)) mprn Nothing) = do
|
addPMCFG opts gr cenv opath am cm seqs id (CncFun mty@(Just (cat,cont,val)) mlin@(Just (L loc term)) mprn Nothing) = do
|
||||||
--when (verbAtLeast opts Verbose) $ ePutStr ("\n+ "++showIdent id++" ...")
|
--when (verbAtLeast opts Verbose) $ ePutStr ("\n+ "++showIdent id++" ...")
|
||||||
let pres = protoFCat gr res val
|
let pres = protoFCat gr res val
|
||||||
pargs = [protoFCat gr (snd $ catSkeleton ty) lincat | ((_,_,ty),(_,_,lincat)) <- zip ctxt cont]
|
pargs = [protoFCat gr (snd $ catSkeleton ty) lincat | ((_,_,ty),(_,_,lincat)) <- zip ctxt cont]
|
||||||
@@ -92,7 +93,7 @@ addPMCFG opts gr cenv opath am cm seqs id (GF.Grammar.CncFun mty@(Just (cat,cont
|
|||||||
ePutStr ("\n+ "++showIdent id++" "++show (product (map catFactor pargs)))
|
ePutStr ("\n+ "++showIdent id++" "++show (product (map catFactor pargs)))
|
||||||
seqs1 `seq` stats `seq` return ()
|
seqs1 `seq` stats `seq` return ()
|
||||||
when (verbAtLeast opts Verbose) $ ePutStr (" "++show stats)
|
when (verbAtLeast opts Verbose) $ ePutStr (" "++show stats)
|
||||||
return (seqs1,GF.Grammar.CncFun mty mlin mprn (Just pmcfg))
|
return (seqs1,CncFun mty mlin mprn (Just pmcfg))
|
||||||
where
|
where
|
||||||
(ctxt,res,_) = err bug typeForm (lookupFunType gr am id)
|
(ctxt,res,_) = err bug typeForm (lookupFunType gr am id)
|
||||||
|
|
||||||
@@ -102,11 +103,11 @@ addPMCFG opts gr cenv opath am cm seqs id (GF.Grammar.CncFun mty@(Just (cat,cont
|
|||||||
newArgs = map getFIds newArgs'
|
newArgs = map getFIds newArgs'
|
||||||
in addFunction env0 newCat fun newArgs
|
in addFunction env0 newCat fun newArgs
|
||||||
|
|
||||||
addPMCFG opts gr cenv opath am cm seqs id (GF.Grammar.CncCat mty@(Just (L _ lincat))
|
addPMCFG opts gr cenv opath am cm seqs id (CncCat mty@(Just (L _ lincat))
|
||||||
mdef@(Just (L loc1 def))
|
mdef@(Just (L loc1 def))
|
||||||
mref@(Just (L loc2 ref))
|
mref@(Just (L loc2 ref))
|
||||||
mprn
|
mprn
|
||||||
Nothing) = do
|
Nothing) = do
|
||||||
let pcat = protoFCat gr (am,id) lincat
|
let pcat = protoFCat gr (am,id) lincat
|
||||||
pvar = protoFCat gr (MN identW,cVar) typeStr
|
pvar = protoFCat gr (MN identW,cVar) typeStr
|
||||||
|
|
||||||
@@ -131,7 +132,7 @@ addPMCFG opts gr cenv opath am cm seqs id (GF.Grammar.CncCat mty@(Just (L _ linc
|
|||||||
let pmcfg = getPMCFG pmcfgEnv2
|
let pmcfg = getPMCFG pmcfgEnv2
|
||||||
|
|
||||||
when (verbAtLeast opts Verbose) $ ePutStr ("\n+ "++showIdent id++" "++show (catFactor pcat))
|
when (verbAtLeast opts Verbose) $ ePutStr ("\n+ "++showIdent id++" "++show (catFactor pcat))
|
||||||
seqs2 `seq` pmcfg `seq` return (seqs2,GF.Grammar.CncCat mty mdef mref mprn (Just pmcfg))
|
seqs2 `seq` pmcfg `seq` return (seqs2,CncCat mty mdef mref mprn (Just pmcfg))
|
||||||
where
|
where
|
||||||
addLindef lins (newCat', newArgs') env0 =
|
addLindef lins (newCat', newArgs') env0 =
|
||||||
let [newCat] = getFIds newCat'
|
let [newCat] = getFIds newCat'
|
||||||
@@ -157,12 +158,15 @@ convert opts gr cenv loc term ty@(_,val) pargs =
|
|||||||
args = map Vr vars
|
args = map Vr vars
|
||||||
vars = map (\(bt,x,t) -> x) context
|
vars = map (\(bt,x,t) -> x) context
|
||||||
|
|
||||||
pgfCncCat :: SourceGrammar -> Type -> Int -> CncCat
|
pgfCncCat :: SourceGrammar -> PGF2.Cat -> Type -> Int -> (PGF2.Cat,Int,Int,[String])
|
||||||
pgfCncCat gr lincat index =
|
pgfCncCat gr id lincat index =
|
||||||
let ((_,size),schema) = computeCatRange gr lincat
|
let ((_,size),schema) = computeCatRange gr lincat
|
||||||
in PGF.CncCat index (index+size-1)
|
in ( id
|
||||||
(mkArray (map (renderStyle style{mode=OneLineMode} . ppPath)
|
, index
|
||||||
(getStrPaths schema)))
|
, index+size-1
|
||||||
|
, map (renderStyle style{mode=OneLineMode} . ppPath)
|
||||||
|
(getStrPaths schema)
|
||||||
|
)
|
||||||
where
|
where
|
||||||
getStrPaths :: Schema Identity s c -> [Path]
|
getStrPaths :: Schema Identity s c -> [Path]
|
||||||
getStrPaths = collect CNil []
|
getStrPaths = collect CNil []
|
||||||
@@ -471,7 +475,7 @@ goV (CPar t) rpath ss = restrictHead (reversePath rpath) t >> return ss
|
|||||||
----------------------------------------------------------------------
|
----------------------------------------------------------------------
|
||||||
-- SeqSet
|
-- SeqSet
|
||||||
|
|
||||||
type SeqSet = Map.Map Sequence SeqId
|
type SeqSet = Map.Map [Symbol] SeqId
|
||||||
|
|
||||||
addSequencesB :: SeqSet -> Branch (Value [Symbol]) -> (SeqSet, Branch (Value SeqId))
|
addSequencesB :: SeqSet -> Branch (Value [Symbol]) -> (SeqSet, Branch (Value SeqId))
|
||||||
addSequencesB seqs (Case nr path bs) = let !(seqs1,bs1) = mapAccumL' (\seqs (trm,b) -> let !(seqs',b') = addSequencesB seqs b
|
addSequencesB seqs (Case nr path bs) = let !(seqs1,bs1) = mapAccumL' (\seqs (trm,b) -> let !(seqs',b') = addSequencesB seqs b
|
||||||
@@ -500,13 +504,11 @@ mapAccumL' f s (x:xs) = (s'',y:ys)
|
|||||||
!(s'',ys) = mapAccumL' f s' xs
|
!(s'',ys) = mapAccumL' f s' xs
|
||||||
|
|
||||||
addSequence :: SeqSet -> [Symbol] -> (SeqSet,SeqId)
|
addSequence :: SeqSet -> [Symbol] -> (SeqSet,SeqId)
|
||||||
addSequence seqs lst =
|
addSequence seqs seq =
|
||||||
case Map.lookup seq seqs of
|
case Map.lookup seq seqs of
|
||||||
Just id -> (seqs,id)
|
Just id -> (seqs,id)
|
||||||
Nothing -> let !last_seq = Map.size seqs
|
Nothing -> let !last_seq = Map.size seqs
|
||||||
in (Map.insert seq last_seq seqs, last_seq)
|
in (Map.insert seq last_seq seqs, last_seq)
|
||||||
where
|
|
||||||
seq = mkArray lst
|
|
||||||
|
|
||||||
|
|
||||||
------------------------------------------------------------
|
------------------------------------------------------------
|
||||||
|
|||||||
@@ -50,20 +50,13 @@ getSourceModule opts file0 =
|
|||||||
Right (i,mi0) ->
|
Right (i,mi0) ->
|
||||||
do liftIO $ removeTemp tmp
|
do liftIO $ removeTemp tmp
|
||||||
let mi =mi0 {mflags=mflags mi0 `addOptions` opts, msrc=file0}
|
let mi =mi0 {mflags=mflags mi0 `addOptions` opts, msrc=file0}
|
||||||
optCoding' = renameEncoding `fmap` flag optEncoding (mflags mi0)
|
case renameEncoding `fmap` flag optEncoding (mflags mi0) of
|
||||||
case (optCoding,optCoding') of
|
Just coding' ->
|
||||||
{-
|
when (coding/=coding') $
|
||||||
(Nothing,Nothing) ->
|
|
||||||
unless (BS.all isAscii raw) $
|
|
||||||
ePutStrLn $ file0++":\n Warning: default encoding has changed from Latin-1 to UTF-8"
|
|
||||||
-}
|
|
||||||
(_,Just coding') ->
|
|
||||||
when (coding/=coding') $
|
|
||||||
raise $ "Encoding mismatch: "++coding++" /= "++coding'
|
raise $ "Encoding mismatch: "++coding++" /= "++coding'
|
||||||
where coding = maybe defaultEncoding renameEncoding optCoding
|
where coding = maybe defaultEncoding renameEncoding optCoding
|
||||||
_ -> return ()
|
_ -> return ()
|
||||||
--liftIO $ transcodeModule' (i,mi) -- old lexer
|
return (i,mi)
|
||||||
return (i,mi) -- new lexer
|
|
||||||
|
|
||||||
getBNFCRules :: Options -> FilePath -> IOE [BNFCRule]
|
getBNFCRules :: Options -> FilePath -> IOE [BNFCRule]
|
||||||
getBNFCRules opts fpath = do
|
getBNFCRules opts fpath = do
|
||||||
|
|||||||
@@ -18,7 +18,7 @@ import GF.Compile.Compute.Predef(predef)
|
|||||||
import GF.Compile.Compute.Value(Predefined(..))
|
import GF.Compile.Compute.Value(Predefined(..))
|
||||||
import GF.Infra.Ident(ModuleName(..),Ident,prefixIdent,showIdent,isWildIdent)
|
import GF.Infra.Ident(ModuleName(..),Ident,prefixIdent,showIdent,isWildIdent)
|
||||||
import GF.Infra.Option(optionsPGF)
|
import GF.Infra.Option(optionsPGF)
|
||||||
import PGF.Internal(Literal(..))
|
import PGF2.Internal(Literal(..))
|
||||||
import GF.Compile.Compute.ConcreteNew(normalForm,resourceValues)
|
import GF.Compile.Compute.ConcreteNew(normalForm,resourceValues)
|
||||||
import GF.Grammar.Canonical as C
|
import GF.Grammar.Canonical as C
|
||||||
import Debug.Trace
|
import Debug.Trace
|
||||||
@@ -215,6 +215,7 @@ convert' gr vs = ppT
|
|||||||
alt (t,p) = (pre p,ppT0 t)
|
alt (t,p) = (pre p,ppT0 t)
|
||||||
|
|
||||||
pre (K s) = [s]
|
pre (K s) = [s]
|
||||||
|
pre Empty = [""] -- Empty == K ""
|
||||||
pre (Strs ts) = concatMap pre ts
|
pre (Strs ts) = concatMap pre ts
|
||||||
pre (EPatt p) = pat p
|
pre (EPatt p) = pat p
|
||||||
pre t = error $ "pre "++show t
|
pre t = error $ "pre "++show t
|
||||||
@@ -352,9 +353,9 @@ paramType gr q@(_,n) =
|
|||||||
[ParamAliasDef ((gQId m n)) (convType t)])
|
[ParamAliasDef ((gQId m n)) (convType t)])
|
||||||
_ -> ((S.empty,S.empty),[])
|
_ -> ((S.empty,S.empty),[])
|
||||||
where
|
where
|
||||||
param m (n,ctx) = Param ((gQId m n)) [toParamId t|(_,_,t)<-ctx]
|
param m (n,ctx,_) = Param ((gQId m n)) [toParamId t|(_,_,t)<-ctx]
|
||||||
argTypes = S.unions . map argTypes1
|
argTypes = S.unions . map argTypes1
|
||||||
argTypes1 (n,ctx) = S.unions [paramTypes gr t|(_,_,t)<-ctx]
|
argTypes1 (n,ctx,_) = S.unions [paramTypes gr t|(_,_,t)<-ctx]
|
||||||
|
|
||||||
lblId = LabelId . render -- hmm
|
lblId = LabelId . render -- hmm
|
||||||
modId (MN m) = ModId (showIdent m)
|
modId (MN m) = ModId (showIdent m)
|
||||||
|
|||||||
@@ -1,23 +1,17 @@
|
|||||||
{-# LANGUAGE BangPatterns, FlexibleContexts, MagicHash #-}
|
{-# LANGUAGE ImplicitParams, BangPatterns, FlexibleContexts, MagicHash #-}
|
||||||
module GF.Compile.GrammarToPGF (mkCanon2pgf) where
|
module GF.Compile.GrammarToPGF (grammar2PGF) where
|
||||||
|
|
||||||
--import GF.Compile.Export
|
|
||||||
import GF.Compile.GeneratePMCFG
|
import GF.Compile.GeneratePMCFG
|
||||||
import GF.Compile.GenerateBC
|
import GF.Compile.GenerateBC
|
||||||
|
import GF.Compile.OptimizePGF
|
||||||
|
|
||||||
import PGF(CId,mkCId,utf8CId)
|
import PGF2 hiding (mkType)
|
||||||
import PGF.Internal(fidInt,fidFloat,fidString,fidVar)
|
import PGF2.Internal
|
||||||
import PGF.Internal(updateProductionIndices)
|
|
||||||
--import qualified PGF.Macros as CM
|
|
||||||
import qualified PGF.Internal as C
|
|
||||||
import qualified PGF.Internal as D
|
|
||||||
import GF.Grammar.Predef
|
import GF.Grammar.Predef
|
||||||
--import GF.Grammar.Printer
|
import GF.Grammar.Grammar hiding (Production)
|
||||||
import GF.Grammar.Grammar
|
|
||||||
import qualified GF.Grammar.Lookup as Look
|
import qualified GF.Grammar.Lookup as Look
|
||||||
import qualified GF.Grammar as A
|
import qualified GF.Grammar as A
|
||||||
import qualified GF.Grammar.Macros as GM
|
import qualified GF.Grammar.Macros as GM
|
||||||
--import GF.Compile.GeneratePMCFG
|
|
||||||
|
|
||||||
import GF.Infra.Ident
|
import GF.Infra.Ident
|
||||||
import GF.Infra.Option
|
import GF.Infra.Option
|
||||||
@@ -25,114 +19,141 @@ import GF.Infra.UseIO (IOE)
|
|||||||
import GF.Data.Operations
|
import GF.Data.Operations
|
||||||
|
|
||||||
import Data.List
|
import Data.List
|
||||||
|
import Data.Char
|
||||||
import qualified Data.Set as Set
|
import qualified Data.Set as Set
|
||||||
import qualified Data.Map as Map
|
import qualified Data.Map as Map
|
||||||
import qualified Data.IntMap as IntMap
|
import qualified Data.IntMap as IntMap
|
||||||
import Data.Array.IArray
|
import Data.Array.IArray
|
||||||
|
import Data.Maybe(fromMaybe)
|
||||||
|
|
||||||
import Data.Char
|
|
||||||
import GHC.Prim
|
import GHC.Prim
|
||||||
import GHC.Base(getTag)
|
import GHC.Base(getTag)
|
||||||
|
|
||||||
mkCanon2pgf :: Options -> SourceGrammar -> ModuleName -> IOE D.PGF
|
grammar2PGF :: Options -> SourceGrammar -> ModuleName -> Map.Map PGF2.Fun Double -> IO PGF
|
||||||
mkCanon2pgf opts gr am = do
|
grammar2PGF opts gr am probs = do
|
||||||
(an,abs) <- mkAbstr am
|
cnc_infos <- getConcreteInfos gr am
|
||||||
cncs <- mapM mkConcr (allConcretes gr am)
|
return $
|
||||||
return $ updateProductionIndices (D.PGF Map.empty an abs (Map.fromList cncs))
|
build (let gflags = if flag optSplitPGF opts
|
||||||
|
then [("split", LStr "true")]
|
||||||
|
else []
|
||||||
|
(an,abs) = mkAbstr am probs
|
||||||
|
cncs = map (mkConcr opts abs) cnc_infos
|
||||||
|
in newPGF gflags an abs cncs)
|
||||||
where
|
where
|
||||||
cenv = resourceValues opts gr
|
cenv = resourceValues opts gr
|
||||||
|
aflags = err (const noOptions) mflags (lookupModule gr am)
|
||||||
|
|
||||||
mkAbstr am = return (mi2i am, D.Abstr flags funs cats)
|
mkAbstr :: (?builder :: Builder s) => ModuleName -> Map.Map PGF2.Fun Double -> (AbsName, B s AbstrInfo)
|
||||||
|
mkAbstr am probs = (mi2i am, newAbstr flags cats funs)
|
||||||
where
|
where
|
||||||
aflags = err (const noOptions) mflags (lookupModule gr am)
|
|
||||||
|
|
||||||
adefs =
|
adefs =
|
||||||
[((cPredefAbs,c), AbsCat (Just (L NoLoc []))) | c <- [cFloat,cInt,cString]] ++
|
[((cPredefAbs,c), AbsCat (Just (L NoLoc []))) | c <- [cFloat,cInt,cString]] ++
|
||||||
Look.allOrigInfos gr am
|
Look.allOrigInfos gr am
|
||||||
|
|
||||||
flags = Map.fromList [(mkCId f,x) | (f,x) <- optionsPGF aflags]
|
flags = optionsPGF aflags
|
||||||
|
|
||||||
funs = Map.fromList [(i2i f, (mkType [] ty, arity, mkDef gr arity mdef, 0)) |
|
toLogProb = realToFrac . negate . log
|
||||||
|
|
||||||
|
cats = [(c', snd (mkContext [] cont), toLogProb (fromMaybe 0 (Map.lookup c' probs))) |
|
||||||
|
((m,c),AbsCat (Just (L _ cont))) <- adefs, let c' = i2i c]
|
||||||
|
|
||||||
|
funs = [(f', mkType [] ty, arity, bcode, toLogProb (fromMaybe 0 (Map.lookup f' funs_probs))) |
|
||||||
((m,f),AbsFun (Just (L _ ty)) ma mdef _) <- adefs,
|
((m,f),AbsFun (Just (L _ ty)) ma mdef _) <- adefs,
|
||||||
let arity = mkArity ma mdef ty]
|
let arity = mkArity ma mdef ty,
|
||||||
|
let bcode = mkDef gr arity mdef,
|
||||||
|
let f' = i2i f]
|
||||||
|
|
||||||
|
funs_probs = (Map.fromList . concat . Map.elems . fmap pad . Map.fromListWith (++))
|
||||||
|
[(i2i cat,[(i2i f,Map.lookup f' probs)]) | ((m,f),AbsFun (Just (L _ ty)) _ _ _) <- adefs,
|
||||||
|
let (_,(_,cat),_) = GM.typeForm ty,
|
||||||
|
let f' = i2i f]
|
||||||
|
where
|
||||||
|
pad :: [(a,Maybe Double)] -> [(a,Double)]
|
||||||
|
pad pfs = [(f,fromMaybe deflt mb_p) | (f,mb_p) <- pfs]
|
||||||
|
where
|
||||||
|
deflt = case length [f | (f,Nothing) <- pfs] of
|
||||||
|
0 -> 0
|
||||||
|
n -> max 0 ((1 - sum [d | (f,Just d) <- pfs]) / fromIntegral n)
|
||||||
|
|
||||||
cats = Map.fromList [(i2i c, (snd (mkContext [] cont),catfuns c, 0)) |
|
mkConcr opts abs (cm,ex_seqs,cdefs) =
|
||||||
((m,c),AbsCat (Just (L _ cont))) <- adefs]
|
|
||||||
|
|
||||||
catfuns cat =
|
|
||||||
[(0,i2i f) | ((m,f),AbsFun (Just (L _ ty)) _ _ (Just True)) <- adefs, snd (GM.valCat ty) == cat]
|
|
||||||
|
|
||||||
mkConcr cm = do
|
|
||||||
let cflags = err (const noOptions) mflags (lookupModule gr cm)
|
let cflags = err (const noOptions) mflags (lookupModule gr cm)
|
||||||
ciCmp | flag optCaseSensitive cflags = compare
|
ciCmp | flag optCaseSensitive cflags = compare
|
||||||
| otherwise = compareCaseInsensitve
|
| otherwise = compareCaseInsensitive
|
||||||
|
|
||||||
(ex_seqs,cdefs) <- addMissingPMCFGs
|
flags = optionsPGF aflags
|
||||||
Map.empty
|
|
||||||
([((cPredefAbs,c), CncCat (Just (L NoLoc GM.defLinType)) Nothing Nothing Nothing Nothing) | c <- [cInt,cFloat,cString]] ++
|
|
||||||
Look.allOrigInfos gr cm)
|
|
||||||
|
|
||||||
let flags = Map.fromList [(mkCId f,x) | (f,x) <- optionsPGF cflags]
|
seqs = (mkSetArray . Set.fromList . concat) $
|
||||||
|
(elems (ex_seqs :: Array SeqId [Symbol]) : [maybe [] elems (mseqs mi) | (m,mi) <- allExtends gr cm])
|
||||||
seqs = (mkArray . sortNubBy ciCmp . concat) $
|
|
||||||
(Map.keys ex_seqs : [maybe [] elems (mseqs mi) | (m,mi) <- allExtends gr cm])
|
|
||||||
|
|
||||||
ex_seqs_arr = mkMapArray ex_seqs :: Array SeqId Sequence
|
|
||||||
|
|
||||||
!(!fid_cnt1,!cnccats) = genCncCats gr am cm cdefs
|
!(!fid_cnt1,!cnccats) = genCncCats gr am cm cdefs
|
||||||
|
cnccat_ranges = Map.fromList (map (\(cid,s,e,_) -> (cid,(s,e))) cnccats)
|
||||||
!(!fid_cnt2,!productions,!lindefs,!linrefs,!cncfuns)
|
!(!fid_cnt2,!productions,!lindefs,!linrefs,!cncfuns)
|
||||||
= genCncFuns gr am cm ex_seqs_arr ciCmp seqs cdefs fid_cnt1 cnccats
|
= genCncFuns gr am cm ex_seqs ciCmp seqs cdefs fid_cnt1 cnccat_ranges
|
||||||
|
|
||||||
printnames = genPrintNames cdefs
|
printnames = genPrintNames cdefs
|
||||||
return (mi2i cm, D.Concr flags
|
|
||||||
printnames
|
startCat = (fromMaybe "S" (flag optStartCat aflags))
|
||||||
cncfuns
|
|
||||||
lindefs
|
(lindefs',linrefs',productions',cncfuns',sequences',cnccats') =
|
||||||
linrefs
|
(if flag optOptimizePGF opts then optimizePGF startCat else id)
|
||||||
seqs
|
(lindefs,linrefs,productions,cncfuns,elems seqs,cnccats)
|
||||||
productions
|
|
||||||
IntMap.empty
|
in (mi2i cm, newConcr abs
|
||||||
Map.empty
|
flags
|
||||||
cnccats
|
printnames
|
||||||
IntMap.empty
|
lindefs'
|
||||||
fid_cnt2)
|
linrefs'
|
||||||
|
productions'
|
||||||
|
cncfuns'
|
||||||
|
sequences'
|
||||||
|
cnccats'
|
||||||
|
fid_cnt2)
|
||||||
|
|
||||||
|
getConcreteInfos gr am = mapM flatten (allConcretes gr am)
|
||||||
where
|
where
|
||||||
|
flatten cm = do
|
||||||
|
(seqs,infos) <- addMissingPMCFGs cm Map.empty
|
||||||
|
(lit_infos ++ Look.allOrigInfos gr cm)
|
||||||
|
return (cm,mkMapArray seqs :: Array SeqId [Symbol],infos)
|
||||||
|
|
||||||
|
lit_infos = [((cPredefAbs,c), CncCat (Just (L NoLoc GM.defLinType)) Nothing Nothing Nothing Nothing) | c <- [cInt,cFloat,cString]]
|
||||||
|
|
||||||
-- if some module was compiled with -no-pmcfg, then
|
-- if some module was compiled with -no-pmcfg, then
|
||||||
-- we have to create the PMCFG code just before linking
|
-- we have to create the PMCFG code just before linking
|
||||||
addMissingPMCFGs seqs [] = return (seqs,[])
|
addMissingPMCFGs cm seqs [] = return (seqs,[])
|
||||||
addMissingPMCFGs seqs (((m,id), info):is) = do
|
addMissingPMCFGs cm seqs (((m,id), info):is) = do
|
||||||
(seqs,info) <- addPMCFG opts gr cenv Nothing am cm seqs id info
|
(seqs,info) <- addPMCFG opts gr cenv Nothing am cm seqs id info
|
||||||
(seqs,is ) <- addMissingPMCFGs seqs is
|
(seqs,infos) <- addMissingPMCFGs cm seqs is
|
||||||
return (seqs, ((m,id), info) : is)
|
return (seqs, ((m,id), info) : infos)
|
||||||
|
|
||||||
i2i :: Ident -> CId
|
i2i :: Ident -> String
|
||||||
i2i = utf8CId . ident2utf8
|
i2i = showIdent
|
||||||
|
|
||||||
mi2i :: ModuleName -> CId
|
mi2i :: ModuleName -> String
|
||||||
mi2i (MN i) = i2i i
|
mi2i (MN i) = i2i i
|
||||||
|
|
||||||
mkType :: [Ident] -> A.Type -> C.Type
|
mkType :: (?builder :: Builder s) => [Ident] -> A.Type -> B s PGF2.Type
|
||||||
mkType scope t =
|
mkType scope t =
|
||||||
case GM.typeForm t of
|
case GM.typeForm t of
|
||||||
(hyps,(_,cat),args) -> let (scope',hyps') = mkContext scope hyps
|
(hyps,(_,cat),args) -> let (scope',hyps') = mkContext scope hyps
|
||||||
in C.DTyp hyps' (i2i cat) (map (mkExp scope') args)
|
in dTyp hyps' (i2i cat) (map (mkExp scope') args)
|
||||||
|
|
||||||
mkExp :: [Ident] -> A.Term -> C.Expr
|
mkExp :: (?builder :: Builder s) => [Ident] -> A.Term -> B s Expr
|
||||||
mkExp scope t =
|
mkExp scope t =
|
||||||
case t of
|
case t of
|
||||||
Q (_,c) -> C.EFun (i2i c)
|
Q (_,c) -> eFun (i2i c)
|
||||||
QC (_,c) -> C.EFun (i2i c)
|
QC (_,c) -> eFun (i2i c)
|
||||||
Vr x -> case lookup x (zip scope [0..]) of
|
Vr x -> case lookup x (zip scope [0..]) of
|
||||||
Just i -> C.EVar i
|
Just i -> eVar i
|
||||||
Nothing -> C.EMeta 0
|
Nothing -> eMeta 0
|
||||||
Abs b x t-> C.EAbs b (i2i x) (mkExp (x:scope) t)
|
Abs b x t-> eAbs b (i2i x) (mkExp (x:scope) t)
|
||||||
App t1 t2-> C.EApp (mkExp scope t1) (mkExp scope t2)
|
App t1 t2-> eApp (mkExp scope t1) (mkExp scope t2)
|
||||||
EInt i -> C.ELit (C.LInt (fromIntegral i))
|
EInt i -> eLit (LInt (fromIntegral i))
|
||||||
EFloat f -> C.ELit (C.LFlt f)
|
EFloat f -> eLit (LFlt f)
|
||||||
K s -> C.ELit (C.LStr s)
|
K s -> eLit (LStr s)
|
||||||
Meta i -> C.EMeta i
|
Meta i -> eMeta i
|
||||||
_ -> C.EMeta 0
|
_ -> eMeta 0
|
||||||
|
{-
|
||||||
mkPatt scope p =
|
mkPatt scope p =
|
||||||
case p of
|
case p of
|
||||||
A.PP (_,c) ps->let (scope',ps') = mapAccumL mkPatt scope ps
|
A.PP (_,c) ps->let (scope',ps') = mapAccumL mkPatt scope ps
|
||||||
@@ -147,67 +168,64 @@ mkPatt scope p =
|
|||||||
A.PImplArg p-> let (scope',p') = mkPatt scope p
|
A.PImplArg p-> let (scope',p') = mkPatt scope p
|
||||||
in (scope',C.PImplArg p')
|
in (scope',C.PImplArg p')
|
||||||
A.PTilde t -> ( scope,C.PTilde (mkExp scope t))
|
A.PTilde t -> ( scope,C.PTilde (mkExp scope t))
|
||||||
|
-}
|
||||||
mkContext :: [Ident] -> A.Context -> ([Ident],[C.Hypo])
|
mkContext :: (?builder :: Builder s) => [Ident] -> A.Context -> ([Ident],[B s PGF2.Hypo])
|
||||||
mkContext scope hyps = mapAccumL (\scope (bt,x,ty) -> let ty' = mkType scope ty
|
mkContext scope hyps = mapAccumL (\scope (bt,x,ty) -> let ty' = mkType scope ty
|
||||||
in if x == identW
|
in if x == identW
|
||||||
then ( scope,(bt,i2i x,ty'))
|
then ( scope,hypo bt (i2i x) ty')
|
||||||
else (x:scope,(bt,i2i x,ty'))) scope hyps
|
else (x:scope,hypo bt (i2i x) ty')) scope hyps
|
||||||
|
|
||||||
mkDef gr arity (Just eqs) = Just ([C.Equ ps' (mkExp scope' e) | L _ (ps,e) <- eqs, let (scope',ps') = mapAccumL mkPatt [] ps]
|
mkDef gr arity (Just eqs) = generateByteCode gr arity eqs
|
||||||
,generateByteCode gr arity eqs
|
mkDef gr arity Nothing = []
|
||||||
)
|
|
||||||
mkDef gr arity Nothing = Nothing
|
|
||||||
|
|
||||||
mkArity (Just a) _ ty = a -- known arity, i.e. defined function
|
mkArity (Just a) _ ty = a -- known arity, i.e. defined function
|
||||||
mkArity Nothing (Just _) ty = 0 -- defined function with no arity - must be an axiom
|
mkArity Nothing (Just _) ty = 0 -- defined function with no arity - must be an axiom
|
||||||
mkArity Nothing _ ty = let (ctxt, _, _) = GM.typeForm ty -- constructor
|
mkArity Nothing _ ty = let (ctxt, _, _) = GM.typeForm ty -- constructor
|
||||||
in length ctxt
|
in length ctxt
|
||||||
|
|
||||||
genCncCats gr am cm cdefs =
|
genCncCats gr am cm cdefs = mkCncCats 0 cdefs
|
||||||
let (index,cats) = mkCncCats 0 cdefs
|
|
||||||
in (index, Map.fromList cats)
|
|
||||||
where
|
where
|
||||||
mkCncCats index [] = (index,[])
|
mkCncCats index [] = (index,[])
|
||||||
mkCncCats index (((m,id),CncCat (Just (L _ lincat)) _ _ _ _):cdefs)
|
mkCncCats index (((m,id),CncCat (Just (L _ lincat)) _ _ _ _):cdefs)
|
||||||
| id == cInt =
|
| id == cInt =
|
||||||
let cc = pgfCncCat gr lincat fidInt
|
let cc = pgfCncCat gr (i2i id) lincat fidInt
|
||||||
(index',cats) = mkCncCats index cdefs
|
(index',cats) = mkCncCats index cdefs
|
||||||
in (index', (i2i id,cc) : cats)
|
in (index', cc : cats)
|
||||||
| id == cFloat =
|
| id == cFloat =
|
||||||
let cc = pgfCncCat gr lincat fidFloat
|
let cc = pgfCncCat gr (i2i id) lincat fidFloat
|
||||||
(index',cats) = mkCncCats index cdefs
|
(index',cats) = mkCncCats index cdefs
|
||||||
in (index', (i2i id,cc) : cats)
|
in (index', cc : cats)
|
||||||
| id == cString =
|
| id == cString =
|
||||||
let cc = pgfCncCat gr lincat fidString
|
let cc = pgfCncCat gr (i2i id) lincat fidString
|
||||||
(index',cats) = mkCncCats index cdefs
|
(index',cats) = mkCncCats index cdefs
|
||||||
in (index', (i2i id,cc) : cats)
|
in (index', cc : cats)
|
||||||
| otherwise =
|
| otherwise =
|
||||||
let cc@(C.CncCat _s e _) = pgfCncCat gr lincat index
|
let cc@(_, _s, e, _) = pgfCncCat gr (i2i id) lincat index
|
||||||
(index',cats) = mkCncCats (e+1) cdefs
|
(index',cats) = mkCncCats (e+1) cdefs
|
||||||
in (index', (i2i id,cc) : cats)
|
in (index', cc : cats)
|
||||||
mkCncCats index (_ :cdefs) = mkCncCats index cdefs
|
mkCncCats index (_ :cdefs) = mkCncCats index cdefs
|
||||||
|
|
||||||
genCncFuns :: Grammar
|
genCncFuns :: Grammar
|
||||||
-> ModuleName
|
-> ModuleName
|
||||||
-> ModuleName
|
-> ModuleName
|
||||||
-> Array SeqId Sequence
|
-> Array SeqId [Symbol]
|
||||||
-> (Sequence -> Sequence -> Ordering)
|
-> ([Symbol] -> [Symbol] -> Ordering)
|
||||||
-> Array SeqId Sequence
|
-> Array SeqId [Symbol]
|
||||||
-> [(QIdent, Info)]
|
-> [(QIdent, Info)]
|
||||||
-> FId
|
-> FId
|
||||||
-> Map.Map CId D.CncCat
|
-> Map.Map PGF2.Cat (Int,Int)
|
||||||
-> (FId,
|
-> (FId,
|
||||||
IntMap.IntMap (Set.Set D.Production),
|
[(FId, [Production])],
|
||||||
IntMap.IntMap [FunId],
|
[(FId, [FunId])],
|
||||||
IntMap.IntMap [FunId],
|
[(FId, [FunId])],
|
||||||
Array FunId D.CncFun)
|
[(PGF2.Fun,[SeqId])])
|
||||||
genCncFuns gr am cm ex_seqs ciCmp seqs cdefs fid_cnt cnccats =
|
genCncFuns gr am cm ex_seqs ciCmp seqs cdefs fid_cnt cnccat_ranges =
|
||||||
let (fid_cnt1,funs_cnt1,funs1,lindefs,linrefs) = mkCncCats cdefs fid_cnt 0 [] IntMap.empty IntMap.empty
|
let (fid_cnt1,funs_cnt1,funs1,lindefs,linrefs) = mkCncCats cdefs fid_cnt 0 [] IntMap.empty IntMap.empty
|
||||||
(fid_cnt2,funs_cnt2,funs2,prods) = mkCncFuns cdefs fid_cnt1 funs_cnt1 funs1 lindefs Map.empty IntMap.empty
|
(fid_cnt2,funs_cnt2,funs2,prods0) = mkCncFuns cdefs fid_cnt1 funs_cnt1 funs1 lindefs Map.empty IntMap.empty
|
||||||
in (fid_cnt2,prods,lindefs,linrefs,array (0,funs_cnt2-1) funs2)
|
prods = [(fid,Set.toList prodSet) | (fid,prodSet) <- IntMap.toList prods0]
|
||||||
|
in (fid_cnt2,prods,IntMap.toList lindefs,IntMap.toList linrefs,reverse funs2)
|
||||||
where
|
where
|
||||||
mkCncCats [] fid_cnt funs_cnt funs lindefs linrefs =
|
mkCncCats [] fid_cnt funs_cnt funs lindefs linrefs =
|
||||||
(fid_cnt,funs_cnt,funs,lindefs,linrefs)
|
(fid_cnt,funs_cnt,funs,lindefs,linrefs)
|
||||||
mkCncCats (((m,id),CncCat _ _ _ _ (Just (PMCFG prods0 funs0))):cdefs) fid_cnt funs_cnt funs lindefs linrefs =
|
mkCncCats (((m,id),CncCat _ _ _ _ (Just (PMCFG prods0 funs0))):cdefs) fid_cnt funs_cnt funs lindefs linrefs =
|
||||||
let !funs_cnt' = let (s_funid, e_funid) = bounds funs0
|
let !funs_cnt' = let (s_funid, e_funid) = bounds funs0
|
||||||
@@ -216,17 +234,16 @@ genCncFuns gr am cm ex_seqs ciCmp seqs cdefs fid_cnt cnccats =
|
|||||||
linrefs' = foldl' (toLinRef (am,id) funs_cnt) linrefs prods0
|
linrefs' = foldl' (toLinRef (am,id) funs_cnt) linrefs prods0
|
||||||
funs' = foldl' (toCncFun funs_cnt (m,mkLinDefId id)) funs (assocs funs0)
|
funs' = foldl' (toCncFun funs_cnt (m,mkLinDefId id)) funs (assocs funs0)
|
||||||
in mkCncCats cdefs fid_cnt funs_cnt' funs' lindefs' linrefs'
|
in mkCncCats cdefs fid_cnt funs_cnt' funs' lindefs' linrefs'
|
||||||
mkCncCats (_ :cdefs) fid_cnt funs_cnt funs lindefs linrefs =
|
mkCncCats (_ :cdefs) fid_cnt funs_cnt funs lindefs linrefs =
|
||||||
mkCncCats cdefs fid_cnt funs_cnt funs lindefs linrefs
|
mkCncCats cdefs fid_cnt funs_cnt funs lindefs linrefs
|
||||||
|
|
||||||
mkCncFuns [] fid_cnt funs_cnt funs lindefs crc prods =
|
mkCncFuns [] fid_cnt funs_cnt funs lindefs crc prods =
|
||||||
(fid_cnt,funs_cnt,funs,prods)
|
(fid_cnt,funs_cnt,funs,prods)
|
||||||
mkCncFuns (((m,id),CncFun _ _ _ (Just (PMCFG prods0 funs0))):cdefs) fid_cnt funs_cnt funs lindefs crc prods =
|
mkCncFuns (((m,id),CncFun _ _ _ (Just (PMCFG prods0 funs0))):cdefs) fid_cnt funs_cnt funs lindefs crc prods =
|
||||||
let ---Ok ty_C = fmap GM.typeForm (Look.lookupFunType gr am id)
|
let ty_C = err error (\x -> x) $ fmap GM.typeForm (Look.lookupFunType gr am id)
|
||||||
ty_C = err error (\x -> x) $ fmap GM.typeForm (Look.lookupFunType gr am id)
|
|
||||||
!funs_cnt' = let (s_funid, e_funid) = bounds funs0
|
!funs_cnt' = let (s_funid, e_funid) = bounds funs0
|
||||||
in funs_cnt+(e_funid-s_funid+1)
|
in funs_cnt+(e_funid-s_funid+1)
|
||||||
!(fid_cnt',crc',prods')
|
!(fid_cnt',crc',prods')
|
||||||
= foldl' (toProd lindefs ty_C funs_cnt)
|
= foldl' (toProd lindefs ty_C funs_cnt)
|
||||||
(fid_cnt,crc,prods) prods0
|
(fid_cnt,crc,prods) prods0
|
||||||
funs' = foldl' (toCncFun funs_cnt (m,id)) funs (assocs funs0)
|
funs' = foldl' (toCncFun funs_cnt (m,id)) funs (assocs funs0)
|
||||||
@@ -234,23 +251,23 @@ genCncFuns gr am cm ex_seqs ciCmp seqs cdefs fid_cnt cnccats =
|
|||||||
mkCncFuns (_ :cdefs) fid_cnt funs_cnt funs lindefs crc prods =
|
mkCncFuns (_ :cdefs) fid_cnt funs_cnt funs lindefs crc prods =
|
||||||
mkCncFuns cdefs fid_cnt funs_cnt funs lindefs crc prods
|
mkCncFuns cdefs fid_cnt funs_cnt funs lindefs crc prods
|
||||||
|
|
||||||
toProd lindefs (ctxt_C,res_C,_) offs st (Production fid0 funid0 args0) =
|
toProd lindefs (ctxt_C,res_C,_) offs st (A.Production fid0 funid0 args0) =
|
||||||
let !((fid_cnt,crc,prods),args) = mapAccumL mkArg st (zip ctxt_C args0)
|
let !((fid_cnt,crc,prods),args) = mapAccumL mkArg st (zip ctxt_C args0)
|
||||||
set0 = Set.fromList (map (C.PApply (offs+funid0)) (sequence args))
|
set0 = Set.fromList (map (PApply (offs+funid0)) (sequence args))
|
||||||
fid = mkFId res_C fid0
|
fid = mkFId res_C fid0
|
||||||
!prods' = case IntMap.lookup fid prods of
|
!prods' = case IntMap.lookup fid prods of
|
||||||
Just set -> IntMap.insert fid (Set.union set0 set) prods
|
Just set -> IntMap.insert fid (Set.union set0 set) prods
|
||||||
Nothing -> IntMap.insert fid set0 prods
|
Nothing -> IntMap.insert fid set0 prods
|
||||||
in (fid_cnt,crc,prods')
|
in (fid_cnt,crc,prods')
|
||||||
where
|
where
|
||||||
mkArg st@(fid_cnt,crc,prods) ((_,_,ty),fid0s ) =
|
mkArg st@(fid_cnt,crc,prods) ((_,_,ty),fid0s) =
|
||||||
case fid0s of
|
case fid0s of
|
||||||
[fid0] -> (st,map (flip C.PArg (mkFId arg_C fid0)) ctxt)
|
[fid0] -> (st,map (flip PArg (mkFId arg_C fid0)) ctxt)
|
||||||
fid0s -> case Map.lookup fids crc of
|
fid0s -> case Map.lookup fids crc of
|
||||||
Just fid -> (st,map (flip C.PArg fid) ctxt)
|
Just fid -> (st,map (flip PArg fid) ctxt)
|
||||||
Nothing -> let !crc' = Map.insert fids fid_cnt crc
|
Nothing -> let !crc' = Map.insert fids fid_cnt crc
|
||||||
!prods' = IntMap.insert fid_cnt (Set.fromList (map C.PCoerce fids)) prods
|
!prods' = IntMap.insert fid_cnt (Set.fromList (map PCoerce fids)) prods
|
||||||
in ((fid_cnt+1,crc',prods'),map (flip C.PArg fid_cnt) ctxt)
|
in ((fid_cnt+1,crc',prods'),map (flip PArg fid_cnt) ctxt)
|
||||||
where
|
where
|
||||||
(hargs_C,arg_C) = GM.catSkeleton ty
|
(hargs_C,arg_C) = GM.catSkeleton ty
|
||||||
ctxt = mapM (mkCtxt lindefs) hargs_C
|
ctxt = mapM (mkCtxt lindefs) hargs_C
|
||||||
@@ -258,14 +275,14 @@ genCncFuns gr am cm ex_seqs ciCmp seqs cdefs fid_cnt cnccats =
|
|||||||
|
|
||||||
mkLinDefId id = prefixIdent "lindef " id
|
mkLinDefId id = prefixIdent "lindef " id
|
||||||
|
|
||||||
toLinDef res offs lindefs (Production fid0 funid0 args) =
|
toLinDef res offs lindefs (A.Production fid0 funid0 args) =
|
||||||
if args == [[fidVar]]
|
if args == [[fidVar]]
|
||||||
then IntMap.insertWith (++) fid [offs+funid0] lindefs
|
then IntMap.insertWith (++) fid [offs+funid0] lindefs
|
||||||
else lindefs
|
else lindefs
|
||||||
where
|
where
|
||||||
fid = mkFId res fid0
|
fid = mkFId res fid0
|
||||||
|
|
||||||
toLinRef res offs linrefs (Production fid0 funid0 [fargs]) =
|
toLinRef res offs linrefs (A.Production fid0 funid0 [fargs]) =
|
||||||
if fid0 == fidVar
|
if fid0 == fidVar
|
||||||
then foldr (\fid -> IntMap.insertWith (++) fid [offs+funid0]) linrefs fids
|
then foldr (\fid -> IntMap.insertWith (++) fid [offs+funid0]) linrefs fids
|
||||||
else linrefs
|
else linrefs
|
||||||
@@ -273,20 +290,20 @@ genCncFuns gr am cm ex_seqs ciCmp seqs cdefs fid_cnt cnccats =
|
|||||||
fids = map (mkFId res) fargs
|
fids = map (mkFId res) fargs
|
||||||
|
|
||||||
mkFId (_,cat) fid0 =
|
mkFId (_,cat) fid0 =
|
||||||
case Map.lookup (i2i cat) cnccats of
|
case Map.lookup (i2i cat) cnccat_ranges of
|
||||||
Just (C.CncCat s e _) -> s+fid0
|
Just (s,e) -> s+fid0
|
||||||
Nothing -> error ("GrammarToPGF.mkFId: missing category "++showIdent cat)
|
Nothing -> error ("GrammarToPGF.mkFId: missing category "++showIdent cat)
|
||||||
|
|
||||||
mkCtxt lindefs (_,cat) =
|
mkCtxt lindefs (_,cat) =
|
||||||
case Map.lookup (i2i cat) cnccats of
|
case Map.lookup (i2i cat) cnccat_ranges of
|
||||||
Just (C.CncCat s e _) -> [(C.fidVar,fid) | fid <- [s..e], Just _ <- [IntMap.lookup fid lindefs]]
|
Just (s,e) -> [(fid,fid) | fid <- [s..e], Just _ <- [IntMap.lookup fid lindefs]]
|
||||||
Nothing -> error "GrammarToPGF.mkCtxt failed"
|
Nothing -> error "GrammarToPGF.mkCtxt failed"
|
||||||
|
|
||||||
toCncFun offs (m,id) funs (funid0,lins0) =
|
toCncFun offs (m,id) funs (funid0,lins0) =
|
||||||
let mseqs = case lookupModule gr m of
|
let mseqs = case lookupModule gr m of
|
||||||
Ok (ModInfo{mseqs=Just mseqs}) -> mseqs
|
Ok (ModInfo{mseqs=Just mseqs}) -> mseqs
|
||||||
_ -> ex_seqs
|
_ -> ex_seqs
|
||||||
in (offs+funid0,C.CncFun (i2i id) (amap (newIndex mseqs) lins0)):funs
|
in (i2i id, map (newIndex mseqs) (elems lins0)):funs
|
||||||
where
|
where
|
||||||
newIndex mseqs i = binSearch (mseqs ! i) seqs (bounds seqs)
|
newIndex mseqs i = binSearch (mseqs ! i) seqs (bounds seqs)
|
||||||
|
|
||||||
@@ -299,8 +316,9 @@ genCncFuns gr am cm ex_seqs ciCmp seqs cdefs fid_cnt cnccats =
|
|||||||
where
|
where
|
||||||
k = (i+j) `div` 2
|
k = (i+j) `div` 2
|
||||||
|
|
||||||
|
|
||||||
genPrintNames cdefs =
|
genPrintNames cdefs =
|
||||||
Map.fromAscList [(i2i id, name) | ((m,id),info) <- cdefs, name <- prn info]
|
[(i2i id, name) | ((m,id),info) <- cdefs, name <- prn info]
|
||||||
where
|
where
|
||||||
prn (CncFun _ _ (Just (L _ tr)) _) = [flatten tr]
|
prn (CncFun _ _ (Just (L _ tr)) _) = [flatten tr]
|
||||||
prn (CncCat _ _ _ (Just (L _ tr)) _) = [flatten tr]
|
prn (CncCat _ _ _ (Just (L _ tr)) _) = [flatten tr]
|
||||||
@@ -312,6 +330,7 @@ genPrintNames cdefs =
|
|||||||
|
|
||||||
mkArray lst = listArray (0,length lst-1) lst
|
mkArray lst = listArray (0,length lst-1) lst
|
||||||
mkMapArray map = array (0,Map.size map-1) [(v,k) | (k,v) <- Map.toList map]
|
mkMapArray map = array (0,Map.size map-1) [(v,k) | (k,v) <- Map.toList map]
|
||||||
|
mkSetArray set = listArray (0,Set.size set-1) (Set.toList set)
|
||||||
|
|
||||||
-- The following is a version of Data.List.sortBy which together
|
-- The following is a version of Data.List.sortBy which together
|
||||||
-- with the sorting also eliminates duplicate values
|
-- with the sorting also eliminates duplicate values
|
||||||
@@ -358,73 +377,71 @@ sortNubBy cmp = mergeAll . sequences
|
|||||||
-- The following function does case-insensitive comparison of sequences.
|
-- The following function does case-insensitive comparison of sequences.
|
||||||
-- This is used to allow case-insensitive parsing, while
|
-- This is used to allow case-insensitive parsing, while
|
||||||
-- the linearizer still has access to the original cases.
|
-- the linearizer still has access to the original cases.
|
||||||
compareCaseInsensitve s1 s2 =
|
|
||||||
compareSeq (elems s1) (elems s2)
|
|
||||||
where
|
|
||||||
compareSeq [] [] = EQ
|
|
||||||
compareSeq [] _ = LT
|
|
||||||
compareSeq _ [] = GT
|
|
||||||
compareSeq (x:xs) (y:ys) =
|
|
||||||
case compareSym x y of
|
|
||||||
EQ -> compareSeq xs ys
|
|
||||||
x -> x
|
|
||||||
|
|
||||||
compareSym s1 s2 =
|
compareCaseInsensitive [] [] = EQ
|
||||||
case s1 of
|
compareCaseInsensitive [] _ = LT
|
||||||
D.SymCat d1 r1
|
compareCaseInsensitive _ [] = GT
|
||||||
-> case s2 of
|
compareCaseInsensitive (x:xs) (y:ys) =
|
||||||
D.SymCat d2 r2
|
case compareSym x y of
|
||||||
-> case compare d1 d2 of
|
EQ -> compareCaseInsensitive xs ys
|
||||||
EQ -> r1 `compare` r2
|
x -> x
|
||||||
x -> x
|
where
|
||||||
_ -> LT
|
compareSym s1 s2 =
|
||||||
D.SymLit d1 r1
|
case s1 of
|
||||||
-> case s2 of
|
SymCat d1 r1
|
||||||
D.SymCat {} -> GT
|
-> case s2 of
|
||||||
D.SymLit d2 r2
|
SymCat d2 r2
|
||||||
-> case compare d1 d2 of
|
-> case compare d1 d2 of
|
||||||
EQ -> r1 `compare` r2
|
EQ -> r1 `compare` r2
|
||||||
x -> x
|
x -> x
|
||||||
_ -> LT
|
_ -> LT
|
||||||
D.SymVar d1 r1
|
SymLit d1 r1
|
||||||
-> if tagToEnum# (getTag s2 ># 2#)
|
-> case s2 of
|
||||||
then LT
|
SymCat {} -> GT
|
||||||
else case s2 of
|
SymLit d2 r2
|
||||||
D.SymVar d2 r2
|
-> case compare d1 d2 of
|
||||||
-> case compare d1 d2 of
|
EQ -> r1 `compare` r2
|
||||||
EQ -> r1 `compare` r2
|
x -> x
|
||||||
x -> x
|
_ -> LT
|
||||||
_ -> GT
|
SymVar d1 r1
|
||||||
D.SymKS t1
|
-> if tagToEnum# (getTag s2 ># 2#)
|
||||||
-> if tagToEnum# (getTag s2 ># 3#)
|
then LT
|
||||||
then LT
|
else case s2 of
|
||||||
else case s2 of
|
SymVar d2 r2
|
||||||
D.SymKS t2 -> t1 `compareToken` t2
|
-> case compare d1 d2 of
|
||||||
_ -> GT
|
EQ -> r1 `compare` r2
|
||||||
D.SymKP a1 b1
|
x -> x
|
||||||
-> if tagToEnum# (getTag s2 ># 4#)
|
_ -> GT
|
||||||
then LT
|
SymKS t1
|
||||||
else case s2 of
|
-> if tagToEnum# (getTag s2 ># 3#)
|
||||||
D.SymKP a2 b2
|
then LT
|
||||||
-> case compare a1 a2 of
|
else case s2 of
|
||||||
EQ -> b1 `compare` b2
|
SymKS t2 -> t1 `compareToken` t2
|
||||||
x -> x
|
_ -> GT
|
||||||
_ -> GT
|
SymKP a1 b1
|
||||||
_ -> let t1 = getTag s1
|
-> if tagToEnum# (getTag s2 ># 4#)
|
||||||
t2 = getTag s2
|
then LT
|
||||||
in if tagToEnum# (t1 <# t2)
|
else case s2 of
|
||||||
then LT
|
SymKP a2 b2
|
||||||
else if tagToEnum# (t1 ==# t2)
|
-> case compare a1 a2 of
|
||||||
then EQ
|
EQ -> b1 `compare` b2
|
||||||
else GT
|
x -> x
|
||||||
|
_ -> GT
|
||||||
|
_ -> let t1 = getTag s1
|
||||||
|
t2 = getTag s2
|
||||||
|
in if tagToEnum# (t1 <# t2)
|
||||||
|
then LT
|
||||||
|
else if tagToEnum# (t1 ==# t2)
|
||||||
|
then EQ
|
||||||
|
else GT
|
||||||
|
|
||||||
compareToken [] [] = EQ
|
compareToken [] [] = EQ
|
||||||
compareToken [] _ = LT
|
compareToken [] _ = LT
|
||||||
compareToken _ [] = GT
|
compareToken _ [] = GT
|
||||||
compareToken (x:xs) (y:ys)
|
compareToken (x:xs) (y:ys)
|
||||||
| x == y = compareToken xs ys
|
| x == y = compareToken xs ys
|
||||||
| otherwise = case compare (toLower x) (toLower y) of
|
| otherwise = case compare (toLower x) (toLower y) of
|
||||||
EQ -> case compareToken xs ys of
|
EQ -> case compareToken xs ys of
|
||||||
EQ -> compare x y
|
EQ -> compare x y
|
||||||
x -> x
|
x -> x
|
||||||
x -> x
|
x -> x
|
||||||
|
|||||||
@@ -21,23 +21,16 @@ import GF.Grammar.Printer
|
|||||||
import GF.Grammar.Macros
|
import GF.Grammar.Macros
|
||||||
import GF.Grammar.Lookup
|
import GF.Grammar.Lookup
|
||||||
import GF.Grammar.Predef
|
import GF.Grammar.Predef
|
||||||
--import GF.Compile.Refresh
|
|
||||||
--import GF.Compile.Compute.Concrete
|
|
||||||
import GF.Compile.Compute.ConcreteNew(GlobalEnv,normalForm,resourceValues)
|
import GF.Compile.Compute.ConcreteNew(GlobalEnv,normalForm,resourceValues)
|
||||||
--import GF.Compile.CheckGrammar
|
|
||||||
--import GF.Compile.Update
|
|
||||||
|
|
||||||
import GF.Data.Operations
|
import GF.Data.Operations
|
||||||
--import GF.Infra.CheckM
|
|
||||||
import GF.Infra.Option
|
import GF.Infra.Option
|
||||||
|
|
||||||
import Control.Monad
|
import Control.Monad
|
||||||
--import Data.List
|
|
||||||
import qualified Data.Set as Set
|
import qualified Data.Set as Set
|
||||||
|
import qualified Data.Map as Map
|
||||||
import GF.Text.Pretty
|
import GF.Text.Pretty
|
||||||
import Debug.Trace
|
import Debug.Trace
|
||||||
|
|
||||||
|
|
||||||
-- | partial evaluation of concrete syntax. AR 6\/2001 -- 16\/5\/2003 -- 5\/2\/2005.
|
-- | partial evaluation of concrete syntax. AR 6\/2001 -- 16\/5\/2003 -- 5\/2\/2005.
|
||||||
|
|
||||||
optimizeModule :: Options -> SourceGrammar -> SourceModule -> Err SourceModule
|
optimizeModule :: Options -> SourceGrammar -> SourceModule -> Err SourceModule
|
||||||
@@ -54,7 +47,7 @@ optimizeModule opts sgr m@(name,mi)
|
|||||||
|
|
||||||
updateEvalInfo mi (i,info) = do
|
updateEvalInfo mi (i,info) = do
|
||||||
info <- evalInfo oopts resenv sgr (name,mi) i info
|
info <- evalInfo oopts resenv sgr (name,mi) i info
|
||||||
return (mi{jments=updateTree (i,info) (jments mi)})
|
return (mi{jments=Map.insert i info (jments mi)})
|
||||||
|
|
||||||
evalInfo :: Options -> GlobalEnv -> SourceGrammar -> SourceModule -> Ident -> Info -> Err Info
|
evalInfo :: Options -> GlobalEnv -> SourceGrammar -> SourceModule -> Ident -> Info -> Err Info
|
||||||
evalInfo opts resenv sgr m c info = do
|
evalInfo opts resenv sgr m c info = do
|
||||||
|
|||||||
189
src/compiler/GF/Compile/OptimizePGF.hs
Normal file
189
src/compiler/GF/Compile/OptimizePGF.hs
Normal file
@@ -0,0 +1,189 @@
|
|||||||
|
{-# LANGUAGE BangPatterns #-}
|
||||||
|
module GF.Compile.OptimizePGF(optimizePGF) where
|
||||||
|
|
||||||
|
import PGF2(Cat,Fun)
|
||||||
|
import PGF2.Internal
|
||||||
|
import Data.Array.ST
|
||||||
|
import Data.Array.Unboxed
|
||||||
|
import qualified Data.Map as Map
|
||||||
|
import qualified Data.Set as Set
|
||||||
|
import qualified Data.IntSet as IntSet
|
||||||
|
import qualified Data.IntMap as IntMap
|
||||||
|
import qualified Data.List as List
|
||||||
|
import Control.Monad.ST
|
||||||
|
|
||||||
|
type ConcrData = ([(FId,[FunId])], -- ^ Lindefs
|
||||||
|
[(FId,[FunId])], -- ^ Linrefs
|
||||||
|
[(FId,[Production])], -- ^ Productions
|
||||||
|
[(Fun,[SeqId])], -- ^ Concrete functions (must be sorted by Fun)
|
||||||
|
[[Symbol]], -- ^ Sequences (must be sorted)
|
||||||
|
[(Cat,FId,FId,[String])]) -- ^ Concrete categories
|
||||||
|
|
||||||
|
optimizePGF :: Cat -> ConcrData -> ConcrData
|
||||||
|
optimizePGF startCat = topDownFilter startCat . bottomUpFilter
|
||||||
|
|
||||||
|
catString = "String"
|
||||||
|
catInt = "Int"
|
||||||
|
catFloat = "Float"
|
||||||
|
catVar = "__gfVar"
|
||||||
|
|
||||||
|
topDownFilter :: Cat -> ConcrData -> ConcrData
|
||||||
|
topDownFilter startCat (lindefs,linrefs,prods,cncfuns,sequences,cnccats) =
|
||||||
|
let env0 = (Map.empty,Map.empty)
|
||||||
|
(env1,lindefs') = List.mapAccumL (\env (fid,funids) -> let (env',funids') = List.mapAccumL (optimizeFun fid [PArg [] fidVar]) env funids in (env',(fid,funids')))
|
||||||
|
env0
|
||||||
|
lindefs
|
||||||
|
(env2,linrefs') = List.mapAccumL (\env (fid,funids) -> let (env',funids') = List.mapAccumL (optimizeFun fidVar [PArg [] fid]) env funids in (env',(fid,funids')))
|
||||||
|
env1
|
||||||
|
linrefs
|
||||||
|
(env3,prods') = List.mapAccumL (\env (fid,set) -> let (env',set') = List.mapAccumL (optimizeProd fid) env set in (env',(fid,set')))
|
||||||
|
env2
|
||||||
|
prods
|
||||||
|
cnccats' = map filterCatLabels cnccats
|
||||||
|
(sequences',cncfuns') = env3
|
||||||
|
in (lindefs',linrefs',prods',mkSetArray cncfuns',mkSetArray sequences',cnccats')
|
||||||
|
where
|
||||||
|
cncfuns_array = listArray (0,length cncfuns-1) cncfuns :: Array FunId (Fun, [SeqId])
|
||||||
|
sequences_array = listArray (0,length sequences-1) sequences :: Array SeqId [Symbol]
|
||||||
|
prods_map = IntMap.fromList prods
|
||||||
|
fid2catMap = IntMap.fromList ((fidVar,catVar) : [(fid,cat) | (cat,start,end,lbls) <- cnccats,
|
||||||
|
fid <- [start..end]])
|
||||||
|
|
||||||
|
fid2cat fid =
|
||||||
|
case IntMap.lookup fid fid2catMap of
|
||||||
|
Just cat -> cat
|
||||||
|
Nothing -> case [fid | Just set <- [IntMap.lookup fid prods_map], PCoerce fid <- set] of
|
||||||
|
(fid:_) -> fid2cat fid
|
||||||
|
_ -> error "unknown forest id"
|
||||||
|
|
||||||
|
starts =
|
||||||
|
[(startCat,lbl) | (cat,_,_,lbls) <- cnccats, cat==startCat, lbl <- [0..length lbls-1]]
|
||||||
|
|
||||||
|
allRelations =
|
||||||
|
Map.unionsWith Set.union
|
||||||
|
[rel fid prod | (fid,set) <- prods, prod <- set]
|
||||||
|
where
|
||||||
|
rel fid (PApply funid args) = Map.fromList [((fid2cat fid,lbl),deps args seqid) | (lbl,seqid) <- zip [0..] lin]
|
||||||
|
where
|
||||||
|
(_,lin) = cncfuns_array ! funid
|
||||||
|
rel fid _ = Map.empty
|
||||||
|
|
||||||
|
deps args seqid = Set.fromList [let PArg _ fid = args !! r in (fid2cat fid,d) | SymCat r d <- seq]
|
||||||
|
where
|
||||||
|
seq = sequences_array ! seqid
|
||||||
|
|
||||||
|
-- here we create a mapping from a category to an array of indices.
|
||||||
|
-- An element of the array is equal to -1 if the corresponding index
|
||||||
|
-- is not going to be used in the optimized grammar, or the new index
|
||||||
|
-- if it will be used
|
||||||
|
closure :: Map.Map Cat [Int]
|
||||||
|
closure = runST $ do
|
||||||
|
set <- initSet
|
||||||
|
addLitCat catString set
|
||||||
|
addLitCat catInt set
|
||||||
|
addLitCat catFloat set
|
||||||
|
addLitCat catVar set
|
||||||
|
closureSet set starts
|
||||||
|
doneSet set
|
||||||
|
where
|
||||||
|
initSet :: ST s (Map.Map Cat (STUArray s Int Int))
|
||||||
|
initSet =
|
||||||
|
fmap Map.fromList $ sequence
|
||||||
|
[fmap ((,) cat) (newArray (0,length lbls-1) (-1))
|
||||||
|
| (cat,_,_,lbls) <- cnccats]
|
||||||
|
|
||||||
|
addLitCat cat set =
|
||||||
|
case Map.lookup cat set of
|
||||||
|
Just indices -> writeArray indices 0 0
|
||||||
|
Nothing -> return ()
|
||||||
|
|
||||||
|
closureSet set [] = return ()
|
||||||
|
closureSet set (x@(cat,index):xs) =
|
||||||
|
case Map.lookup cat set of
|
||||||
|
Just indices -> do v <- readArray indices index
|
||||||
|
writeArray indices index 0
|
||||||
|
if v < 0
|
||||||
|
then case Map.lookup x allRelations of
|
||||||
|
Just ys -> closureSet set (Set.toList ys++xs)
|
||||||
|
Nothing -> closureSet set xs
|
||||||
|
else closureSet set xs
|
||||||
|
Nothing -> error "unknown cat"
|
||||||
|
|
||||||
|
doneSet :: Map.Map Cat (STUArray s Int Int) -> ST s (Map.Map Cat [Int])
|
||||||
|
doneSet set =
|
||||||
|
fmap Map.fromAscList $ mapM done (Map.toAscList set)
|
||||||
|
where
|
||||||
|
done (cat,indices) = do
|
||||||
|
indices <- fmap (reindex 0) (getElems indices)
|
||||||
|
return (cat,indices)
|
||||||
|
|
||||||
|
reindex k [] = []
|
||||||
|
reindex k (v:vs)
|
||||||
|
| v < 0 = v : reindex k vs
|
||||||
|
| otherwise = k : reindex (k+1) vs
|
||||||
|
|
||||||
|
optimizeProd res env (PApply funid args) =
|
||||||
|
let (env',funid') = optimizeFun res args env funid
|
||||||
|
in (env', PApply funid' args)
|
||||||
|
optimizeProd res env prod = (env,prod)
|
||||||
|
|
||||||
|
optimizeFun res args (seqs,funs) funid =
|
||||||
|
let (seqs',lin') = List.mapAccumL addUnique seqs [map updateSymbol (sequences_array ! seqid) |
|
||||||
|
(idx,seqid) <- zip (indicesOf res) lin, idx >= 0]
|
||||||
|
(funs',funid') = addUnique funs (fun, lin')
|
||||||
|
in ((seqs',funs'), funid')
|
||||||
|
where
|
||||||
|
(fun,lin) = cncfuns_array ! funid
|
||||||
|
|
||||||
|
indicesOf fid
|
||||||
|
| fid < 0 = [0]
|
||||||
|
| otherwise =
|
||||||
|
case Map.lookup (fid2cat fid) closure of
|
||||||
|
Just indices -> indices
|
||||||
|
Nothing -> error "unknown category"
|
||||||
|
|
||||||
|
addUnique seqs seq =
|
||||||
|
case Map.lookup seq seqs of
|
||||||
|
Just seqid -> (seqs,seqid)
|
||||||
|
Nothing -> let seqid = Map.size seqs
|
||||||
|
in (Map.insert seq seqid seqs, seqid)
|
||||||
|
|
||||||
|
updateSymbol (SymCat r d) = let PArg _ fid = args !! r in SymCat r (indicesOf fid !! d)
|
||||||
|
updateSymbol s = s
|
||||||
|
|
||||||
|
filterCatLabels (cat,start,end,lbls) =
|
||||||
|
case Map.lookup cat closure of
|
||||||
|
Just indices -> let lbls' = [lbl | (idx,lbl) <- zip indices lbls, idx >= 0]
|
||||||
|
in (cat,start,end,lbls')
|
||||||
|
Nothing -> error ("unknown category")
|
||||||
|
|
||||||
|
mkSetArray map = sortSnd (Map.toList map)
|
||||||
|
where
|
||||||
|
sortSnd = List.map fst . List.sortBy (\(_,i) (_,j) -> compare i j)
|
||||||
|
|
||||||
|
|
||||||
|
bottomUpFilter :: ConcrData -> ConcrData
|
||||||
|
bottomUpFilter (lindefs,linrefs,prods,cncfuns,sequences,cnccats) =
|
||||||
|
(lindefs,linrefs,filterProductions IntMap.empty IntSet.empty prods,cncfuns,sequences,cnccats)
|
||||||
|
|
||||||
|
filterProductions prods0 hoc0 prods
|
||||||
|
| prods0 == prods1 = IntMap.toList prods0
|
||||||
|
| otherwise = filterProductions prods1 hoc1 prods
|
||||||
|
where
|
||||||
|
(prods1,hoc1) = foldl foldProdSet (IntMap.empty,IntSet.empty) prods
|
||||||
|
|
||||||
|
foldProdSet (!prods,!hoc) (fid,set)
|
||||||
|
| null set1 = (prods,hoc)
|
||||||
|
| otherwise = (IntMap.insert fid set1 prods,hoc1)
|
||||||
|
where
|
||||||
|
set1 = filter filterRule set
|
||||||
|
hoc1 = foldl accumHOC hoc set1
|
||||||
|
|
||||||
|
filterRule (PApply funid args) = all (\(PArg _ fid) -> isLive fid) args
|
||||||
|
filterRule (PCoerce fid) = isLive fid
|
||||||
|
filterRule _ = True
|
||||||
|
|
||||||
|
isLive fid = isPredefFId fid || IntMap.member fid prods0 || IntSet.member fid hoc0
|
||||||
|
|
||||||
|
accumHOC hoc (PApply funid args) = List.foldl' (\hoc (PArg hypos _) -> List.foldl' (\hoc fid -> IntSet.insert fid hoc) hoc (map snd hypos)) hoc args
|
||||||
|
accumHOC hoc _ = hoc
|
||||||
@@ -16,13 +16,14 @@
|
|||||||
|
|
||||||
module GF.Compile.PGFtoHaskell (grammar2haskell) where
|
module GF.Compile.PGFtoHaskell (grammar2haskell) where
|
||||||
|
|
||||||
import PGF(showCId)
|
import PGF2
|
||||||
import PGF.Internal
|
import PGF2.Internal
|
||||||
|
|
||||||
import GF.Data.Operations
|
import GF.Data.Operations
|
||||||
import GF.Infra.Option
|
import GF.Infra.Option
|
||||||
|
|
||||||
import Data.List --(isPrefixOf, find, intersperse)
|
import Data.List
|
||||||
|
import Data.Maybe(mapMaybe)
|
||||||
import qualified Data.Map as Map
|
import qualified Data.Map as Map
|
||||||
|
|
||||||
type Prefix = String -> String
|
type Prefix = String -> String
|
||||||
@@ -39,7 +40,7 @@ grammar2haskell opts name gr = foldr (++++) [] $
|
|||||||
lexical cat = haskellOption opts HaskellLexical && isLexicalCat opts cat
|
lexical cat = haskellOption opts HaskellLexical && isLexicalCat opts cat
|
||||||
gId | haskellOption opts HaskellNoPrefix = id
|
gId | haskellOption opts HaskellNoPrefix = id
|
||||||
| otherwise = ("G"++)
|
| otherwise = ("G"++)
|
||||||
pragmas | gadt = ["{-# OPTIONS_GHC -fglasgow-exts #-}","{-# LANGUAGE GADTs #-}"]
|
pragmas | gadt = ["{-# OPTIONS_GHC -fglasgow-exts #-}"]
|
||||||
| otherwise = []
|
| otherwise = []
|
||||||
types | gadt = datatypesGADT gId lexical gr'
|
types | gadt = datatypesGADT gId lexical gr'
|
||||||
| otherwise = datatypes gId lexical gr'
|
| otherwise = datatypes gId lexical gr'
|
||||||
@@ -241,7 +242,7 @@ fInstance gId lexical m (cat,rules) =
|
|||||||
then " " ++ gId cat ++ " (fgs t) where\n fgs t = case unApp t of"
|
then " " ++ gId cat ++ " (fgs t) where\n fgs t = case unApp t of"
|
||||||
else " case unApp t of") ++++
|
else " case unApp t of") ++++
|
||||||
unlines [mkInst f xx | (f,xx) <- nonLexicalRules (lexical cat) rules] ++++
|
unlines [mkInst f xx | (f,xx) <- nonLexicalRules (lexical cat) rules] ++++
|
||||||
(if lexical cat then " Just (i,[]) -> " ++ lexicalConstructor cat +++ "(showCId i)" else "") ++++
|
(if lexical cat then " Just (i,[]) -> " ++ lexicalConstructor cat +++ "i" else "") ++++
|
||||||
" _ -> error (\"no" +++ cat ++ " \" ++ show t)"
|
" _ -> error (\"no" +++ cat ++ " \" ++ show t)"
|
||||||
where
|
where
|
||||||
isList = isListCat (cat,rules)
|
isList = isListCat (cat,rules)
|
||||||
@@ -262,18 +263,21 @@ fInstance gId lexical m (cat,rules) =
|
|||||||
--type HSkeleton = [(OIdent, [(OIdent, [OIdent])])]
|
--type HSkeleton = [(OIdent, [(OIdent, [OIdent])])]
|
||||||
hSkeleton :: PGF -> (String,HSkeleton)
|
hSkeleton :: PGF -> (String,HSkeleton)
|
||||||
hSkeleton gr =
|
hSkeleton gr =
|
||||||
(showCId (absname gr),
|
(abstractName gr,
|
||||||
let fs =
|
let fs =
|
||||||
[(showCId c, [(showCId f, map showCId cs) | (f, (cs,_)) <- fs]) |
|
[(c, [(f, cs) | (f, cs,_) <- fs]) |
|
||||||
fs@((_, (_,c)):_) <- fns]
|
fs@((_, _,c):_) <- fns]
|
||||||
in fs ++ [(sc, []) | c <- cts, let sc = showCId c, notElem sc (["Int", "Float", "String"] ++ map fst fs)]
|
in fs ++ [(c, []) | c <- cts, notElem c (["Int", "Float", "String"] ++ map fst fs)]
|
||||||
)
|
)
|
||||||
where
|
where
|
||||||
cts = Map.keys (cats (abstract gr))
|
cts = categories gr
|
||||||
fns = groupBy valtypg (sortBy valtyps (map jty (Map.assocs (funs (abstract gr)))))
|
fns = groupBy valtypg (sortBy valtyps (mapMaybe jty (functions gr)))
|
||||||
valtyps (_, (_,x)) (_, (_,y)) = compare x y
|
valtyps (_,_,x) (_,_,y) = compare x y
|
||||||
valtypg (_, (_,x)) (_, (_,y)) = x == y
|
valtypg (_,_,x) (_,_,y) = x == y
|
||||||
jty (f,(ty,_,_,_)) = (f,catSkeleton ty)
|
jty f = case functionType gr f of
|
||||||
|
Just ty -> let (hypos,valcat,_) = unType ty
|
||||||
|
in Just (f,[argcat | (_,_,ty) <- hypos, let (_,argcat,_) = unType ty],valcat)
|
||||||
|
Nothing -> Nothing
|
||||||
{-
|
{-
|
||||||
updateSkeleton :: OIdent -> HSkeleton -> (OIdent, [OIdent]) -> HSkeleton
|
updateSkeleton :: OIdent -> HSkeleton -> (OIdent, [OIdent]) -> HSkeleton
|
||||||
updateSkeleton cat skel rule =
|
updateSkeleton cat skel rule =
|
||||||
|
|||||||
@@ -1,105 +0,0 @@
|
|||||||
module GF.Compile.PGFtoJS (pgf2js) where
|
|
||||||
|
|
||||||
import PGF(showCId)
|
|
||||||
import PGF.Internal as M
|
|
||||||
import qualified GF.JavaScript.AbsJS as JS
|
|
||||||
import qualified GF.JavaScript.PrintJS as JS
|
|
||||||
|
|
||||||
--import GF.Data.ErrM
|
|
||||||
--import GF.Infra.Option
|
|
||||||
|
|
||||||
--import Control.Monad (mplus)
|
|
||||||
--import Data.Array.Unboxed (UArray)
|
|
||||||
import qualified Data.Array.IArray as Array
|
|
||||||
--import Data.Maybe (fromMaybe)
|
|
||||||
import Data.Map (Map)
|
|
||||||
import qualified Data.Set as Set
|
|
||||||
import qualified Data.Map as Map
|
|
||||||
import qualified Data.IntMap as IntMap
|
|
||||||
|
|
||||||
pgf2js :: PGF -> String
|
|
||||||
pgf2js pgf =
|
|
||||||
JS.printTree $ JS.Program [JS.ElStmt $ JS.SDeclOrExpr $ JS.Decl [JS.DInit (JS.Ident n) grammar]]
|
|
||||||
where
|
|
||||||
n = showCId $ absname pgf
|
|
||||||
as = abstract pgf
|
|
||||||
cs = Map.assocs (concretes pgf)
|
|
||||||
start = showCId $ M.lookStartCat pgf
|
|
||||||
grammar = new "GFGrammar" [js_abstract, js_concrete]
|
|
||||||
js_abstract = abstract2js start as
|
|
||||||
js_concrete = JS.EObj $ map concrete2js cs
|
|
||||||
|
|
||||||
abstract2js :: String -> Abstr -> JS.Expr
|
|
||||||
abstract2js start ds = new "GFAbstract" [JS.EStr start, JS.EObj $ map absdef2js (Map.assocs (funs ds))]
|
|
||||||
|
|
||||||
absdef2js :: (CId,(Type,Int,Maybe ([Equation],[[M.Instr]]),Double)) -> JS.Property
|
|
||||||
absdef2js (f,(typ,_,_,_)) =
|
|
||||||
let (args,cat) = M.catSkeleton typ in
|
|
||||||
JS.Prop (JS.IdentPropName (JS.Ident (showCId f))) (new "Type" [JS.EArray [JS.EStr (showCId x) | x <- args], JS.EStr (showCId cat)])
|
|
||||||
|
|
||||||
lit2js (LStr s) = JS.EStr s
|
|
||||||
lit2js (LInt n) = JS.EInt n
|
|
||||||
lit2js (LFlt d) = JS.EDbl d
|
|
||||||
|
|
||||||
concrete2js :: (CId,Concr) -> JS.Property
|
|
||||||
concrete2js (c,cnc) =
|
|
||||||
JS.Prop l (new "GFConcrete" [mapToJSObj (lit2js) $ cflags cnc,
|
|
||||||
JS.EObj $ [JS.Prop (JS.IntPropName cat) (JS.EArray (map frule2js (Set.toList set))) | (cat,set) <- IntMap.toList (productions cnc)],
|
|
||||||
JS.EArray $ (map ffun2js (Array.elems (cncfuns cnc))),
|
|
||||||
JS.EArray $ (map seq2js (Array.elems (sequences cnc))),
|
|
||||||
JS.EObj $ map cats (Map.assocs (cnccats cnc)),
|
|
||||||
JS.EInt (totalCats cnc)])
|
|
||||||
where
|
|
||||||
l = JS.IdentPropName (JS.Ident (showCId c))
|
|
||||||
{-
|
|
||||||
litslins = [JS.Prop (JS.StringPropName "Int") (JS.EFun [children] [JS.SReturn $ new "Arr" [JS.EIndex (JS.EVar children) (JS.EInt 0)]]),
|
|
||||||
JS.Prop (JS.StringPropName "Float") (JS.EFun [children] [JS.SReturn $ new "Arr" [JS.EIndex (JS.EVar children) (JS.EInt 0)]]),
|
|
||||||
JS.Prop (JS.StringPropName "String") (JS.EFun [children] [JS.SReturn $ new "Arr" [JS.EIndex (JS.EVar children) (JS.EInt 0)]])]
|
|
||||||
-}
|
|
||||||
cats (c,CncCat start end _) = JS.Prop (JS.IdentPropName (JS.Ident (showCId c))) (JS.EObj [JS.Prop (JS.IdentPropName (JS.Ident "s")) (JS.EInt start)
|
|
||||||
,JS.Prop (JS.IdentPropName (JS.Ident "e")) (JS.EInt end)])
|
|
||||||
{-
|
|
||||||
mkStr :: String -> JS.Expr
|
|
||||||
mkStr s = new "Str" [JS.EStr s]
|
|
||||||
|
|
||||||
mkSeq :: [JS.Expr] -> JS.Expr
|
|
||||||
mkSeq [x] = x
|
|
||||||
mkSeq xs = new "Seq" xs
|
|
||||||
|
|
||||||
argIdent :: Integer -> JS.Ident
|
|
||||||
argIdent n = JS.Ident ("x" ++ show n)
|
|
||||||
-}
|
|
||||||
children :: JS.Ident
|
|
||||||
children = JS.Ident "cs"
|
|
||||||
|
|
||||||
frule2js :: Production -> JS.Expr
|
|
||||||
frule2js (PApply funid args) = new "Apply" [JS.EInt funid, JS.EArray (map farg2js args)]
|
|
||||||
frule2js (PCoerce arg) = new "Coerce" [JS.EInt arg]
|
|
||||||
|
|
||||||
farg2js (PArg hypos fid) = new "PArg" (map (JS.EInt . snd) hypos ++ [JS.EInt fid])
|
|
||||||
|
|
||||||
ffun2js (CncFun f lins) = new "CncFun" [JS.EStr (showCId f), JS.EArray (map JS.EInt (Array.elems lins))]
|
|
||||||
|
|
||||||
seq2js :: Array.Array DotPos Symbol -> JS.Expr
|
|
||||||
seq2js seq = JS.EArray [sym2js s | s <- Array.elems seq]
|
|
||||||
|
|
||||||
sym2js :: Symbol -> JS.Expr
|
|
||||||
sym2js (SymCat n l) = new "SymCat" [JS.EInt n, JS.EInt l]
|
|
||||||
sym2js (SymLit n l) = new "SymLit" [JS.EInt n, JS.EInt l]
|
|
||||||
sym2js (SymVar n l) = new "SymVar" [JS.EInt n, JS.EInt l]
|
|
||||||
sym2js (SymKS t) = new "SymKS" [JS.EStr t]
|
|
||||||
sym2js (SymKP ts alts) = new "SymKP" [JS.EArray (map sym2js ts), JS.EArray (map alt2js alts)]
|
|
||||||
sym2js SymBIND = new "SymKS" [JS.EStr "&+"]
|
|
||||||
sym2js SymSOFT_BIND = new "SymKS" [JS.EStr "&+"]
|
|
||||||
sym2js SymSOFT_SPACE = new "SymKS" [JS.EStr "&+"]
|
|
||||||
sym2js SymCAPIT = new "SymKS" [JS.EStr "&|"]
|
|
||||||
sym2js SymALL_CAPIT = new "SymKS" [JS.EStr "&|"]
|
|
||||||
sym2js SymNE = new "SymNE" []
|
|
||||||
|
|
||||||
alt2js (ps,ts) = new "Alt" [JS.EArray (map sym2js ps), JS.EArray (map JS.EStr ts)]
|
|
||||||
|
|
||||||
new :: String -> [JS.Expr] -> JS.Expr
|
|
||||||
new f xs = JS.ENew (JS.Ident f) xs
|
|
||||||
|
|
||||||
mapToJSObj :: (a -> JS.Expr) -> Map CId a -> JS.Expr
|
|
||||||
mapToJSObj f m = JS.EObj [ JS.Prop (JS.IdentPropName (JS.Ident (showCId k))) (f v) | (k,v) <- Map.toList m ]
|
|
||||||
@@ -1,156 +1,110 @@
|
|||||||
module GF.Compile.PGFtoJSON (pgf2json) where
|
module GF.Compile.PGFtoJSON (pgf2json) where
|
||||||
|
|
||||||
import PGF (showCId)
|
import PGF2
|
||||||
import qualified PGF.Internal as M
|
import PGF2.Internal
|
||||||
import PGF.Internal (
|
import Text.JSON
|
||||||
Abstr,
|
|
||||||
CId,
|
|
||||||
CncCat(..),
|
|
||||||
CncFun(..),
|
|
||||||
Concr,
|
|
||||||
DotPos,
|
|
||||||
Equation(..),
|
|
||||||
Literal(..),
|
|
||||||
PArg(..),
|
|
||||||
PGF,
|
|
||||||
Production(..),
|
|
||||||
Symbol(..),
|
|
||||||
Type,
|
|
||||||
absname,
|
|
||||||
abstract,
|
|
||||||
cflags,
|
|
||||||
cnccats,
|
|
||||||
cncfuns,
|
|
||||||
concretes,
|
|
||||||
funs,
|
|
||||||
productions,
|
|
||||||
sequences,
|
|
||||||
totalCats
|
|
||||||
)
|
|
||||||
|
|
||||||
import qualified Text.JSON as JSON
|
|
||||||
import Text.JSON (JSValue(..))
|
|
||||||
|
|
||||||
import qualified Data.Array.IArray as Array
|
|
||||||
import Data.Map (Map)
|
|
||||||
import qualified Data.Set as Set
|
|
||||||
import qualified Data.Map as Map
|
import qualified Data.Map as Map
|
||||||
import qualified Data.IntMap as IntMap
|
|
||||||
|
|
||||||
pgf2json :: PGF -> String
|
pgf2json :: PGF -> String
|
||||||
pgf2json pgf =
|
pgf2json pgf =
|
||||||
JSON.encode $ JSON.makeObj
|
encode $ makeObj
|
||||||
[ ("abstract", json_abstract)
|
[ ("abstract", abstract2json pgf)
|
||||||
, ("concretes", json_concretes)
|
, ("concretes", makeObj $ map concrete2json
|
||||||
]
|
(Map.toList (languages pgf)))
|
||||||
where
|
|
||||||
n = showCId $ absname pgf
|
|
||||||
as = abstract pgf
|
|
||||||
cs = Map.assocs (concretes pgf)
|
|
||||||
start = showCId $ M.lookStartCat pgf
|
|
||||||
json_abstract = abstract2json n start as
|
|
||||||
json_concretes = JSON.makeObj $ map concrete2json cs
|
|
||||||
|
|
||||||
abstract2json :: String -> String -> Abstr -> JSValue
|
|
||||||
abstract2json name start ds =
|
|
||||||
JSON.makeObj
|
|
||||||
[ ("name", mkJSStr name)
|
|
||||||
, ("startcat", mkJSStr start)
|
|
||||||
, ("funs", JSON.makeObj $ map absdef2json (Map.assocs (funs ds)))
|
|
||||||
]
|
]
|
||||||
|
|
||||||
absdef2json :: (CId,(Type,Int,Maybe ([Equation],[[M.Instr]]),Double)) -> (String,JSValue)
|
abstract2json :: PGF -> JSValue
|
||||||
absdef2json (f,(typ,_,_,_)) = (showCId f,sig)
|
abstract2json pgf =
|
||||||
|
makeObj
|
||||||
|
[ ("name", showJSON (abstractName pgf))
|
||||||
|
, ("startcat", showJSON (showType [] (startCat pgf)))
|
||||||
|
, ("funs", makeObj $ map (absdef2json pgf) (functions pgf))
|
||||||
|
]
|
||||||
|
|
||||||
|
absdef2json :: PGF -> Fun -> (String,JSValue)
|
||||||
|
absdef2json pgf f = (f,sig)
|
||||||
where
|
where
|
||||||
(args,cat) = M.catSkeleton typ
|
Just (hypos,cat,_) = fmap unType (functionType pgf f)
|
||||||
sig = JSON.makeObj
|
sig = makeObj
|
||||||
[ ("args", JSArray $ map (mkJSStr.showCId) args)
|
[ ("args", showJSON $ map (\(_,_,ty) -> showType [] ty) hypos)
|
||||||
, ("cat", mkJSStr $ showCId cat)
|
, ("cat", showJSON cat)
|
||||||
]
|
]
|
||||||
|
|
||||||
lit2json :: Literal -> JSValue
|
lit2json :: Literal -> JSValue
|
||||||
lit2json (LStr s) = mkJSStr s
|
lit2json (LStr s) = showJSON s
|
||||||
lit2json (LInt n) = mkJSInt n
|
lit2json (LInt n) = showJSON n
|
||||||
lit2json (LFlt d) = JSRational True (toRational d)
|
lit2json (LFlt d) = showJSON d
|
||||||
|
|
||||||
concrete2json :: (CId,Concr) -> (String,JSValue)
|
concrete2json :: (ConcName,Concr) -> (String,JSValue)
|
||||||
concrete2json (c,cnc) = (showCId c,obj)
|
concrete2json (c,cnc) = (c,obj)
|
||||||
where
|
where
|
||||||
obj = JSON.makeObj
|
obj = makeObj
|
||||||
[ ("flags", JSON.makeObj [ (showCId k, lit2json v) | (k,v) <- Map.toList (cflags cnc) ])
|
[ ("flags", makeObj [(k, lit2json v) | (k,v) <- concrFlags cnc])
|
||||||
, ("productions", JSON.makeObj [ (show cat, JSArray (map frule2json (Set.toList set))) | (cat,set) <- IntMap.toList (productions cnc)])
|
, ("productions", makeObj [(show fid, showJSON (map frule2json (concrProductions cnc fid))) | (_,start,end,_) <- concrCategories cnc, fid <- [start..end]])
|
||||||
, ("functions", JSArray (map ffun2json (Array.elems (cncfuns cnc))))
|
, ("functions", showJSON [ffun2json funid (concrFunction cnc funid) | funid <- [0..concrTotalFuns cnc-1]])
|
||||||
, ("sequences", JSArray (map seq2json (Array.elems (sequences cnc))))
|
, ("sequences", showJSON [seq2json seqid (concrSequence cnc seqid) | seqid <- [0..concrTotalSeqs cnc-1]])
|
||||||
, ("categories", JSON.makeObj $ map cats2json (Map.assocs (cnccats cnc)))
|
, ("categories", makeObj $ map cat2json (concrCategories cnc))
|
||||||
, ("totalfids", mkJSInt (totalCats cnc))
|
, ("totalfids", showJSON (concrTotalCats cnc))
|
||||||
]
|
]
|
||||||
|
|
||||||
cats2json :: (CId, CncCat) -> (String,JSValue)
|
cat2json :: (Cat,FId,FId,[String]) -> (String,JSValue)
|
||||||
cats2json (c,CncCat start end _) = (showCId c, ixs)
|
cat2json (cat,start,end,_) = (cat, ixs)
|
||||||
where
|
where
|
||||||
ixs = JSON.makeObj
|
ixs = makeObj
|
||||||
[ ("start", mkJSInt start)
|
[ ("start", showJSON start)
|
||||||
, ("end", mkJSInt end)
|
, ("end", showJSON end)
|
||||||
]
|
]
|
||||||
|
|
||||||
frule2json :: Production -> JSValue
|
frule2json :: Production -> JSValue
|
||||||
frule2json (PApply fid args) =
|
frule2json (PApply fid args) =
|
||||||
JSON.makeObj
|
makeObj
|
||||||
[ ("type", mkJSStr "Apply")
|
[ ("type", showJSON "Apply")
|
||||||
, ("fid", mkJSInt fid)
|
, ("fid", showJSON fid)
|
||||||
, ("args", JSArray (map farg2json args))
|
, ("args", showJSON (map farg2json args))
|
||||||
]
|
]
|
||||||
frule2json (PCoerce arg) =
|
frule2json (PCoerce arg) =
|
||||||
JSON.makeObj
|
makeObj
|
||||||
[ ("type", mkJSStr "Coerce")
|
[ ("type", showJSON "Coerce")
|
||||||
, ("arg", mkJSInt arg)
|
, ("arg", showJSON arg)
|
||||||
]
|
]
|
||||||
|
|
||||||
farg2json :: PArg -> JSValue
|
farg2json :: PArg -> JSValue
|
||||||
farg2json (PArg hypos fid) =
|
farg2json (PArg hypos fid) =
|
||||||
JSON.makeObj
|
makeObj
|
||||||
[ ("type", mkJSStr "PArg")
|
[ ("type", showJSON "PArg")
|
||||||
, ("hypos", JSArray $ map (mkJSInt . snd) hypos)
|
, ("hypos", JSArray $ map (showJSON . snd) hypos)
|
||||||
, ("fid", mkJSInt fid)
|
, ("fid", showJSON fid)
|
||||||
]
|
]
|
||||||
|
|
||||||
ffun2json :: CncFun -> JSValue
|
ffun2json :: FunId -> (Fun,[SeqId]) -> JSValue
|
||||||
ffun2json (CncFun f lins) =
|
ffun2json funid (fun,seqids) =
|
||||||
JSON.makeObj
|
makeObj
|
||||||
[ ("name", mkJSStr $ showCId f)
|
[ ("name", showJSON fun)
|
||||||
, ("lins", JSArray (map mkJSInt (Array.elems lins)))
|
, ("lins", showJSON seqids)
|
||||||
]
|
]
|
||||||
|
|
||||||
seq2json :: Array.Array DotPos Symbol -> JSValue
|
seq2json :: SeqId -> [Symbol] -> JSValue
|
||||||
seq2json seq = JSArray [sym2json s | s <- Array.elems seq]
|
seq2json seqid seq = showJSON [sym2json sym | sym <- seq]
|
||||||
|
|
||||||
sym2json :: Symbol -> JSValue
|
sym2json :: Symbol -> JSValue
|
||||||
sym2json (SymCat n l) = new "SymCat" [mkJSInt n, mkJSInt l]
|
sym2json (SymCat n l) = new "SymCat" [showJSON n, showJSON l]
|
||||||
sym2json (SymLit n l) = new "SymLit" [mkJSInt n, mkJSInt l]
|
sym2json (SymLit n l) = new "SymLit" [showJSON n, showJSON l]
|
||||||
sym2json (SymVar n l) = new "SymVar" [mkJSInt n, mkJSInt l]
|
sym2json (SymVar n l) = new "SymVar" [showJSON n, showJSON l]
|
||||||
sym2json (SymKS t) = new "SymKS" [mkJSStr t]
|
sym2json (SymKS t) = new "SymKS" [showJSON t]
|
||||||
sym2json (SymKP ts alts) = new "SymKP" [JSArray (map sym2json ts), JSArray (map alt2json alts)]
|
sym2json (SymKP ts alts) = new "SymKP" [JSArray (map sym2json ts), JSArray (map alt2json alts)]
|
||||||
sym2json SymBIND = new "SymKS" [mkJSStr "&+"]
|
sym2json SymBIND = new "SymKS" [showJSON "&+"]
|
||||||
sym2json SymSOFT_BIND = new "SymKS" [mkJSStr "&+"]
|
sym2json SymSOFT_BIND = new "SymKS" [showJSON "&+"]
|
||||||
sym2json SymSOFT_SPACE = new "SymKS" [mkJSStr "&+"]
|
sym2json SymSOFT_SPACE = new "SymKS" [showJSON "&+"]
|
||||||
sym2json SymCAPIT = new "SymKS" [mkJSStr "&|"]
|
sym2json SymCAPIT = new "SymKS" [showJSON "&|"]
|
||||||
sym2json SymALL_CAPIT = new "SymKS" [mkJSStr "&|"]
|
sym2json SymALL_CAPIT = new "SymKS" [showJSON "&|"]
|
||||||
sym2json SymNE = new "SymNE" []
|
sym2json SymNE = new "SymNE" []
|
||||||
|
|
||||||
alt2json :: ([Symbol],[String]) -> JSValue
|
alt2json :: ([Symbol],[String]) -> JSValue
|
||||||
alt2json (ps,ts) = new "Alt" [JSArray (map sym2json ps), JSArray (map mkJSStr ts)]
|
alt2json (ps,ts) = new "Alt" [showJSON (map sym2json ps), showJSON ts]
|
||||||
|
|
||||||
new :: String -> [JSValue] -> JSValue
|
new :: String -> [JSValue] -> JSValue
|
||||||
new f xs =
|
new f xs =
|
||||||
JSON.makeObj
|
makeObj
|
||||||
[ ("type", mkJSStr f)
|
[ ("type", showJSON f)
|
||||||
, ("args", JSArray xs)
|
, ("args", showJSON xs)
|
||||||
]
|
]
|
||||||
|
|
||||||
-- | Make JSON value from string
|
|
||||||
mkJSStr :: String -> JSValue
|
|
||||||
mkJSStr = JSString . JSON.toJSString
|
|
||||||
|
|
||||||
-- | Make JSON value from integer
|
|
||||||
mkJSInt :: Integral a => a -> JSValue
|
|
||||||
mkJSInt = JSRational False . toRational
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
module GF.Compile.PGFtoJava (grammar2java) where
|
module GF.Compile.PGFtoJava (grammar2java) where
|
||||||
|
|
||||||
import PGF
|
import PGF2
|
||||||
import Data.Maybe(maybe)
|
import Data.Maybe(maybe)
|
||||||
import Data.List(intercalate)
|
import Data.List(intercalate)
|
||||||
import GF.Infra.Option
|
import GF.Infra.Option
|
||||||
@@ -24,9 +24,8 @@ javaPreamble name =
|
|||||||
]
|
]
|
||||||
|
|
||||||
javaMethod gr fun =
|
javaMethod gr fun =
|
||||||
" public static Expr "++name++"("++arg_decls++") { return new Expr("++show name++args++"); }"
|
" public static Expr "++fun++"("++arg_decls++") { return new Expr("++show fun++args++"); }"
|
||||||
where
|
where
|
||||||
name = showCId fun
|
|
||||||
arity = maybe 0 getArrity (functionType gr fun)
|
arity = maybe 0 getArrity (functionType gr fun)
|
||||||
vars = ['e':show i | i <- [1..arity]]
|
vars = ['e':show i | i <- [1..arity]]
|
||||||
|
|
||||||
|
|||||||
@@ -1,262 +0,0 @@
|
|||||||
----------------------------------------------------------------------
|
|
||||||
-- |
|
|
||||||
-- Module : PGFtoProlog
|
|
||||||
-- Maintainer : Peter Ljunglöf
|
|
||||||
--
|
|
||||||
-- exports a GF grammar into a Prolog module
|
|
||||||
-----------------------------------------------------------------------------
|
|
||||||
|
|
||||||
module GF.Compile.PGFtoProlog (grammar2prolog) where
|
|
||||||
|
|
||||||
import PGF(mkCId,wildCId,showCId)
|
|
||||||
import PGF.Internal
|
|
||||||
--import PGF.Macros
|
|
||||||
|
|
||||||
import GF.Data.Operations
|
|
||||||
|
|
||||||
import qualified Data.Array.IArray as Array
|
|
||||||
import qualified Data.Set as Set
|
|
||||||
import qualified Data.Map as Map
|
|
||||||
import qualified Data.IntMap as IntMap
|
|
||||||
import Data.Char (isAlphaNum, isAscii, isAsciiLower, isAsciiUpper, ord)
|
|
||||||
import Data.List (isPrefixOf, mapAccumL)
|
|
||||||
|
|
||||||
grammar2prolog :: PGF -> String
|
|
||||||
grammar2prolog pgf
|
|
||||||
= ("%% This file was automatically generated by GF" +++++
|
|
||||||
":- style_check(-singleton)." +++++
|
|
||||||
plFacts wildCId "abstract" 1 "(?AbstractName)"
|
|
||||||
[[plp name]] ++++
|
|
||||||
plFacts wildCId "concrete" 2 "(?AbstractName, ?ConcreteName)"
|
|
||||||
[[plp name, plp cncname] |
|
|
||||||
cncname <- Map.keys (concretes pgf)] ++++
|
|
||||||
plFacts wildCId "flag" 2 "(?Flag, ?Value): global flags"
|
|
||||||
[[plp f, plp v] |
|
|
||||||
(f, v) <- Map.assocs (gflags pgf)] ++++
|
|
||||||
plAbstract name (abstract pgf) ++++
|
|
||||||
unlines (map plConcrete (Map.assocs (concretes pgf)))
|
|
||||||
)
|
|
||||||
where name = absname pgf
|
|
||||||
|
|
||||||
----------------------------------------------------------------------
|
|
||||||
-- abstract syntax
|
|
||||||
|
|
||||||
plAbstract :: CId -> Abstr -> String
|
|
||||||
plAbstract name abs
|
|
||||||
= (plHeader "Abstract syntax" ++++
|
|
||||||
plFacts name "flag" 2 "(?Flag, ?Value): flags for abstract syntax"
|
|
||||||
[[plp f, plp v] |
|
|
||||||
(f, v) <- Map.assocs (aflags abs)] ++++
|
|
||||||
plFacts name "cat" 2 "(?Type, ?[X:Type,...])"
|
|
||||||
[[plType cat args, plHypos hypos'] |
|
|
||||||
(cat, (hypos,_,_)) <- Map.assocs (cats abs),
|
|
||||||
let ((_, subst), hypos') = mapAccumL alphaConvertHypo emptyEnv hypos,
|
|
||||||
let args = reverse [EFun x | (_,x) <- subst]] ++++
|
|
||||||
plFacts name "fun" 3 "(?Fun, ?Type, ?[X:Type,...])"
|
|
||||||
[[plp fun, plType cat args, plHypos hypos] |
|
|
||||||
(fun, (typ, _, _, _)) <- Map.assocs (funs abs),
|
|
||||||
let (_, DTyp hypos cat args) = alphaConvert emptyEnv typ] ++++
|
|
||||||
plFacts name "def" 2 "(?Fun, ?Expr)"
|
|
||||||
[[plp fun, plp expr] |
|
|
||||||
(fun, (_, _, Just (eqs,_), _)) <- Map.assocs (funs abs),
|
|
||||||
let (_, expr) = alphaConvert emptyEnv eqs]
|
|
||||||
)
|
|
||||||
where plType cat args = plTerm (plp cat) (map plp args)
|
|
||||||
plHypos hypos = plList [plOper ":" (plp x) (plp ty) | (_, x, ty) <- hypos]
|
|
||||||
|
|
||||||
----------------------------------------------------------------------
|
|
||||||
-- concrete syntax
|
|
||||||
|
|
||||||
plConcrete :: (CId, Concr) -> String
|
|
||||||
plConcrete (name, cnc)
|
|
||||||
= (plHeader ("Concrete syntax: " ++ plp name) ++++
|
|
||||||
plFacts name "flag" 2 "(?Flag, ?Value): flags for concrete syntax"
|
|
||||||
[[plp f, plp v] |
|
|
||||||
(f, v) <- Map.assocs (cflags cnc)] ++++
|
|
||||||
plFacts name "printname" 2 "(?AbsFun/AbsCat, ?Atom)"
|
|
||||||
[[plp f, plp n] |
|
|
||||||
(f, n) <- Map.assocs (printnames cnc)] ++++
|
|
||||||
plFacts name "lindef" 2 "(?CncCat, ?CncFun)"
|
|
||||||
[[plCat cat, plFun fun] |
|
|
||||||
(cat, funs) <- IntMap.assocs (lindefs cnc),
|
|
||||||
fun <- funs] ++++
|
|
||||||
plFacts name "prod" 3 "(?CncCat, ?CncFun, ?[CncCat])"
|
|
||||||
[[plCat cat, fun, plTerm "c" (map plCat args)] |
|
|
||||||
(cat, set) <- IntMap.toList (productions cnc),
|
|
||||||
(fun, args) <- map plProduction (Set.toList set)] ++++
|
|
||||||
plFacts name "cncfun" 3 "(?CncFun, ?[Seq,...], ?AbsFun)"
|
|
||||||
[[plFun fun, plTerm "s" (map plSeq (Array.elems lins)), plp absfun] |
|
|
||||||
(fun, CncFun absfun lins) <- Array.assocs (cncfuns cnc)] ++++
|
|
||||||
plFacts name "seq" 2 "(?Seq, ?[Term])"
|
|
||||||
[[plSeq seq, plp (Array.elems symbols)] |
|
|
||||||
(seq, symbols) <- Array.assocs (sequences cnc)] ++++
|
|
||||||
plFacts name "cnccat" 2 "(?AbsCat, ?[CnCCat])"
|
|
||||||
[[plp cat, plList (map plCat [start..end])] |
|
|
||||||
(cat, CncCat start end _) <- Map.assocs (cnccats cnc)]
|
|
||||||
)
|
|
||||||
where plProduction (PCoerce arg) = ("-", [arg])
|
|
||||||
plProduction (PApply funid args) = (plFun funid, [fid | PArg hypos fid <- args])
|
|
||||||
|
|
||||||
----------------------------------------------------------------------
|
|
||||||
-- prolog-printing pgf datatypes
|
|
||||||
|
|
||||||
instance PLPrint Type where
|
|
||||||
plp (DTyp hypos cat args)
|
|
||||||
| null hypos = result
|
|
||||||
| otherwise = plOper " -> " plHypos result
|
|
||||||
where result = plTerm (plp cat) (map plp args)
|
|
||||||
plHypos = plList [plOper ":" (plp x) (plp ty) | (_,x,ty) <- hypos]
|
|
||||||
|
|
||||||
instance PLPrint Expr where
|
|
||||||
plp (EFun x) = plp x
|
|
||||||
plp (EAbs _ x e)= plOper "^" (plp x) (plp e)
|
|
||||||
plp (EApp e e') = plOper " * " (plp e) (plp e')
|
|
||||||
plp (ELit lit) = plp lit
|
|
||||||
plp (EMeta n) = "Meta_" ++ show n
|
|
||||||
|
|
||||||
instance PLPrint Patt where
|
|
||||||
plp (PVar x) = plp x
|
|
||||||
plp (PApp f ps) = plOper " * " (plp f) (plp ps)
|
|
||||||
plp (PLit lit) = plp lit
|
|
||||||
|
|
||||||
instance PLPrint Equation where
|
|
||||||
plp (Equ patterns result) = plOper ":" (plp patterns) (plp result)
|
|
||||||
|
|
||||||
instance PLPrint CId where
|
|
||||||
plp cid | isLogicalVariable str || cid == wildCId = plVar str
|
|
||||||
| otherwise = plAtom str
|
|
||||||
where str = showCId cid
|
|
||||||
|
|
||||||
instance PLPrint Literal where
|
|
||||||
plp (LStr s) = plp s
|
|
||||||
plp (LInt n) = plp (show n)
|
|
||||||
plp (LFlt f) = plp (show f)
|
|
||||||
|
|
||||||
instance PLPrint Symbol where
|
|
||||||
plp (SymCat n l) = plOper ":" (show n) (show l)
|
|
||||||
plp (SymLit n l) = plTerm "lit" [show n, show l]
|
|
||||||
plp (SymVar n l) = plTerm "var" [show n, show l]
|
|
||||||
plp (SymKS t) = plAtom t
|
|
||||||
plp (SymKP ts alts) = plTerm "pre" [plList (map plp ts), plList (map plAlt alts)]
|
|
||||||
where plAlt (ps,ts) = plOper "/" (plList (map plp ps)) (plList (map plAtom ts))
|
|
||||||
|
|
||||||
class PLPrint a where
|
|
||||||
plp :: a -> String
|
|
||||||
plps :: [a] -> String
|
|
||||||
plps = plList . map plp
|
|
||||||
|
|
||||||
instance PLPrint Char where
|
|
||||||
plp c = plAtom [c]
|
|
||||||
plps s = plAtom s
|
|
||||||
|
|
||||||
instance PLPrint a => PLPrint [a] where
|
|
||||||
plp = plps
|
|
||||||
|
|
||||||
----------------------------------------------------------------------
|
|
||||||
-- other prolog-printing functions
|
|
||||||
|
|
||||||
plCat :: Int -> String
|
|
||||||
plCat n = plAtom ('c' : show n)
|
|
||||||
|
|
||||||
plFun :: Int -> String
|
|
||||||
plFun n = plAtom ('f' : show n)
|
|
||||||
|
|
||||||
plSeq :: Int -> String
|
|
||||||
plSeq n = plAtom ('s' : show n)
|
|
||||||
|
|
||||||
plHeader :: String -> String
|
|
||||||
plHeader hdr = "%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%\n%% " ++ hdr ++ "\n"
|
|
||||||
|
|
||||||
plFacts :: CId -> String -> Int -> String -> [[String]] -> String
|
|
||||||
plFacts mod pred arity comment facts = "%% " ++ pred ++ comment ++++ clauses
|
|
||||||
where clauses = (if facts == [] then ":- dynamic " ++ pred ++ "/" ++ show arity ++ ".\n"
|
|
||||||
else unlines [mod' ++ plTerm pred args ++ "." | args <- facts])
|
|
||||||
mod' = if mod == wildCId then "" else plp mod ++ ": "
|
|
||||||
|
|
||||||
plTerm :: String -> [String] -> String
|
|
||||||
plTerm fun args = plAtom fun ++ prParenth (prTList ", " args)
|
|
||||||
|
|
||||||
plList :: [String] -> String
|
|
||||||
plList xs = prBracket (prTList "," xs)
|
|
||||||
|
|
||||||
plOper :: String -> String -> String -> String
|
|
||||||
plOper op a b = prParenth (a ++ op ++ b)
|
|
||||||
|
|
||||||
plVar :: String -> String
|
|
||||||
plVar = varPrefix . concatMap changeNonAlphaNum
|
|
||||||
where varPrefix var@(c:_) | isAsciiUpper c || c=='_' = var
|
|
||||||
| otherwise = "_" ++ var
|
|
||||||
changeNonAlphaNum c | isAlphaNumUnderscore c = [c]
|
|
||||||
| otherwise = "_" ++ show (ord c) ++ "_"
|
|
||||||
|
|
||||||
plAtom :: String -> String
|
|
||||||
plAtom "" = "''"
|
|
||||||
plAtom atom@(c:cs) | isAsciiLower c && all isAlphaNumUnderscore cs
|
|
||||||
|| c == '\'' && cs /= "" && last cs == '\'' = atom
|
|
||||||
| otherwise = "'" ++ changeQuote atom ++ "'"
|
|
||||||
where changeQuote ('\'':cs) = '\\' : '\'' : changeQuote cs
|
|
||||||
changeQuote ('\\':cs) = '\\' : '\\' : changeQuote cs
|
|
||||||
changeQuote (c:cs) = c : changeQuote cs
|
|
||||||
changeQuote "" = ""
|
|
||||||
|
|
||||||
isAlphaNumUnderscore :: Char -> Bool
|
|
||||||
isAlphaNumUnderscore c = (isAscii c && isAlphaNum c) || c == '_'
|
|
||||||
|
|
||||||
----------------------------------------------------------------------
|
|
||||||
-- prolog variables
|
|
||||||
|
|
||||||
createLogicalVariable :: Int -> CId
|
|
||||||
createLogicalVariable n = mkCId (logicalVariablePrefix ++ show n)
|
|
||||||
|
|
||||||
isLogicalVariable :: String -> Bool
|
|
||||||
isLogicalVariable = isPrefixOf logicalVariablePrefix
|
|
||||||
|
|
||||||
logicalVariablePrefix :: String
|
|
||||||
logicalVariablePrefix = "X"
|
|
||||||
|
|
||||||
----------------------------------------------------------------------
|
|
||||||
-- alpha convert variables to (unique) logical variables
|
|
||||||
-- * this is needed if we want to translate variables to Prolog variables
|
|
||||||
-- * used for abstract syntax, not concrete
|
|
||||||
-- * not (yet?) used for variables bound in pattern equations
|
|
||||||
|
|
||||||
type ConvertEnv = (Int, [(CId,CId)])
|
|
||||||
|
|
||||||
emptyEnv :: ConvertEnv
|
|
||||||
emptyEnv = (0, [])
|
|
||||||
|
|
||||||
class AlphaConvert a where
|
|
||||||
alphaConvert :: ConvertEnv -> a -> (ConvertEnv, a)
|
|
||||||
|
|
||||||
instance AlphaConvert a => AlphaConvert [a] where
|
|
||||||
alphaConvert env [] = (env, [])
|
|
||||||
alphaConvert env (a:as) = (env'', a':as')
|
|
||||||
where (env', a') = alphaConvert env a
|
|
||||||
(env'', as') = alphaConvert env' as
|
|
||||||
|
|
||||||
instance AlphaConvert Type where
|
|
||||||
alphaConvert env@(_,subst) (DTyp hypos cat args)
|
|
||||||
= ((ctr,subst), DTyp hypos' cat args')
|
|
||||||
where (env', hypos') = mapAccumL alphaConvertHypo env hypos
|
|
||||||
((ctr,_), args') = alphaConvert env' args
|
|
||||||
|
|
||||||
alphaConvertHypo env (b,x,typ) = ((ctr+1,(x,x'):subst), (b,x',typ'))
|
|
||||||
where ((ctr,subst), typ') = alphaConvert env typ
|
|
||||||
x' = createLogicalVariable ctr
|
|
||||||
|
|
||||||
instance AlphaConvert Expr where
|
|
||||||
alphaConvert (ctr,subst) (EAbs b x e) = ((ctr',subst), EAbs b x' e')
|
|
||||||
where ((ctr',_), e') = alphaConvert (ctr+1,(x,x'):subst) e
|
|
||||||
x' = createLogicalVariable ctr
|
|
||||||
alphaConvert env (EApp e1 e2) = (env'', EApp e1' e2')
|
|
||||||
where (env', e1') = alphaConvert env e1
|
|
||||||
(env'', e2') = alphaConvert env' e2
|
|
||||||
alphaConvert env expr@(EFun i) = (env, maybe expr EFun (lookup i (snd env)))
|
|
||||||
alphaConvert env expr = (env, expr)
|
|
||||||
|
|
||||||
-- pattern variables are not alpha converted
|
|
||||||
-- (but they probably should be...)
|
|
||||||
instance AlphaConvert Equation where
|
|
||||||
alphaConvert env@(_,subst) (Equ patterns result)
|
|
||||||
= ((ctr,subst), Equ patterns result')
|
|
||||||
where ((ctr,_), result') = alphaConvert env result
|
|
||||||
@@ -1,122 +0,0 @@
|
|||||||
----------------------------------------------------------------------
|
|
||||||
-- |
|
|
||||||
-- Module : PGFtoPython
|
|
||||||
-- Maintainer : Peter Ljunglöf
|
|
||||||
--
|
|
||||||
-- exports a GF grammar into a Python module
|
|
||||||
-----------------------------------------------------------------------------
|
|
||||||
|
|
||||||
{-# LANGUAGE FlexibleContexts #-}
|
|
||||||
module GF.Compile.PGFtoPython (pgf2python) where
|
|
||||||
|
|
||||||
import PGF(showCId)
|
|
||||||
import PGF.Internal as M
|
|
||||||
|
|
||||||
import GF.Data.Operations
|
|
||||||
|
|
||||||
import qualified Data.Array.IArray as Array
|
|
||||||
import qualified Data.Set as Set
|
|
||||||
import qualified Data.Map as Map
|
|
||||||
import qualified Data.IntMap as IntMap
|
|
||||||
--import Data.List (intersperse)
|
|
||||||
|
|
||||||
pgf2python :: PGF -> String
|
|
||||||
pgf2python pgf = ("# -*- coding: utf-8 -*-" ++++
|
|
||||||
"# This file was automatically generated by GF" +++++
|
|
||||||
showCId name +++ "=" +++
|
|
||||||
pyDict 1 pyStr id [
|
|
||||||
("flags", pyDict 2 pyCId pyLiteral (Map.assocs (gflags pgf))),
|
|
||||||
("abstract", pyDict 2 pyStr id [
|
|
||||||
("name", pyCId name),
|
|
||||||
("start", pyCId start),
|
|
||||||
("flags", pyDict 3 pyCId pyLiteral (Map.assocs (aflags abs))),
|
|
||||||
("funs", pyDict 3 pyCId pyAbsdef (Map.assocs (funs abs)))
|
|
||||||
]),
|
|
||||||
("concretes", pyDict 2 pyCId pyConcrete (Map.assocs cncs))
|
|
||||||
] ++ "\n")
|
|
||||||
where
|
|
||||||
name = absname pgf
|
|
||||||
start = M.lookStartCat pgf
|
|
||||||
abs = abstract pgf
|
|
||||||
cncs = concretes pgf
|
|
||||||
|
|
||||||
pyAbsdef :: (Type, Int, Maybe ([Equation], [[M.Instr]]), Double) -> String
|
|
||||||
pyAbsdef (typ, _, _, _) = pyTuple 0 id [pyCId cat, pyList 0 pyCId args]
|
|
||||||
where (args, cat) = M.catSkeleton typ
|
|
||||||
|
|
||||||
pyLiteral :: Literal -> String
|
|
||||||
pyLiteral (LStr s) = pyStr s
|
|
||||||
pyLiteral (LInt n) = show n
|
|
||||||
pyLiteral (LFlt d) = show d
|
|
||||||
|
|
||||||
pyConcrete :: Concr -> String
|
|
||||||
pyConcrete cnc = pyDict 3 pyStr id [
|
|
||||||
("flags", pyDict 0 pyCId pyLiteral (Map.assocs (cflags cnc))),
|
|
||||||
("printnames", pyDict 4 pyCId pyStr (Map.assocs (printnames cnc))),
|
|
||||||
("lindefs", pyDict 4 pyCat (pyList 0 pyFun) (IntMap.assocs (lindefs cnc))),
|
|
||||||
("productions", pyDict 4 pyCat pyProds (IntMap.assocs (productions cnc))),
|
|
||||||
("cncfuns", pyDict 4 pyFun pyCncFun (Array.assocs (cncfuns cnc))),
|
|
||||||
("sequences", pyDict 4 pySeq pySymbols (Array.assocs (sequences cnc))),
|
|
||||||
("cnccats", pyDict 4 pyCId pyCncCat (Map.assocs (cnccats cnc))),
|
|
||||||
("size", show (totalCats cnc))
|
|
||||||
]
|
|
||||||
where pyProds prods = pyList 5 pyProduction (Set.toList prods)
|
|
||||||
pyCncCat (CncCat start end _) = pyList 0 pyCat [start..end]
|
|
||||||
pyCncFun (CncFun f lins) = pyTuple 0 id [pyList 0 pySeq (Array.elems lins), pyCId f]
|
|
||||||
pySymbols syms = pyList 0 pySymbol (Array.elems syms)
|
|
||||||
|
|
||||||
pyProduction :: Production -> String
|
|
||||||
pyProduction (PCoerce arg) = pyTuple 0 id [pyStr "", pyList 0 pyCat [arg]]
|
|
||||||
pyProduction (PApply funid args) = pyTuple 0 id [pyFun funid, pyList 0 pyPArg args]
|
|
||||||
where pyPArg (PArg [] fid) = pyCat fid
|
|
||||||
pyPArg (PArg hypos fid) = pyTuple 0 pyCat (fid : map snd hypos)
|
|
||||||
|
|
||||||
pySymbol :: Symbol -> String
|
|
||||||
pySymbol (SymCat n l) = pyTuple 0 show [n, l]
|
|
||||||
pySymbol (SymLit n l) = pyDict 0 pyStr id [("lit", pyTuple 0 show [n, l])]
|
|
||||||
pySymbol (SymVar n l) = pyDict 0 pyStr id [("var", pyTuple 0 show [n, l])]
|
|
||||||
pySymbol (SymKS t) = pyStr t
|
|
||||||
pySymbol (SymKP ts alts) = pyDict 0 pyStr id [("pre", pyList 0 pySymbol ts), ("alts", pyList 0 alt2py alts)]
|
|
||||||
where alt2py (ps,ts) = pyTuple 0 (pyList 0 pyStr) [map pySymbol ps, ts]
|
|
||||||
pySymbol SymBIND = pyStr "&+"
|
|
||||||
pySymbol SymSOFT_BIND = pyStr "&+"
|
|
||||||
pySymbol SymSOFT_SPACE = pyStr "&+"
|
|
||||||
pySymbol SymCAPIT = pyStr "&|"
|
|
||||||
pySymbol SymALL_CAPIT = pyStr "&|"
|
|
||||||
pySymbol SymNE = pyDict 0 pyStr id [("nonExist", pyTuple 0 id [])]
|
|
||||||
|
|
||||||
----------------------------------------------------------------------
|
|
||||||
-- python helpers
|
|
||||||
|
|
||||||
pyDict :: Int -> (k -> String) -> (v -> String) -> [(k, v)] -> String
|
|
||||||
pyDict n pk pv [] = "{}"
|
|
||||||
pyDict n pk pv kvlist = prCurly (pyIndent n ++ prTList ("," ++ pyIndent n) (map pyKV kvlist) ++ pyIndent n)
|
|
||||||
where pyKV (k, v) = pk k ++ ":" ++ pv v
|
|
||||||
|
|
||||||
pyList :: Int -> (v -> String) -> [v] -> String
|
|
||||||
pyList n pv [] = "[]"
|
|
||||||
pyList n pv xs = prBracket (pyIndent n ++ prTList ("," ++ pyIndent n) (map pv xs) ++ pyIndent n)
|
|
||||||
|
|
||||||
pyTuple :: Int -> (v -> String) -> [v] -> String
|
|
||||||
pyTuple n pv [] = "()"
|
|
||||||
pyTuple n pv [x] = prParenth (pyIndent n ++ pv x ++ "," ++ pyIndent n)
|
|
||||||
pyTuple n pv xs = prParenth (pyIndent n ++ prTList ("," ++ pyIndent n) (map pv xs) ++ pyIndent n)
|
|
||||||
|
|
||||||
pyCat :: Int -> String
|
|
||||||
pyCat n = pyStr ('C' : show n)
|
|
||||||
|
|
||||||
pyFun :: Int -> String
|
|
||||||
pyFun n = pyStr ('F' : show n)
|
|
||||||
|
|
||||||
pySeq :: Int -> String
|
|
||||||
pySeq n = pyStr ('S' : show n)
|
|
||||||
|
|
||||||
pyStr :: String -> String
|
|
||||||
pyStr s = 'u' : prQuotedString s
|
|
||||||
|
|
||||||
pyCId :: CId -> String
|
|
||||||
pyCId = pyStr . showCId
|
|
||||||
|
|
||||||
pyIndent :: Int -> String
|
|
||||||
pyIndent n | n > 0 = "\n" ++ replicate n ' '
|
|
||||||
| otherwise = ""
|
|
||||||
@@ -27,19 +27,20 @@ module GF.Compile.Rename (
|
|||||||
renameModule
|
renameModule
|
||||||
) where
|
) where
|
||||||
|
|
||||||
|
import GF.Infra.Ident
|
||||||
|
import GF.Infra.CheckM
|
||||||
import GF.Grammar.Grammar
|
import GF.Grammar.Grammar
|
||||||
import GF.Grammar.Values
|
import GF.Grammar.Values
|
||||||
import GF.Grammar.Predef
|
import GF.Grammar.Predef
|
||||||
import GF.Infra.Ident
|
import GF.Grammar.Lookup
|
||||||
import GF.Infra.CheckM
|
|
||||||
import GF.Grammar.Macros
|
import GF.Grammar.Macros
|
||||||
import GF.Grammar.Printer
|
import GF.Grammar.Printer
|
||||||
--import GF.Grammar.Lookup
|
|
||||||
--import GF.Grammar.Printer
|
|
||||||
import GF.Data.Operations
|
import GF.Data.Operations
|
||||||
|
|
||||||
import Control.Monad
|
import Control.Monad
|
||||||
import Data.List (nub,(\\))
|
import Data.List (nub,(\\))
|
||||||
|
import qualified Data.Map as Map
|
||||||
|
import Data.Maybe(mapMaybe)
|
||||||
import GF.Text.Pretty
|
import GF.Text.Pretty
|
||||||
|
|
||||||
-- | this gives top-level access to renaming term input in the cc command
|
-- | this gives top-level access to renaming term input in the cc command
|
||||||
@@ -55,9 +56,9 @@ renameModule cwd gr mo@(m,mi) = do
|
|||||||
js <- checkMapRecover (renameInfo cwd status mo) (jments mi)
|
js <- checkMapRecover (renameInfo cwd status mo) (jments mi)
|
||||||
return (m, mi{jments = js})
|
return (m, mi{jments = js})
|
||||||
|
|
||||||
type Status = (StatusTree, [(OpenSpec, StatusTree)])
|
type Status = (StatusMap, [(OpenSpec, StatusMap)])
|
||||||
|
|
||||||
type StatusTree = BinTree Ident StatusInfo
|
type StatusMap = Map.Map Ident StatusInfo
|
||||||
|
|
||||||
type StatusInfo = Ident -> Term
|
type StatusInfo = Ident -> Term
|
||||||
|
|
||||||
@@ -73,12 +74,12 @@ renameIdentTerm' env@(act,imps) t0 =
|
|||||||
Q (m',c) | m' == cPredef {- && isInPredefined c -} -> return t0
|
Q (m',c) | m' == cPredef {- && isInPredefined c -} -> return t0
|
||||||
Q (m',c) -> do
|
Q (m',c) -> do
|
||||||
m <- lookupErr m' qualifs
|
m <- lookupErr m' qualifs
|
||||||
f <- lookupTree showIdent c m
|
f <- lookupIdent c m
|
||||||
return $ f c
|
return $ f c
|
||||||
QC (m',c) | m' == cPredef {- && isInPredefined c -} -> return t0
|
QC (m',c) | m' == cPredef {- && isInPredefined c -} -> return t0
|
||||||
QC (m',c) -> do
|
QC (m',c) -> do
|
||||||
m <- lookupErr m' qualifs
|
m <- lookupErr m' qualifs
|
||||||
f <- lookupTree showIdent c m
|
f <- lookupIdent c m
|
||||||
return $ f c
|
return $ f c
|
||||||
_ -> return t0
|
_ -> return t0
|
||||||
where
|
where
|
||||||
@@ -93,41 +94,32 @@ renameIdentTerm' env@(act,imps) t0 =
|
|||||||
| otherwise = checkError s
|
| otherwise = checkError s
|
||||||
|
|
||||||
ident alt c =
|
ident alt c =
|
||||||
case lookupTree showIdent c act of
|
case Map.lookup c act of
|
||||||
Ok f -> return (f c)
|
Just f -> return (f c)
|
||||||
_ -> case lookupTreeManyAll showIdent opens c of
|
_ -> case mapMaybe (Map.lookup c) opens of
|
||||||
[f] -> return (f c)
|
[f] -> return (f c)
|
||||||
[] -> alt c ("constant not found:" <+> c $$
|
[] -> alt c ("constant not found:" <+> c $$
|
||||||
"given" <+> fsep (punctuate ',' (map fst qualifs)))
|
"given" <+> fsep (punctuate ',' (map fst qualifs)))
|
||||||
fs -> case nub [f c | f <- fs] of
|
fs -> case nub [f c | f <- fs] of
|
||||||
[tr] -> return tr
|
[tr] -> return tr
|
||||||
{-
|
ts@(t:_) -> do checkWarn ("atomic term" <+> ppTerm Qualified 0 t0 $$
|
||||||
ts -> return $ AdHocOverload ts
|
"conflict" <+> hsep (punctuate ',' (map (ppTerm Qualified 0) ts)) $$
|
||||||
-- name conflicts resolved as overloading in TypeCheck.RConcrete AR 31/1/2014
|
"given" <+> fsep (punctuate ',' (map fst qualifs)))
|
||||||
-- the old definition is below and still presupposed in TypeCheck.Concrete
|
return t
|
||||||
-}
|
|
||||||
ts@(t:_) -> do checkWarn ("atomic term" <+> ppTerm Qualified 0 t0 $$
|
|
||||||
"conflict" <+> hsep (punctuate ',' (map (ppTerm Qualified 0) ts)) $$
|
|
||||||
"given" <+> fsep (punctuate ',' (map fst qualifs)))
|
|
||||||
return t
|
|
||||||
|
|
||||||
-- a warning will be generated in CheckGrammar, and the head returned
|
info2status :: Maybe ModuleName -> Ident -> Info -> StatusInfo
|
||||||
-- in next V:
|
info2status mq c i = case i of
|
||||||
-- Bad $ "conflicting imports:" +++ unwords (map prt ts)
|
|
||||||
|
|
||||||
info2status :: Maybe ModuleName -> (Ident,Info) -> StatusInfo
|
|
||||||
info2status mq (c,i) = case i of
|
|
||||||
AbsFun _ _ Nothing _ -> maybe Con (curry QC) mq
|
AbsFun _ _ Nothing _ -> maybe Con (curry QC) mq
|
||||||
ResValue _ -> maybe Con (curry QC) mq
|
ResValue _ _ -> maybe Con (curry QC) mq
|
||||||
ResParam _ _ -> maybe Con (curry QC) mq
|
ResParam _ _ -> maybe Con (curry QC) mq
|
||||||
AnyInd True m -> maybe Con (const (curry QC m)) mq
|
AnyInd True m -> maybe Con (const (curry QC m)) mq
|
||||||
AnyInd False m -> maybe Cn (const (curry Q m)) mq
|
AnyInd False m -> maybe Cn (const (curry Q m)) mq
|
||||||
_ -> maybe Cn (curry Q) mq
|
_ -> maybe Cn (curry Q) mq
|
||||||
|
|
||||||
tree2status :: OpenSpec -> BinTree Ident Info -> BinTree Ident StatusInfo
|
tree2status :: OpenSpec -> Map.Map Ident Info -> StatusMap
|
||||||
tree2status o = case o of
|
tree2status o = case o of
|
||||||
OSimple i -> mapTree (info2status (Just i))
|
OSimple i -> Map.mapWithKey (info2status (Just i))
|
||||||
OQualif i j -> mapTree (info2status (Just j))
|
OQualif i j -> Map.mapWithKey (info2status (Just j))
|
||||||
|
|
||||||
buildStatus :: FilePath -> Grammar -> Module -> Check Status
|
buildStatus :: FilePath -> Grammar -> Module -> Check Status
|
||||||
buildStatus cwd gr mo@(m,mi) = checkInModule cwd mi NoLoc empty $ do
|
buildStatus cwd gr mo@(m,mi) = checkInModule cwd mi NoLoc empty $ do
|
||||||
@@ -136,14 +128,14 @@ buildStatus cwd gr mo@(m,mi) = checkInModule cwd mi NoLoc empty $ do
|
|||||||
ops <- mapM (\o -> lookupModule gr1 (openedModule o) >>= \mi -> return (o,mi)) (mopens mi)
|
ops <- mapM (\o -> lookupModule gr1 (openedModule o) >>= \mi -> return (o,mi)) (mopens mi)
|
||||||
let sts = map modInfo2status (exts++ops)
|
let sts = map modInfo2status (exts++ops)
|
||||||
return (if isModCnc mi
|
return (if isModCnc mi
|
||||||
then (emptyBinTree, reverse sts) -- the module itself does not define any names
|
then (Map.empty, reverse sts) -- the module itself does not define any names
|
||||||
else (self2status m mi,reverse sts)) -- so the empty ident is not needed
|
else (self2status m mi,reverse sts)) -- so the empty ident is not needed
|
||||||
|
|
||||||
modInfo2status :: (OpenSpec,ModuleInfo) -> (OpenSpec, StatusTree)
|
modInfo2status :: (OpenSpec,ModuleInfo) -> (OpenSpec, StatusMap)
|
||||||
modInfo2status (o,mo) = (o,tree2status o (jments mo))
|
modInfo2status (o,mo) = (o,tree2status o (jments mo))
|
||||||
|
|
||||||
self2status :: ModuleName -> ModuleInfo -> StatusTree
|
self2status :: ModuleName -> ModuleInfo -> StatusMap
|
||||||
self2status c m = mapTree (info2status (Just c)) (jments m)
|
self2status c m = Map.mapWithKey (info2status (Just c)) (jments m)
|
||||||
|
|
||||||
|
|
||||||
renameInfo :: FilePath -> Status -> Module -> Ident -> Info -> Check Info
|
renameInfo :: FilePath -> Status -> Module -> Ident -> Info -> Check Info
|
||||||
@@ -156,9 +148,9 @@ renameInfo cwd status (m,mi) i info =
|
|||||||
ResParam (Just pp) m -> do
|
ResParam (Just pp) m -> do
|
||||||
pp' <- renLoc (mapM (renParam status)) pp
|
pp' <- renLoc (mapM (renParam status)) pp
|
||||||
return (ResParam (Just pp') m)
|
return (ResParam (Just pp') m)
|
||||||
ResValue t -> do
|
ResValue ty offset -> do
|
||||||
t <- renLoc (renameTerm status []) t
|
t <- renLoc (renameTerm status []) ty
|
||||||
return (ResValue t)
|
return (ResValue ty offset)
|
||||||
CncCat mcat mdef mref mpr mpmcfg -> liftM5 CncCat (renTerm mcat) (renTerm mdef) (renTerm mref) (renTerm mpr) (return mpmcfg)
|
CncCat mcat mdef mref mpr mpmcfg -> liftM5 CncCat (renTerm mcat) (renTerm mdef) (renTerm mref) (renTerm mpr) (return mpmcfg)
|
||||||
CncFun mty mtr mpr mpmcfg -> liftM3 (CncFun mty) (renTerm mtr) (renTerm mpr) (return mpmcfg)
|
CncFun mty mtr mpr mpmcfg -> liftM3 (CncFun mty) (renTerm mtr) (renTerm mpr) (return mpmcfg)
|
||||||
_ -> return info
|
_ -> return info
|
||||||
@@ -186,9 +178,9 @@ renameInfo cwd status (m,mi) i info =
|
|||||||
return (ps',t')
|
return (ps',t')
|
||||||
|
|
||||||
renParam :: Status -> Param -> Check Param
|
renParam :: Status -> Param -> Check Param
|
||||||
renParam env (c,co) = do
|
renParam env (c,co,i) = do
|
||||||
co' <- renameContext env co
|
co' <- renameContext env co
|
||||||
return (c,co')
|
return (c,co',i)
|
||||||
|
|
||||||
renameTerm :: Status -> [Ident] -> Term -> Check Term
|
renameTerm :: Status -> [Ident] -> Term -> Check Term
|
||||||
renameTerm env vars = ren vars where
|
renameTerm env vars = ren vars where
|
||||||
|
|||||||
@@ -31,7 +31,7 @@ getLocalTags x (m,mi) =
|
|||||||
getLocations (AbsFun mb_type _ mb_eqs _) = maybe (ltype "fun") mb_type ++
|
getLocations (AbsFun mb_type _ mb_eqs _) = maybe (ltype "fun") mb_type ++
|
||||||
maybe (list (loc "def")) mb_eqs
|
maybe (list (loc "def")) mb_eqs
|
||||||
getLocations (ResParam mb_params _) = maybe (loc "param") mb_params
|
getLocations (ResParam mb_params _) = maybe (loc "param") mb_params
|
||||||
getLocations (ResValue mb_type) = ltype "param-value" mb_type
|
getLocations (ResValue mb_type _) = ltype "param-value" mb_type
|
||||||
getLocations (ResOper mb_type mb_def) = maybe (ltype "oper-type") mb_type ++
|
getLocations (ResOper mb_type mb_def) = maybe (ltype "oper-type") mb_type ++
|
||||||
maybe (loc "oper-def") mb_def
|
maybe (loc "oper-def") mb_def
|
||||||
getLocations (ResOverload _ defs) = list (\(x,y) -> ltype "overload-type" x ++
|
getLocations (ResOverload _ defs) = list (\(x,y) -> ltype "overload-type" x ++
|
||||||
|
|||||||
@@ -2,8 +2,7 @@ module GF.Compile.ToAPI
|
|||||||
(stringToAPI,exprToAPI)
|
(stringToAPI,exprToAPI)
|
||||||
where
|
where
|
||||||
|
|
||||||
import PGF.Internal
|
import PGF2
|
||||||
import PGF(showCId)
|
|
||||||
import Data.Maybe
|
import Data.Maybe
|
||||||
--import System.IO
|
--import System.IO
|
||||||
--import Control.Monad
|
--import Control.Monad
|
||||||
@@ -47,12 +46,12 @@ exprToFunc :: Expr -> APIfunc
|
|||||||
exprToFunc expr =
|
exprToFunc expr =
|
||||||
case unApp expr of
|
case unApp expr of
|
||||||
Just (cid,l) ->
|
Just (cid,l) ->
|
||||||
case Map.lookup (showCId cid) syntaxFuncs of
|
case Map.lookup cid syntaxFuncs of
|
||||||
Just sig -> mkAPI True (fst sig,expr)
|
Just sig -> mkAPI True (fst sig,expr)
|
||||||
_ -> case l of
|
_ -> case l of
|
||||||
[] -> BasicFunc (showCId cid)
|
[] -> BasicFunc cid
|
||||||
_ -> let es = map exprToFunc l
|
_ -> let es = map exprToFunc l
|
||||||
in AppFunc (showCId cid) es
|
in AppFunc cid es
|
||||||
_ -> BasicFunc (showExpr [] expr)
|
_ -> BasicFunc (showExpr [] expr)
|
||||||
|
|
||||||
|
|
||||||
@@ -69,8 +68,8 @@ mkAPI opt (ty,expr) =
|
|||||||
where
|
where
|
||||||
rephraseSentence ty expr =
|
rephraseSentence ty expr =
|
||||||
case unApp expr of
|
case unApp expr of
|
||||||
Just (cid,es) -> if isPrefixOf "Use" (showCId cid) then
|
Just (cid,es) -> if isPrefixOf "Use" cid then
|
||||||
let newCat = drop 3 (showCId cid)
|
let newCat = drop 3 cid
|
||||||
afClause = mkAPI True (newCat, es !! 2)
|
afClause = mkAPI True (newCat, es !! 2)
|
||||||
afPol = mkAPI True ("Pol",es !! 1)
|
afPol = mkAPI True ("Pol",es !! 1)
|
||||||
lTense = mkAPI True ("Temp", head es)
|
lTense = mkAPI True ("Temp", head es)
|
||||||
@@ -98,9 +97,9 @@ mkAPI opt (ty,expr) =
|
|||||||
computeAPI :: (String,Expr) -> APIfunc
|
computeAPI :: (String,Expr) -> APIfunc
|
||||||
computeAPI (ty,expr) =
|
computeAPI (ty,expr) =
|
||||||
case (unApp expr) of
|
case (unApp expr) of
|
||||||
Just (cid,[]) -> getSimpCat (showCId cid) ty
|
Just (cid,[]) -> getSimpCat cid ty
|
||||||
Just (cid,es) ->
|
Just (cid,es) ->
|
||||||
let p = specFunction (showCId cid) es
|
let p = specFunction cid es
|
||||||
in if isJust p then fromJust p
|
in if isJust p then fromJust p
|
||||||
else case Map.lookup (show cid) syntaxFuncs of
|
else case Map.lookup (show cid) syntaxFuncs of
|
||||||
Nothing -> exprToFunc expr
|
Nothing -> exprToFunc expr
|
||||||
@@ -147,23 +146,23 @@ optimize expr = optimizeNP expr
|
|||||||
optimizeNP expr =
|
optimizeNP expr =
|
||||||
case unApp expr of
|
case unApp expr of
|
||||||
Just (cid,es) ->
|
Just (cid,es) ->
|
||||||
if showCId cid == "MassNP" then let afs = nounAsCN (head es)
|
if cid == "MassNP" then let afs = nounAsCN (head es)
|
||||||
in AppFunc "mkNP" [afs]
|
in AppFunc "mkNP" [afs]
|
||||||
else if showCId cid == "DetCN" then let quants = quantAsDet (head es)
|
else if cid == "DetCN" then let quants = quantAsDet (head es)
|
||||||
ns = nounAsCN (head $ tail es)
|
ns = nounAsCN (head $ tail es)
|
||||||
in AppFunc "mkNP" (quants ++ [ns])
|
in AppFunc "mkNP" (quants ++ [ns])
|
||||||
else mkAPI False ("NP",expr)
|
else mkAPI False ("NP",expr)
|
||||||
_ -> error $ "incorrect expression " ++ (showExpr [] expr)
|
_ -> error $ "incorrect expression " ++ (showExpr [] expr)
|
||||||
where
|
where
|
||||||
nounAsCN expr =
|
nounAsCN expr =
|
||||||
case unApp expr of
|
case unApp expr of
|
||||||
Just (cid,es) -> if showCId cid == "UseN" then (mkAPI False) ("N",head es)
|
Just (cid,es) -> if cid == "UseN" then (mkAPI False) ("N",head es)
|
||||||
else (mkAPI False) ("CN",expr)
|
else (mkAPI False) ("CN",expr)
|
||||||
_ -> error $ "incorrect expression "++ (showExpr [] expr)
|
_ -> error $ "incorrect expression "++ (showExpr [] expr)
|
||||||
|
|
||||||
quantAsDet expr =
|
quantAsDet expr =
|
||||||
case unApp expr of
|
case unApp expr of
|
||||||
Just (cid,es) -> if showCId cid == "DetQuant" then map (mkAPI False) [("Quant", head es),("Num",head $ tail es)]
|
Just (cid,es) -> if cid == "DetQuant" then map (mkAPI False) [("Quant", head es),("Num",head $ tail es)]
|
||||||
else [mkAPI False ("Det",expr)]
|
else [mkAPI False ("Det",expr)]
|
||||||
|
|
||||||
_ -> error $ "incorrect expression "++ (showExpr [] expr)
|
_ -> error $ "incorrect expression "++ (showExpr [] expr)
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ typPredefined :: Ident -> Maybe Type
|
|||||||
typPredefined f = case Map.lookup f primitives of
|
typPredefined f = case Map.lookup f primitives of
|
||||||
Just (ResOper (Just (L _ ty)) _) -> Just ty
|
Just (ResOper (Just (L _ ty)) _) -> Just ty
|
||||||
Just (ResParam _ _) -> Just typePType
|
Just (ResParam _ _) -> Just typePType
|
||||||
Just (ResValue (L _ ty)) -> Just ty
|
Just (ResValue (L _ ty) _) -> Just ty
|
||||||
_ -> Nothing
|
_ -> Nothing
|
||||||
|
|
||||||
primitives = Map.fromList
|
primitives = Map.fromList
|
||||||
@@ -16,9 +16,9 @@ primitives = Map.fromList
|
|||||||
, (cInt , ResOper (Just (noLoc typePType)) Nothing)
|
, (cInt , ResOper (Just (noLoc typePType)) Nothing)
|
||||||
, (cFloat , ResOper (Just (noLoc typePType)) Nothing)
|
, (cFloat , ResOper (Just (noLoc typePType)) Nothing)
|
||||||
, (cInts , fun [typeInt] typePType)
|
, (cInts , fun [typeInt] typePType)
|
||||||
, (cPBool , ResParam (Just (noLoc [(cPTrue,[]),(cPFalse,[])])) (Just [QC (cPredef,cPTrue), QC (cPredef,cPFalse)]))
|
, (cPBool , ResParam (Just (noLoc [(cPTrue,[],0),(cPFalse,[],1)])) (Just [QC (cPredef,cPTrue), QC (cPredef,cPFalse)]))
|
||||||
, (cPTrue , ResValue (noLoc typePBool))
|
, (cPTrue , ResValue (noLoc typePBool) 0)
|
||||||
, (cPFalse , ResValue (noLoc typePBool))
|
, (cPFalse , ResValue (noLoc typePBool) 1)
|
||||||
, (cError , fun [typeStr] typeError) -- non-can. of empty set
|
, (cError , fun [typeStr] typeError) -- non-can. of empty set
|
||||||
, (cLength , fun [typeTok] typeInt)
|
, (cLength , fun [typeTok] typeInt)
|
||||||
, (cDrop , fun [typeInt,typeTok] typeTok)
|
, (cDrop , fun [typeInt,typeTok] typeTok)
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
{-# LANGUAGE PatternGuards #-}
|
{-# LANGUAGE PatternGuards #-}
|
||||||
module GF.Compile.TypeCheck.RConcrete( checkLType, inferLType, computeLType, ppType ) where
|
module GF.Compile.TypeCheck.RConcrete( checkLType, inferLType, computeLType, ppType ) where
|
||||||
import Prelude hiding ((<>)) -- GHC 8.4.1 clash with Text.PrettyPrint
|
|
||||||
|
|
||||||
import GF.Infra.CheckM
|
import GF.Infra.CheckM
|
||||||
import GF.Data.Operations
|
import GF.Data.Operations
|
||||||
|
|||||||
@@ -29,7 +29,7 @@ import Control.Monad
|
|||||||
import GF.Text.Pretty
|
import GF.Text.Pretty
|
||||||
|
|
||||||
-- | combine a list of definitions into a balanced binary search tree
|
-- | combine a list of definitions into a balanced binary search tree
|
||||||
buildAnyTree :: Monad m => ModuleName -> [(Ident,Info)] -> m (BinTree Ident Info)
|
buildAnyTree :: Monad m => ModuleName -> [(Ident,Info)] -> m (Map.Map Ident Info)
|
||||||
buildAnyTree m = go Map.empty
|
buildAnyTree m = go Map.empty
|
||||||
where
|
where
|
||||||
go map [] = return map
|
go map [] = return map
|
||||||
@@ -101,8 +101,8 @@ rebuildModule cwd gr mo@(i,mi@(ModInfo mt stat fs_ me mw ops_ med_ msrc_ env_ js
|
|||||||
[] -> return mi{jments=js'}
|
[] -> return mi{jments=js'}
|
||||||
j0s -> do
|
j0s -> do
|
||||||
m0s <- mapM (lookupModule gr) j0s
|
m0s <- mapM (lookupModule gr) j0s
|
||||||
let notInM0 c _ = all (not . isInBinTree c . jments) m0s
|
let notInM0 c _ = all (not . Map.member c . jments) m0s
|
||||||
let js2 = filterBinTree notInM0 js'
|
let js2 = Map.filterWithKey notInM0 js'
|
||||||
return mi{jments=js2}
|
return mi{jments=js2}
|
||||||
_ -> return mi
|
_ -> return mi
|
||||||
|
|
||||||
@@ -123,8 +123,11 @@ rebuildModule cwd gr mo@(i,mi@(ModInfo mt stat fs_ me mw ops_ med_ msrc_ env_ js
|
|||||||
|
|
||||||
--- check if me is incomplete
|
--- check if me is incomplete
|
||||||
let fs1 = fs `addOptions` fs_ -- new flags have priority
|
let fs1 = fs `addOptions` fs_ -- new flags have priority
|
||||||
let js0 = [(c,globalizeLoc fpath j) | (c,j) <- tree2list js, isInherited incl c]
|
let js0 = Map.mapMaybeWithKey (\c j -> if isInherited incl c
|
||||||
let js1 = buildTree (tree2list js_ ++ js0)
|
then Just (globalizeLoc fpath j)
|
||||||
|
else Nothing)
|
||||||
|
js
|
||||||
|
let js1 = Map.union js0 js_
|
||||||
let med1= nub (ext : infs ++ insts ++ med_)
|
let med1= nub (ext : infs ++ insts ++ med_)
|
||||||
return $ ModInfo mt0 stat' fs1 me Nothing ops1 med1 msrc_ env_ js1
|
return $ ModInfo mt0 stat' fs1 me Nothing ops1 med1 msrc_ env_ js1
|
||||||
|
|
||||||
@@ -135,14 +138,14 @@ rebuildModule cwd gr mo@(i,mi@(ModInfo mt stat fs_ me mw ops_ med_ msrc_ env_ js
|
|||||||
-- If the extended module is incomplete, its judgements are just copied.
|
-- If the extended module is incomplete, its judgements are just copied.
|
||||||
extendMod :: Grammar ->
|
extendMod :: Grammar ->
|
||||||
Bool -> (Module,Ident -> Bool) -> ModuleName ->
|
Bool -> (Module,Ident -> Bool) -> ModuleName ->
|
||||||
BinTree Ident Info -> Check (BinTree Ident Info)
|
Map.Map Ident Info -> Check (Map.Map Ident Info)
|
||||||
extendMod gr isCompl ((name,mi),cond) base new = foldM try new $ Map.toList (jments mi)
|
extendMod gr isCompl ((name,mi),cond) base new = foldM try new $ Map.toList (jments mi)
|
||||||
where
|
where
|
||||||
try new (c,i0)
|
try new (c,i0)
|
||||||
| not (cond c) = return new
|
| not (cond c) = return new
|
||||||
| otherwise = case Map.lookup c new of
|
| otherwise = case Map.lookup c new of
|
||||||
Just j -> case unifyAnyInfo name i j of
|
Just j -> case unifyAnyInfo name i j of
|
||||||
Ok k -> return $ updateTree (c,k) new
|
Ok k -> return $ Map.insert c k new
|
||||||
Bad _ -> do (base,j) <- case j of
|
Bad _ -> do (base,j) <- case j of
|
||||||
AnyInd _ m -> lookupOrigInfo gr (m,c)
|
AnyInd _ m -> lookupOrigInfo gr (m,c)
|
||||||
_ -> return (base,j)
|
_ -> return (base,j)
|
||||||
@@ -155,15 +158,15 @@ extendMod gr isCompl ((name,mi),cond) base new = foldM try new $ Map.toList (jme
|
|||||||
nest 4 (ppJudgement Qualified (c,j)) $$
|
nest 4 (ppJudgement Qualified (c,j)) $$
|
||||||
"in module" <+> base)
|
"in module" <+> base)
|
||||||
Nothing-> if isCompl
|
Nothing-> if isCompl
|
||||||
then return $ updateTree (c,indirInfo name i) new
|
then return $ Map.insert c (indirInfo name i) new
|
||||||
else return $ updateTree (c,i) new
|
else return $ Map.insert c i new
|
||||||
where
|
where
|
||||||
i = globalizeLoc (msrc mi) i0
|
i = globalizeLoc (msrc mi) i0
|
||||||
|
|
||||||
indirInfo :: ModuleName -> Info -> Info
|
indirInfo :: ModuleName -> Info -> Info
|
||||||
indirInfo n info = AnyInd b n' where
|
indirInfo n info = AnyInd b n' where
|
||||||
(b,n') = case info of
|
(b,n') = case info of
|
||||||
ResValue _ -> (True,n)
|
ResValue _ _ -> (True,n)
|
||||||
ResParam _ _ -> (True,n)
|
ResParam _ _ -> (True,n)
|
||||||
AbsFun _ _ Nothing _ -> (True,n)
|
AbsFun _ _ Nothing _ -> (True,n)
|
||||||
AnyInd b k -> (b,k)
|
AnyInd b k -> (b,k)
|
||||||
@@ -174,7 +177,7 @@ globalizeLoc fpath i =
|
|||||||
AbsCat mc -> AbsCat (fmap gl mc)
|
AbsCat mc -> AbsCat (fmap gl mc)
|
||||||
AbsFun mt ma md moper -> AbsFun (fmap gl mt) ma (fmap (fmap gl) md) moper
|
AbsFun mt ma md moper -> AbsFun (fmap gl mt) ma (fmap (fmap gl) md) moper
|
||||||
ResParam mt mv -> ResParam (fmap gl mt) mv
|
ResParam mt mv -> ResParam (fmap gl mt) mv
|
||||||
ResValue t -> ResValue (gl t)
|
ResValue t offset -> ResValue (gl t) offset
|
||||||
ResOper mt m -> ResOper (fmap gl mt) (fmap gl m)
|
ResOper mt m -> ResOper (fmap gl mt) (fmap gl m)
|
||||||
ResOverload ms os -> ResOverload ms (map (\(x,y) -> (gl x,gl y)) os)
|
ResOverload ms os -> ResOverload ms (map (\(x,y) -> (gl x,gl y)) os)
|
||||||
CncCat mc md mr mp mpmcfg-> CncCat (fmap gl mc) (fmap gl md) (fmap gl mr) (fmap gl mp) mpmcfg
|
CncCat mc md mr mp mpmcfg-> CncCat (fmap gl mc) (fmap gl md) (fmap gl mr) (fmap gl mp) mpmcfg
|
||||||
@@ -196,9 +199,9 @@ unifyAnyInfo m i j = case (i,j) of
|
|||||||
|
|
||||||
(ResParam mt1 mv1, ResParam mt2 mv2) ->
|
(ResParam mt1 mv1, ResParam mt2 mv2) ->
|
||||||
liftM2 ResParam (unifyMaybeL mt1 mt2) (unifyMaybe mv1 mv2)
|
liftM2 ResParam (unifyMaybeL mt1 mt2) (unifyMaybe mv1 mv2)
|
||||||
(ResValue (L l1 t1), ResValue (L l2 t2))
|
(ResValue (L l1 t1) i1, ResValue (L l2 t2) i2)
|
||||||
| t1==t2 -> return (ResValue (L l1 t1))
|
| t1==t2 && i1 == i2 -> return (ResValue (L l1 t1) i1)
|
||||||
| otherwise -> fail ""
|
| otherwise -> fail ""
|
||||||
(_, ResOverload ms t) | elem m ms ->
|
(_, ResOverload ms t) | elem m ms ->
|
||||||
return $ ResOverload ms t
|
return $ ResOverload ms t
|
||||||
(ResOper mt1 m1, ResOper mt2 m2) ->
|
(ResOper mt1 m1, ResOper mt2 m2) ->
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
-- | Parallel grammar compilation
|
-- | Parallel grammar compilation
|
||||||
module GF.CompileInParallel(parallelBatchCompile) where
|
module GF.CompileInParallel(parallelBatchCompile) where
|
||||||
import Prelude hiding (catch,(<>)) -- GHC 8.4.1 clash with Text.PrettyPrint
|
import Prelude hiding (catch)
|
||||||
import Control.Monad(join,ap,when,unless)
|
import Control.Monad(join,ap,when,unless)
|
||||||
import Control.Applicative
|
import Control.Applicative
|
||||||
import GF.Infra.Concurrency
|
import GF.Infra.Concurrency
|
||||||
@@ -34,11 +34,8 @@ import qualified Data.ByteString.Lazy as BS
|
|||||||
parallelBatchCompile jobs opts rootfiles0 =
|
parallelBatchCompile jobs opts rootfiles0 =
|
||||||
do setJobs jobs
|
do setJobs jobs
|
||||||
rootfiles <- mapM canonical rootfiles0
|
rootfiles <- mapM canonical rootfiles0
|
||||||
lib_dirs1 <- getLibraryDirectory opts
|
lib_dir <- canonical =<< getLibraryDirectory opts
|
||||||
lib_dirs2 <- mapM canonical lib_dirs1
|
filepaths <- mapM (getPathFromFile lib_dir opts) rootfiles
|
||||||
let lib_dir = head lib_dirs2
|
|
||||||
when (length lib_dirs2 >1) $ ePutStrLn ("GF_LIB_PATH defines more than one directory; using the first, " ++ show lib_dir)
|
|
||||||
filepaths <- mapM (getPathFromFile [lib_dir] opts) rootfiles
|
|
||||||
let groups = groupFiles lib_dir filepaths
|
let groups = groupFiles lib_dir filepaths
|
||||||
n = length groups
|
n = length groups
|
||||||
when (n>1) $ ePutStrLn "Grammar mixes present and alltenses, dividing modules into two groups"
|
when (n>1) $ ePutStrLn "Grammar mixes present and alltenses, dividing modules into two groups"
|
||||||
|
|||||||
@@ -1,8 +1,7 @@
|
|||||||
module GF.Compiler (mainGFC, linkGrammars, writePGF, writeOutputs) where
|
module GF.Compiler (mainGFC, linkGrammars, writeGrammar, writeOutputs) where
|
||||||
|
|
||||||
import PGF
|
import PGF2
|
||||||
import PGF.Internal(concretes,optimizePGF,unionPGF)
|
import PGF2.Internal(unionPGF,writePGF,writeConcr)
|
||||||
import PGF.Internal(putSplitAbs,encodeFile,runPut)
|
|
||||||
import GF.Compile as S(batchCompile,link,srcAbsName)
|
import GF.Compile as S(batchCompile,link,srcAbsName)
|
||||||
import GF.CompileInParallel as P(parallelBatchCompile)
|
import GF.CompileInParallel as P(parallelBatchCompile)
|
||||||
import GF.Compile.Export
|
import GF.Compile.Export
|
||||||
@@ -92,7 +91,7 @@ compileSourceFiles opts fs =
|
|||||||
-- in the 'Options') from the output of 'parallelBatchCompile'.
|
-- in the 'Options') from the output of 'parallelBatchCompile'.
|
||||||
-- If a @.pgf@ file by the same name already exists and it is newer than the
|
-- If a @.pgf@ file by the same name already exists and it is newer than the
|
||||||
-- source grammar files (as indicated by the 'UTCTime' argument), it is not
|
-- source grammar files (as indicated by the 'UTCTime' argument), it is not
|
||||||
-- recreated. Calls 'writePGF' and 'writeOutputs'.
|
-- recreated. Calls 'writeGrammar' and 'writeOutputs'.
|
||||||
linkGrammars opts (t_src,~cnc_grs@(~(cnc,gr):_)) =
|
linkGrammars opts (t_src,~cnc_grs@(~(cnc,gr):_)) =
|
||||||
do let abs = render (srcAbsName gr cnc)
|
do let abs = render (srcAbsName gr cnc)
|
||||||
pgfFile = outputPath opts (grammarName' opts abs<.>"pgf")
|
pgfFile = outputPath opts (grammarName' opts abs<.>"pgf")
|
||||||
@@ -102,10 +101,8 @@ linkGrammars opts (t_src,~cnc_grs@(~(cnc,gr):_)) =
|
|||||||
if t_pgf >= Just t_src
|
if t_pgf >= Just t_src
|
||||||
then putIfVerb opts $ pgfFile ++ " is up-to-date."
|
then putIfVerb opts $ pgfFile ++ " is up-to-date."
|
||||||
else do pgfs <- mapM (link opts) cnc_grs
|
else do pgfs <- mapM (link opts) cnc_grs
|
||||||
let pgf0 = foldl1 unionPGF pgfs
|
let pgf = foldl1 (\one two -> fromMaybe two (unionPGF one two)) pgfs
|
||||||
probs <- maybe (return . defaultProbabilities) readProbabilitiesFromFile (flag optProbsFile opts) pgf0
|
writeGrammar opts pgf
|
||||||
let pgf = setProbabilities probs pgf0
|
|
||||||
writePGF opts pgf
|
|
||||||
writeOutputs opts pgf
|
writeOutputs opts pgf
|
||||||
|
|
||||||
compileCFFiles :: Options -> [FilePath] -> IOE ()
|
compileCFFiles :: Options -> [FilePath] -> IOE ()
|
||||||
@@ -115,12 +112,11 @@ compileCFFiles opts fs = do
|
|||||||
startCat <- case rules of
|
startCat <- case rules of
|
||||||
(Rule cat _ _ : _) -> return cat
|
(Rule cat _ _ : _) -> return cat
|
||||||
_ -> fail "empty CFG"
|
_ -> fail "empty CFG"
|
||||||
let pgf = cf2pgf (last fs) (mkCFG startCat Set.empty rules)
|
probs <- liftIO (maybe (return Map.empty) readProbabilitiesFromFile (flag optProbsFile opts))
|
||||||
|
let pgf = cf2pgf opts (last fs) (mkCFG startCat Set.empty rules) probs
|
||||||
unless (flag optStopAfterPhase opts == Compile) $
|
unless (flag optStopAfterPhase opts == Compile) $
|
||||||
do probs <- liftIO (maybe (return . defaultProbabilities) readProbabilitiesFromFile (flag optProbsFile opts) pgf)
|
do writeGrammar opts pgf
|
||||||
let pgf' = setProbabilities probs $ if flag optOptimizePGF opts then optimizePGF pgf else pgf
|
writeOutputs opts pgf
|
||||||
writePGF opts pgf'
|
|
||||||
writeOutputs opts pgf'
|
|
||||||
|
|
||||||
unionPGFFiles :: Options -> [FilePath] -> IOE ()
|
unionPGFFiles :: Options -> [FilePath] -> IOE ()
|
||||||
unionPGFFiles opts fs =
|
unionPGFFiles opts fs =
|
||||||
@@ -138,14 +134,11 @@ unionPGFFiles opts fs =
|
|||||||
|
|
||||||
doIt =
|
doIt =
|
||||||
do pgfs <- mapM readPGFVerbose fs
|
do pgfs <- mapM readPGFVerbose fs
|
||||||
let pgf0 = foldl1 unionPGF pgfs
|
let pgf = foldl1 (\one two -> fromMaybe two (unionPGF one two)) pgfs
|
||||||
pgf1 = if flag optOptimizePGF opts then optimizePGF pgf0 else pgf0
|
let pgfFile = outputPath opts (grammarName opts pgf <.> "pgf")
|
||||||
probs <- liftIO (maybe (return . defaultProbabilities) readProbabilitiesFromFile (flag optProbsFile opts) pgf1)
|
|
||||||
let pgf = setProbabilities probs pgf1
|
|
||||||
pgfFile = outputPath opts (grammarName opts pgf <.> "pgf")
|
|
||||||
if pgfFile `elem` fs
|
if pgfFile `elem` fs
|
||||||
then putStrLnE $ "Refusing to overwrite " ++ pgfFile
|
then putStrLnE $ "Refusing to overwrite " ++ pgfFile
|
||||||
else writePGF opts pgf
|
else writeGrammar opts pgf
|
||||||
writeOutputs opts pgf
|
writeOutputs opts pgf
|
||||||
|
|
||||||
readPGFVerbose f =
|
readPGFVerbose f =
|
||||||
@@ -162,21 +155,20 @@ writeOutputs opts pgf = do
|
|||||||
-- | Write the result of compiling a grammar (e.g. with 'compileToPGF' or
|
-- | Write the result of compiling a grammar (e.g. with 'compileToPGF' or
|
||||||
-- 'link') to a @.pgf@ file.
|
-- 'link') to a @.pgf@ file.
|
||||||
-- A split PGF file is output if the @-split-pgf@ option is used.
|
-- A split PGF file is output if the @-split-pgf@ option is used.
|
||||||
writePGF :: Options -> PGF -> IOE ()
|
writeGrammar :: Options -> PGF -> IOE ()
|
||||||
writePGF opts pgf =
|
writeGrammar opts pgf =
|
||||||
if flag optSplitPGF opts then writeSplitPGF else writeNormalPGF
|
if flag optSplitPGF opts then writeSplitPGF else writeNormalPGF
|
||||||
where
|
where
|
||||||
writeNormalPGF =
|
writeNormalPGF =
|
||||||
do let outfile = outputPath opts (grammarName opts pgf <.> "pgf")
|
do let outfile = outputPath opts (grammarName opts pgf <.> "pgf")
|
||||||
writing opts outfile $ encodeFile outfile pgf
|
writing opts outfile (writePGF outfile pgf)
|
||||||
|
|
||||||
writeSplitPGF =
|
writeSplitPGF =
|
||||||
do let outfile = outputPath opts (grammarName opts pgf <.> "pgf")
|
do let outfile = outputPath opts (grammarName opts pgf <.> "pgf")
|
||||||
writing opts outfile $ BSL.writeFile outfile (runPut (putSplitAbs pgf))
|
writing opts outfile $ writePGF outfile pgf
|
||||||
--encodeFile_ outfile (putSplitAbs pgf)
|
forM_ (Map.toList (languages pgf)) $ \(concrname,concr) -> do
|
||||||
forM_ (Map.toList (concretes pgf)) $ \cnc -> do
|
let outfile = outputPath opts (concrname <.> "pgf_c")
|
||||||
let outfile = outputPath opts (showCId (fst cnc) <.> "pgf_c")
|
writing opts outfile (writeConcr outfile concr)
|
||||||
writing opts outfile $ encodeFile outfile cnc
|
|
||||||
|
|
||||||
|
|
||||||
writeOutput :: Options -> FilePath-> String -> IOE ()
|
writeOutput :: Options -> FilePath-> String -> IOE ()
|
||||||
@@ -186,7 +178,7 @@ writeOutput opts file str = writing opts path $ writeUTF8File path str
|
|||||||
-- * Useful helper functions
|
-- * Useful helper functions
|
||||||
|
|
||||||
grammarName :: Options -> PGF -> String
|
grammarName :: Options -> PGF -> String
|
||||||
grammarName opts pgf = grammarName' opts (showCId (abstractName pgf))
|
grammarName opts pgf = grammarName' opts (abstractName pgf)
|
||||||
grammarName' opts abs = fromMaybe abs (flag optName opts)
|
grammarName' opts abs = fromMaybe abs (flag optName opts)
|
||||||
|
|
||||||
outputJustPGF opts = null (flag optOutputFormats opts) && not (flag optSplitPGF opts)
|
outputJustPGF opts = null (flag optOutputFormats opts) && not (flag optSplitPGF opts)
|
||||||
|
|||||||
@@ -16,8 +16,6 @@ import GF.Compile.ReadFiles
|
|||||||
import GF.Compile.Update
|
import GF.Compile.Update
|
||||||
import GF.Compile.Refresh
|
import GF.Compile.Refresh
|
||||||
|
|
||||||
import GF.Compile.Coding
|
|
||||||
|
|
||||||
import GF.Grammar.Grammar
|
import GF.Grammar.Grammar
|
||||||
import GF.Grammar.Lookup
|
import GF.Grammar.Lookup
|
||||||
import GF.Grammar.Printer
|
import GF.Grammar.Printer
|
||||||
|
|||||||
@@ -28,14 +28,6 @@ module GF.Data.Operations (
|
|||||||
|
|
||||||
-- ** Monadic operations on lists and pairs
|
-- ** Monadic operations on lists and pairs
|
||||||
mapPairListM, mapPairsM, pairM,
|
mapPairListM, mapPairsM, pairM,
|
||||||
|
|
||||||
-- ** Binary search trees; now with FiniteMap
|
|
||||||
BinTree, emptyBinTree, isInBinTree, --justLookupTree,
|
|
||||||
lookupTree, --lookupTreeMany,
|
|
||||||
lookupTreeManyAll, updateTree,
|
|
||||||
buildTree, filterBinTree,
|
|
||||||
mapTree, --mapMTree,
|
|
||||||
tree2list,
|
|
||||||
|
|
||||||
-- ** Printing
|
-- ** Printing
|
||||||
indent, (+++), (++-), (++++), (+++-), (+++++),
|
indent, (+++), (++-), (++++), (+++-), (+++++),
|
||||||
@@ -50,10 +42,6 @@ module GF.Data.Operations (
|
|||||||
ifNull,
|
ifNull,
|
||||||
combinations, done, readIntArg, --singleton,
|
combinations, done, readIntArg, --singleton,
|
||||||
iterFix, chunks,
|
iterFix, chunks,
|
||||||
{-
|
|
||||||
-- ** State monad with error; from Agda 6\/11\/2001
|
|
||||||
STM(..), appSTM, stm, stmr, readSTM, updateSTM, writeSTM,
|
|
||||||
-}
|
|
||||||
|
|
||||||
) where
|
) where
|
||||||
|
|
||||||
@@ -117,44 +105,6 @@ unifyMaybeBy f (Just p1) (Just p2)
|
|||||||
unifyMaybeBy _ Nothing mp2 = return mp2
|
unifyMaybeBy _ Nothing mp2 = return mp2
|
||||||
unifyMaybeBy _ mp1 _ = return mp1
|
unifyMaybeBy _ mp1 _ = return mp1
|
||||||
|
|
||||||
-- binary search trees
|
|
||||||
|
|
||||||
type BinTree a b = Map a b
|
|
||||||
|
|
||||||
emptyBinTree :: BinTree a b
|
|
||||||
emptyBinTree = Map.empty
|
|
||||||
|
|
||||||
isInBinTree :: (Ord a) => a -> BinTree a b -> Bool
|
|
||||||
isInBinTree = Map.member
|
|
||||||
{-
|
|
||||||
justLookupTree :: (ErrorMonad m,Ord a) => a -> BinTree a b -> m b
|
|
||||||
justLookupTree = lookupTree (const [])
|
|
||||||
-}
|
|
||||||
lookupTree :: (ErrorMonad m,Ord a) => (a -> String) -> a -> BinTree a b -> m b
|
|
||||||
lookupTree pr x = maybeErr no . Map.lookup x
|
|
||||||
where no = "no occurrence of element" +++ pr x
|
|
||||||
|
|
||||||
lookupTreeManyAll :: Ord a => (a -> String) -> [BinTree a b] -> a -> [b]
|
|
||||||
lookupTreeManyAll pr (t:ts) x = case lookupTree pr x t of
|
|
||||||
Ok v -> v : lookupTreeManyAll pr ts x
|
|
||||||
_ -> lookupTreeManyAll pr ts x
|
|
||||||
lookupTreeManyAll pr [] x = []
|
|
||||||
|
|
||||||
updateTree :: (Ord a) => (a,b) -> BinTree a b -> BinTree a b
|
|
||||||
updateTree (a,b) = Map.insert a b
|
|
||||||
|
|
||||||
buildTree :: (Ord a) => [(a,b)] -> BinTree a b
|
|
||||||
buildTree = Map.fromList
|
|
||||||
|
|
||||||
mapTree :: ((a,b) -> c) -> BinTree a b -> BinTree a c
|
|
||||||
mapTree f = Map.mapWithKey (\k v -> f (k,v))
|
|
||||||
|
|
||||||
filterBinTree :: Ord a => (a -> b -> Bool) -> BinTree a b -> BinTree a b
|
|
||||||
filterBinTree = Map.filterWithKey
|
|
||||||
|
|
||||||
tree2list :: BinTree a b -> [(a,b)] -- inorder
|
|
||||||
tree2list = Map.toList
|
|
||||||
|
|
||||||
-- printing
|
-- printing
|
||||||
|
|
||||||
indent :: Int -> String -> String
|
indent :: Int -> String -> String
|
||||||
@@ -297,42 +247,6 @@ chunks sep ws = case span (/= sep) ws of
|
|||||||
readIntArg :: String -> Int
|
readIntArg :: String -> Int
|
||||||
readIntArg n = if (not (null n) && all isDigit n) then read n else 0
|
readIntArg n = if (not (null n) && all isDigit n) then read n else 0
|
||||||
|
|
||||||
{-
|
|
||||||
-- state monad with error; from Agda 6/11/2001
|
|
||||||
|
|
||||||
newtype STM s a = STM (s -> Err (a,s))
|
|
||||||
|
|
||||||
appSTM :: STM s a -> s -> Err (a,s)
|
|
||||||
appSTM (STM f) s = f s
|
|
||||||
|
|
||||||
stm :: (s -> Err (a,s)) -> STM s a
|
|
||||||
stm = STM
|
|
||||||
|
|
||||||
stmr :: (s -> (a,s)) -> STM s a
|
|
||||||
stmr f = stm (\s -> return (f s))
|
|
||||||
|
|
||||||
instance Functor (STM s) where fmap = liftM
|
|
||||||
|
|
||||||
instance Applicative (STM s) where
|
|
||||||
pure = return
|
|
||||||
(<*>) = ap
|
|
||||||
|
|
||||||
instance Monad (STM s) where
|
|
||||||
return a = STM (\s -> return (a,s))
|
|
||||||
STM c >>= f = STM (\s -> do
|
|
||||||
(x,s') <- c s
|
|
||||||
let STM f' = f x
|
|
||||||
f' s')
|
|
||||||
|
|
||||||
readSTM :: STM s s
|
|
||||||
readSTM = stmr (\s -> (s,s))
|
|
||||||
|
|
||||||
updateSTM :: (s -> s) -> STM s ()
|
|
||||||
updateSTM f = stmr (\s -> ((),f s))
|
|
||||||
|
|
||||||
writeSTM :: s -> STM s ()
|
|
||||||
writeSTM s = stmr (const ((),s))
|
|
||||||
-}
|
|
||||||
-- | @return ()@
|
-- | @return ()@
|
||||||
done :: Monad m => m ()
|
done :: Monad m => m ()
|
||||||
done = return ()
|
done = return ()
|
||||||
@@ -377,4 +291,4 @@ doUntil cond ms = case ms of
|
|||||||
v <- a
|
v <- a
|
||||||
if cond v then return v else doUntil cond as
|
if cond v then return v else doUntil cond as
|
||||||
_ -> raise "no result"
|
_ -> raise "no result"
|
||||||
-}
|
-}
|
||||||
|
|||||||
@@ -29,7 +29,7 @@ stripInfo i = case i of
|
|||||||
AbsCat _ -> i
|
AbsCat _ -> i
|
||||||
AbsFun mt mi me mb -> AbsFun mt mi Nothing mb
|
AbsFun mt mi me mb -> AbsFun mt mi Nothing mb
|
||||||
ResParam mp mt -> ResParam mp Nothing
|
ResParam mp mt -> ResParam mp Nothing
|
||||||
ResValue lt -> i ----
|
ResValue _ lt -> i ----
|
||||||
ResOper mt md -> ResOper mt Nothing
|
ResOper mt md -> ResOper mt Nothing
|
||||||
ResOverload is fs -> ResOverload is [(lty, L loc (EInt 0)) | (lty,L loc _) <- fs]
|
ResOverload is fs -> ResOverload is [(lty, L loc (EInt 0)) | (lty,L loc _) <- fs]
|
||||||
CncCat mty mte _ mtf mpmcfg -> CncCat mty Nothing Nothing Nothing Nothing
|
CncCat mty mte _ mtf mpmcfg -> CncCat mty Nothing Nothing Nothing Nothing
|
||||||
@@ -107,8 +107,8 @@ sizeInfo i = case i of
|
|||||||
AbsFun mt mi me mb -> 1 + msize mt +
|
AbsFun mt mi me mb -> 1 + msize mt +
|
||||||
sum [sum (map (sizeTerm . patt2term) ps) + sizeTerm t | Just es <- [me], L _ (ps,t) <- es]
|
sum [sum (map (sizeTerm . patt2term) ps) + sizeTerm t | Just es <- [me], L _ (ps,t) <- es]
|
||||||
ResParam mp mt ->
|
ResParam mp mt ->
|
||||||
1 + sum [1 + sum [1 + sizeTerm ty | (_,_,ty) <- co] | Just (L _ ps) <- [mp], (_,co) <- ps]
|
1 + sum [1 + sum [1 + sizeTerm ty | (_,_,ty) <- co] | Just (L _ ps) <- [mp], (_,co,_) <- ps]
|
||||||
ResValue lt -> 0
|
ResValue _ lt -> 0
|
||||||
ResOper mt md -> 1 + msize mt + msize md
|
ResOper mt md -> 1 + msize mt + msize md
|
||||||
ResOverload is fs -> 1 + sum [sizeTerm ty + sizeTerm tr | (L _ ty, L _ tr) <- fs]
|
ResOverload is fs -> 1 + sum [sizeTerm ty + sizeTerm tr | (L _ ty, L _ tr) <- fs]
|
||||||
CncCat mty _ _ _ _ -> 1 + msize mty -- ignoring lindef, linref and printname
|
CncCat mty _ _ _ _ -> 1 + msize mty -- ignoring lindef, linref and printname
|
||||||
|
|||||||
@@ -15,7 +15,6 @@
|
|||||||
module GF.Grammar.BNFC(BNFCRule(..), BNFCSymbol, Symbol(..), CFTerm(..), bnfc2cf) where
|
module GF.Grammar.BNFC(BNFCRule(..), BNFCSymbol, Symbol(..), CFTerm(..), bnfc2cf) where
|
||||||
|
|
||||||
import GF.Grammar.CFG
|
import GF.Grammar.CFG
|
||||||
import PGF (Token, mkCId)
|
|
||||||
import Data.List (partition)
|
import Data.List (partition)
|
||||||
|
|
||||||
type IsList = Bool
|
type IsList = Bool
|
||||||
@@ -64,12 +63,12 @@ transformRules sepMap (BNFCCoercions c num) = rules ++ [lastRule]
|
|||||||
lastRule = Rule (c',[0]) ss rn
|
lastRule = Rule (c',[0]) ss rn
|
||||||
where c' = c ++ show num
|
where c' = c ++ show num
|
||||||
ss = [Terminal "(", NonTerminal (c,[0]), Terminal ")"]
|
ss = [Terminal "(", NonTerminal (c,[0]), Terminal ")"]
|
||||||
rn = CFObj (mkCId $ "coercion_" ++ c) []
|
rn = CFObj ("coercion_" ++ c) []
|
||||||
|
|
||||||
fRules c n = Rule (c',[0]) ss rn
|
fRules c n = Rule (c',[0]) ss rn
|
||||||
where c' = if n == 0 then c else c ++ show n
|
where c' = if n == 0 then c else c ++ show n
|
||||||
ss = [NonTerminal (c ++ show (n+1),[0])]
|
ss = [NonTerminal (c ++ show (n+1),[0])]
|
||||||
rn = CFObj (mkCId $ "coercion_" ++ c') []
|
rn = CFObj ("coercion_" ++ c') []
|
||||||
|
|
||||||
transformSymb :: SepMap -> BNFCSymbol -> (String, ParamCFSymbol)
|
transformSymb :: SepMap -> BNFCSymbol -> (String, ParamCFSymbol)
|
||||||
transformSymb sepMap s = case s of
|
transformSymb sepMap s = case s of
|
||||||
@@ -94,7 +93,7 @@ createListRules' ne isSep symb c = ruleBase : ruleCons
|
|||||||
then [NonTerminal (c,[0]) | ne]
|
then [NonTerminal (c,[0]) | ne]
|
||||||
else [NonTerminal (c,[0]) | ne] ++
|
else [NonTerminal (c,[0]) | ne] ++
|
||||||
[Terminal symb | symb /= "" && ne]
|
[Terminal symb | symb /= "" && ne]
|
||||||
rn = CFObj (mkCId $ "Base" ++ c) []
|
rn = CFObj ("Base" ++ c) []
|
||||||
ruleCons
|
ruleCons
|
||||||
| isSep && symb /= "" && not ne = [Rule ("List" ++ c,[1]) smbs0 rn
|
| isSep && symb /= "" && not ne = [Rule ("List" ++ c,[1]) smbs0 rn
|
||||||
,Rule ("List" ++ c,[1]) smbs1 rn]
|
,Rule ("List" ++ c,[1]) smbs1 rn]
|
||||||
@@ -107,4 +106,4 @@ createListRules' ne isSep symb c = ruleBase : ruleCons
|
|||||||
smbs = [NonTerminal (c,[0])] ++
|
smbs = [NonTerminal (c,[0])] ++
|
||||||
[Terminal symb | symb /= ""] ++
|
[Terminal symb | symb /= ""] ++
|
||||||
[NonTerminal ("List" ++ c,[0])]
|
[NonTerminal ("List" ++ c,[0])]
|
||||||
rn = CFObj (mkCId $ "Cons" ++ c) []
|
rn = CFObj ("Cons" ++ c) []
|
||||||
|
|||||||
@@ -10,9 +10,9 @@
|
|||||||
module GF.Grammar.Binary(VersionTagged(..),decodeModuleHeader,decodeModule,encodeModule) where
|
module GF.Grammar.Binary(VersionTagged(..),decodeModuleHeader,decodeModule,encodeModule) where
|
||||||
|
|
||||||
import Prelude hiding (catch)
|
import Prelude hiding (catch)
|
||||||
|
import Control.Monad
|
||||||
import Control.Exception(catch,ErrorCall(..),throwIO)
|
import Control.Exception(catch,ErrorCall(..),throwIO)
|
||||||
|
import Data.Binary
|
||||||
import PGF.Internal(Binary(..),Word8,putWord8,getWord8,encodeFile,decodeFile)
|
|
||||||
import qualified Data.Map as Map(empty)
|
import qualified Data.Map as Map(empty)
|
||||||
import qualified Data.ByteString.Char8 as BS
|
import qualified Data.ByteString.Char8 as BS
|
||||||
|
|
||||||
@@ -22,11 +22,10 @@ import GF.Infra.Option
|
|||||||
import GF.Infra.UseIO(MonadIO(..))
|
import GF.Infra.UseIO(MonadIO(..))
|
||||||
import GF.Grammar.Grammar
|
import GF.Grammar.Grammar
|
||||||
|
|
||||||
import PGF() -- Binary instances
|
import PGF2.Internal(Literal(..),Symbol(..))
|
||||||
import PGF.Internal(Literal(..))
|
|
||||||
|
|
||||||
-- Please change this every time when the GFO format is changed
|
-- Please change this every time when the GFO format is changed
|
||||||
gfoVersion = "GF04"
|
gfoVersion = "GF05"
|
||||||
|
|
||||||
instance Binary Ident where
|
instance Binary Ident where
|
||||||
put id = put (ident2utf8 id)
|
put id = put (ident2utf8 id)
|
||||||
@@ -120,7 +119,7 @@ instance Binary Info where
|
|||||||
put (AbsCat x) = putWord8 0 >> put x
|
put (AbsCat x) = putWord8 0 >> put x
|
||||||
put (AbsFun w x y z) = putWord8 1 >> put (w,x,y,z)
|
put (AbsFun w x y z) = putWord8 1 >> put (w,x,y,z)
|
||||||
put (ResParam x y) = putWord8 2 >> put (x,y)
|
put (ResParam x y) = putWord8 2 >> put (x,y)
|
||||||
put (ResValue x) = putWord8 3 >> put x
|
put (ResValue x y) = putWord8 3 >> put (x,y)
|
||||||
put (ResOper x y) = putWord8 4 >> put (x,y)
|
put (ResOper x y) = putWord8 4 >> put (x,y)
|
||||||
put (ResOverload x y)= putWord8 5 >> put (x,y)
|
put (ResOverload x y)= putWord8 5 >> put (x,y)
|
||||||
put (CncCat v w x y z)=putWord8 6 >> put (v,w,x,y,z)
|
put (CncCat v w x y z)=putWord8 6 >> put (v,w,x,y,z)
|
||||||
@@ -131,7 +130,7 @@ instance Binary Info where
|
|||||||
0 -> get >>= \x -> return (AbsCat x)
|
0 -> get >>= \x -> return (AbsCat x)
|
||||||
1 -> get >>= \(w,x,y,z) -> return (AbsFun w x y z)
|
1 -> get >>= \(w,x,y,z) -> return (AbsFun w x y z)
|
||||||
2 -> get >>= \(x,y) -> return (ResParam x y)
|
2 -> get >>= \(x,y) -> return (ResParam x y)
|
||||||
3 -> get >>= \x -> return (ResValue x)
|
3 -> get >>= \(x,y) -> return (ResValue x y)
|
||||||
4 -> get >>= \(x,y) -> return (ResOper x y)
|
4 -> get >>= \(x,y) -> return (ResOper x y)
|
||||||
5 -> get >>= \(x,y) -> return (ResOverload x y)
|
5 -> get >>= \(x,y) -> return (ResOverload x y)
|
||||||
6 -> get >>= \(v,w,x,y,z)->return (CncCat v w x y z)
|
6 -> get >>= \(v,w,x,y,z)->return (CncCat v w x y z)
|
||||||
@@ -298,6 +297,53 @@ instance Binary Label where
|
|||||||
1 -> fmap LVar get
|
1 -> fmap LVar get
|
||||||
_ -> decodingError
|
_ -> decodingError
|
||||||
|
|
||||||
|
instance Binary BindType where
|
||||||
|
put Explicit = putWord8 0
|
||||||
|
put Implicit = putWord8 1
|
||||||
|
get = do tag <- getWord8
|
||||||
|
case tag of
|
||||||
|
0 -> return Explicit
|
||||||
|
1 -> return Implicit
|
||||||
|
_ -> decodingError
|
||||||
|
|
||||||
|
instance Binary Literal where
|
||||||
|
put (LStr s) = putWord8 0 >> put s
|
||||||
|
put (LInt i) = putWord8 1 >> put i
|
||||||
|
put (LFlt d) = putWord8 2 >> put d
|
||||||
|
get = do tag <- getWord8
|
||||||
|
case tag of
|
||||||
|
0 -> liftM LStr get
|
||||||
|
1 -> liftM LInt get
|
||||||
|
2 -> liftM LFlt get
|
||||||
|
_ -> decodingError
|
||||||
|
|
||||||
|
instance Binary Symbol where
|
||||||
|
put (SymCat n l) = putWord8 0 >> put (n,l)
|
||||||
|
put (SymLit n l) = putWord8 1 >> put (n,l)
|
||||||
|
put (SymVar n l) = putWord8 2 >> put (n,l)
|
||||||
|
put (SymKS ts) = putWord8 3 >> put ts
|
||||||
|
put (SymKP d vs) = putWord8 4 >> put (d,vs)
|
||||||
|
put SymBIND = putWord8 5
|
||||||
|
put SymSOFT_BIND = putWord8 6
|
||||||
|
put SymNE = putWord8 7
|
||||||
|
put SymSOFT_SPACE = putWord8 8
|
||||||
|
put SymCAPIT = putWord8 9
|
||||||
|
put SymALL_CAPIT = putWord8 10
|
||||||
|
get = do tag <- getWord8
|
||||||
|
case tag of
|
||||||
|
0 -> liftM2 SymCat get get
|
||||||
|
1 -> liftM2 SymLit get get
|
||||||
|
2 -> liftM2 SymVar get get
|
||||||
|
3 -> liftM SymKS get
|
||||||
|
4 -> liftM2 (\d vs -> SymKP d vs) get get
|
||||||
|
5 -> return SymBIND
|
||||||
|
6 -> return SymSOFT_BIND
|
||||||
|
7 -> return SymNE
|
||||||
|
8 -> return SymSOFT_SPACE
|
||||||
|
9 -> return SymCAPIT
|
||||||
|
10-> return SymALL_CAPIT
|
||||||
|
_ -> decodingError
|
||||||
|
|
||||||
--putGFOVersion = mapM_ (putWord8 . fromIntegral . ord) gfoVersion
|
--putGFOVersion = mapM_ (putWord8 . fromIntegral . ord) gfoVersion
|
||||||
--getGFOVersion = replicateM (length gfoVersion) (fmap (chr . fromIntegral) getWord8)
|
--getGFOVersion = replicateM (length gfoVersion) (fmap (chr . fromIntegral) getWord8)
|
||||||
--putGFOVersion = put gfoVersion
|
--putGFOVersion = put gfoVersion
|
||||||
|
|||||||
@@ -4,10 +4,11 @@
|
|||||||
--
|
--
|
||||||
-- Context-free grammar representation and manipulation.
|
-- Context-free grammar representation and manipulation.
|
||||||
----------------------------------------------------------------------
|
----------------------------------------------------------------------
|
||||||
module GF.Grammar.CFG where
|
module GF.Grammar.CFG(Cat,Token, module GF.Grammar.CFG) where
|
||||||
|
|
||||||
import GF.Data.Utilities
|
import GF.Data.Utilities
|
||||||
import PGF
|
import PGF2(Fun,Cat)
|
||||||
|
import PGF2.Internal(Token)
|
||||||
import GF.Data.Relation
|
import GF.Data.Relation
|
||||||
|
|
||||||
import Data.Map (Map)
|
import Data.Map (Map)
|
||||||
@@ -20,8 +21,6 @@ import qualified Data.Set as Set
|
|||||||
-- * Types
|
-- * Types
|
||||||
--
|
--
|
||||||
|
|
||||||
type Cat = String
|
|
||||||
|
|
||||||
data Symbol c t = NonTerminal c | Terminal t
|
data Symbol c t = NonTerminal c | Terminal t
|
||||||
deriving (Eq, Ord, Show)
|
deriving (Eq, Ord, Show)
|
||||||
|
|
||||||
@@ -39,12 +38,12 @@ data Grammar c t = Grammar {
|
|||||||
deriving (Eq, Ord, Show)
|
deriving (Eq, Ord, Show)
|
||||||
|
|
||||||
data CFTerm
|
data CFTerm
|
||||||
= CFObj CId [CFTerm] -- ^ an abstract syntax function with arguments
|
= CFObj Fun [CFTerm] -- ^ an abstract syntax function with arguments
|
||||||
| CFAbs Int CFTerm -- ^ A lambda abstraction. The Int is the variable id.
|
| CFAbs Int CFTerm -- ^ A lambda abstraction. The Int is the variable id.
|
||||||
| CFApp CFTerm CFTerm -- ^ Application
|
| CFApp CFTerm CFTerm -- ^ Application
|
||||||
| CFRes Int -- ^ The result of the n:th (0-based) non-terminal
|
| CFRes Int -- ^ The result of the n:th (0-based) non-terminal
|
||||||
| CFVar Int -- ^ A lambda-bound variable
|
| CFVar Int -- ^ A lambda-bound variable
|
||||||
| CFMeta CId -- ^ A metavariable
|
| CFMeta Fun -- ^ A metavariable
|
||||||
deriving (Eq, Ord, Show)
|
deriving (Eq, Ord, Show)
|
||||||
|
|
||||||
type CFSymbol = Symbol Cat Token
|
type CFSymbol = Symbol Cat Token
|
||||||
@@ -232,7 +231,7 @@ uniqueFuns = snd . mapAccumL uniqueFun Set.empty
|
|||||||
uniqueFun funs (Rule cat items (CFObj fun args)) = (Set.insert fun' funs,Rule cat items (CFObj fun' args))
|
uniqueFun funs (Rule cat items (CFObj fun args)) = (Set.insert fun' funs,Rule cat items (CFObj fun' args))
|
||||||
where
|
where
|
||||||
fun' = head [fun'|suffix<-"":map show ([2..]::[Int]),
|
fun' = head [fun'|suffix<-"":map show ([2..]::[Int]),
|
||||||
let fun'=mkCId (showCId fun++suffix),
|
let fun'=fun++suffix,
|
||||||
not (fun' `Set.member` funs)]
|
not (fun' `Set.member` funs)]
|
||||||
|
|
||||||
-- | Gets all rules in a CFG.
|
-- | Gets all rules in a CFG.
|
||||||
@@ -310,12 +309,12 @@ prProductions prods =
|
|||||||
prCFTerm :: CFTerm -> String
|
prCFTerm :: CFTerm -> String
|
||||||
prCFTerm = pr 0
|
prCFTerm = pr 0
|
||||||
where
|
where
|
||||||
pr p (CFObj f args) = paren p (showCId f ++ " (" ++ concat (intersperse "," (map (pr 0) args)) ++ ")")
|
pr p (CFObj f args) = paren p (f ++ " (" ++ concat (intersperse "," (map (pr 0) args)) ++ ")")
|
||||||
pr p (CFAbs i t) = paren p ("\\x" ++ show i ++ ". " ++ pr 0 t)
|
pr p (CFAbs i t) = paren p ("\\x" ++ show i ++ ". " ++ pr 0 t)
|
||||||
pr p (CFApp t1 t2) = paren p (pr 1 t1 ++ "(" ++ pr 0 t2 ++ ")")
|
pr p (CFApp t1 t2) = paren p (pr 1 t1 ++ "(" ++ pr 0 t2 ++ ")")
|
||||||
pr _ (CFRes i) = "$" ++ show i
|
pr _ (CFRes i) = "$" ++ show i
|
||||||
pr _ (CFVar i) = "x" ++ show i
|
pr _ (CFVar i) = "x" ++ show i
|
||||||
pr _ (CFMeta c) = "?" ++ showCId c
|
pr _ (CFMeta c) = "?" ++ c
|
||||||
paren 0 x = x
|
paren 0 x = x
|
||||||
paren 1 x = "(" ++ x ++ ")"
|
paren 1 x = "(" ++ x ++ ")"
|
||||||
|
|
||||||
@@ -323,12 +322,12 @@ prCFTerm = pr 0
|
|||||||
-- * CFRule Utilities
|
-- * CFRule Utilities
|
||||||
--
|
--
|
||||||
|
|
||||||
ruleFun :: Rule c t -> CId
|
ruleFun :: Rule c t -> Fun
|
||||||
ruleFun (Rule _ _ t) = f t
|
ruleFun (Rule _ _ t) = f t
|
||||||
where f (CFObj n _) = n
|
where f (CFObj n _) = n
|
||||||
f (CFApp _ x) = f x
|
f (CFApp _ x) = f x
|
||||||
f (CFAbs _ x) = f x
|
f (CFAbs _ x) = f x
|
||||||
f _ = mkCId ""
|
f _ = ""
|
||||||
|
|
||||||
-- | Check if any of the categories used on the right-hand side
|
-- | Check if any of the categories used on the right-hand side
|
||||||
-- are in the given list of categories.
|
-- are in the given list of categories.
|
||||||
@@ -336,7 +335,7 @@ anyUsedBy :: Eq c => [c] -> Rule c t -> Bool
|
|||||||
anyUsedBy cs (Rule _ ss _) = any (`elem` cs) (filterCats ss)
|
anyUsedBy cs (Rule _ ss _) = any (`elem` cs) (filterCats ss)
|
||||||
|
|
||||||
mkCFTerm :: String -> CFTerm
|
mkCFTerm :: String -> CFTerm
|
||||||
mkCFTerm n = CFObj (mkCId n) []
|
mkCFTerm n = CFObj n []
|
||||||
|
|
||||||
ruleIsNonRecursive :: Ord c => Set c -> Rule c t -> Bool
|
ruleIsNonRecursive :: Ord c => Set c -> Rule c t -> Bool
|
||||||
ruleIsNonRecursive cs = noCatsInSet cs . ruleRhs
|
ruleIsNonRecursive cs = noCatsInSet cs . ruleRhs
|
||||||
|
|||||||
@@ -16,7 +16,6 @@ module GF.Grammar.EBNF (EBNF, ERule, ERHS(..), ebnf2cf) where
|
|||||||
|
|
||||||
import GF.Data.Operations
|
import GF.Data.Operations
|
||||||
import GF.Grammar.CFG
|
import GF.Grammar.CFG
|
||||||
import PGF (mkCId)
|
|
||||||
|
|
||||||
type EBNF = [ERule]
|
type EBNF = [ERule]
|
||||||
type ERule = (ECat, ERHS)
|
type ERule = (ECat, ERHS)
|
||||||
@@ -40,7 +39,7 @@ ebnf2cf :: EBNF -> [ParamCFRule]
|
|||||||
ebnf2cf ebnf =
|
ebnf2cf ebnf =
|
||||||
[Rule cat items (mkCFF i cat) | (i,(cat,items)) <- zip [0..] (normEBNF ebnf)]
|
[Rule cat items (mkCFF i cat) | (i,(cat,items)) <- zip [0..] (normEBNF ebnf)]
|
||||||
where
|
where
|
||||||
mkCFF i (c,_) = CFObj (mkCId ("Mk" ++ c ++ "_" ++ show i)) []
|
mkCFF i (c,_) = CFObj ("Mk" ++ c ++ "_" ++ show i) []
|
||||||
|
|
||||||
normEBNF :: EBNF -> [CFJustRule]
|
normEBNF :: EBNF -> [CFJustRule]
|
||||||
normEBNF erules = let
|
normEBNF erules = let
|
||||||
|
|||||||
@@ -64,7 +64,7 @@ module GF.Grammar.Grammar (
|
|||||||
Location(..), L(..), unLoc, noLoc, ppLocation, ppL,
|
Location(..), L(..), unLoc, noLoc, ppLocation, ppL,
|
||||||
|
|
||||||
-- ** PMCFG
|
-- ** PMCFG
|
||||||
PMCFG(..), Production(..), FId, FunId, SeqId, LIndex, Sequence
|
PMCFG(..), Production(..), FId, FunId, SeqId, LIndex
|
||||||
) where
|
) where
|
||||||
|
|
||||||
import GF.Infra.Ident
|
import GF.Infra.Ident
|
||||||
@@ -73,7 +73,8 @@ import GF.Infra.Location
|
|||||||
|
|
||||||
import GF.Data.Operations
|
import GF.Data.Operations
|
||||||
|
|
||||||
import PGF.Internal (FId, FunId, SeqId, LIndex, Sequence, BindType(..))
|
import PGF2(LIndex, BindType(..))
|
||||||
|
import PGF2.Internal(FId, FunId, SeqId, Symbol)
|
||||||
|
|
||||||
import Data.Array.IArray(Array)
|
import Data.Array.IArray(Array)
|
||||||
import Data.Array.Unboxed(UArray)
|
import Data.Array.Unboxed(UArray)
|
||||||
@@ -99,7 +100,7 @@ data ModuleInfo = ModInfo {
|
|||||||
mopens :: [OpenSpec],
|
mopens :: [OpenSpec],
|
||||||
mexdeps :: [ModuleName],
|
mexdeps :: [ModuleName],
|
||||||
msrc :: FilePath,
|
msrc :: FilePath,
|
||||||
mseqs :: Maybe (Array SeqId Sequence),
|
mseqs :: Maybe (Array SeqId [Symbol]),
|
||||||
jments :: Map.Map Ident Info
|
jments :: Map.Map Ident Info
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -329,7 +330,7 @@ data Info =
|
|||||||
|
|
||||||
-- judgements in resource
|
-- judgements in resource
|
||||||
| ResParam (Maybe (L [Param])) (Maybe [Term]) -- ^ (/RES/) the second parameter is list of all possible values
|
| ResParam (Maybe (L [Param])) (Maybe [Term]) -- ^ (/RES/) the second parameter is list of all possible values
|
||||||
| ResValue (L Type) -- ^ (/RES/) to mark parameter constructors for lookup
|
| ResValue (L Type) Int -- ^ (/RES/) to mark parameter constructors for lookup
|
||||||
| ResOper (Maybe (L Type)) (Maybe (L Term)) -- ^ (/RES/)
|
| ResOper (Maybe (L Type)) (Maybe (L Term)) -- ^ (/RES/)
|
||||||
|
|
||||||
| ResOverload [ModuleName] [(L Type,L Term)] -- ^ (/RES/) idents: modules inherited
|
| ResOverload [ModuleName] [(L Type,L Term)] -- ^ (/RES/) idents: modules inherited
|
||||||
@@ -458,7 +459,7 @@ type Case = (Patt, Term)
|
|||||||
--type Cases = ([Patt], Term)
|
--type Cases = ([Patt], Term)
|
||||||
type LocalDef = (Ident, (Maybe Type, Term))
|
type LocalDef = (Ident, (Maybe Type, Term))
|
||||||
|
|
||||||
type Param = (Ident, Context)
|
type Param = (Ident, Context, Int)
|
||||||
type Altern = (Term, [(Term, Term)])
|
type Altern = (Term, [(Term, Term)])
|
||||||
|
|
||||||
type Substitution = [(Ident, Term)]
|
type Substitution = [(Ident, Term)]
|
||||||
|
|||||||
@@ -23,10 +23,11 @@ module GF.Grammar.Lookup (
|
|||||||
lookupResType,
|
lookupResType,
|
||||||
lookupOverload,
|
lookupOverload,
|
||||||
lookupOverloadTypes,
|
lookupOverloadTypes,
|
||||||
lookupParamValues,
|
lookupParamValues,
|
||||||
allParamValues,
|
allParamValues,
|
||||||
lookupAbsDef,
|
lookupParamValueIndex,
|
||||||
lookupLincat,
|
lookupAbsDef,
|
||||||
|
lookupLincat,
|
||||||
lookupFunType,
|
lookupFunType,
|
||||||
lookupCatContext,
|
lookupCatContext,
|
||||||
allOpers, allOpersTo
|
allOpers, allOpersTo
|
||||||
@@ -51,11 +52,11 @@ lock c = lockRecType c -- return
|
|||||||
unlock c = unlockRecord c -- return
|
unlock c = unlockRecord c -- return
|
||||||
|
|
||||||
-- to look up a constant etc in a search tree --- why here? AR 29/5/2008
|
-- to look up a constant etc in a search tree --- why here? AR 29/5/2008
|
||||||
lookupIdent :: ErrorMonad m => Ident -> BinTree Ident b -> m b
|
lookupIdent :: ErrorMonad m => Ident -> Map.Map Ident b -> m b
|
||||||
lookupIdent c t =
|
lookupIdent c t =
|
||||||
case lookupTree showIdent c t of
|
case Map.lookup c t of
|
||||||
Ok v -> return v
|
Just v -> return v
|
||||||
Bad _ -> raise ("unknown identifier" +++ showIdent c)
|
Nothing -> raise ("unknown identifier" +++ showIdent c)
|
||||||
|
|
||||||
lookupIdentInfo :: ErrorMonad m => SourceModInfo -> Ident -> m Info
|
lookupIdentInfo :: ErrorMonad m => SourceModInfo -> Ident -> m Info
|
||||||
lookupIdentInfo mo i = lookupIdent i (jments mo)
|
lookupIdentInfo mo i = lookupIdent i (jments mo)
|
||||||
@@ -83,7 +84,7 @@ lookupResDefLoc gr (m,c)
|
|||||||
|
|
||||||
AnyInd _ n -> look n c
|
AnyInd _ n -> look n c
|
||||||
ResParam _ _ -> return (noLoc (QC (m,c)))
|
ResParam _ _ -> return (noLoc (QC (m,c)))
|
||||||
ResValue _ -> return (noLoc (QC (m,c)))
|
ResValue _ _ -> return (noLoc (QC (m,c)))
|
||||||
_ -> raise $ render (c <+> "is not defined in resource" <+> m)
|
_ -> raise $ render (c <+> "is not defined in resource" <+> m)
|
||||||
|
|
||||||
lookupResType :: ErrorMonad m => Grammar -> QIdent -> m Type
|
lookupResType :: ErrorMonad m => Grammar -> QIdent -> m Type
|
||||||
@@ -99,7 +100,7 @@ lookupResType gr (m,c) = do
|
|||||||
return $ mkProd cont val' []
|
return $ mkProd cont val' []
|
||||||
AnyInd _ n -> lookupResType gr (n,c)
|
AnyInd _ n -> lookupResType gr (n,c)
|
||||||
ResParam _ _ -> return typePType
|
ResParam _ _ -> return typePType
|
||||||
ResValue (L _ t) -> return t
|
ResValue (L _ t) _-> return t
|
||||||
_ -> raise $ render (c <+> "has no type defined in resource" <+> m)
|
_ -> raise $ render (c <+> "has no type defined in resource" <+> m)
|
||||||
|
|
||||||
lookupOverloadTypes :: ErrorMonad m => Grammar -> QIdent -> m [(Term,Type)]
|
lookupOverloadTypes :: ErrorMonad m => Grammar -> QIdent -> m [(Term,Type)]
|
||||||
@@ -113,8 +114,8 @@ lookupOverloadTypes gr id@(m,c) = do
|
|||||||
CncFun (Just (cat,cont,val)) _ _ _ -> do
|
CncFun (Just (cat,cont,val)) _ _ _ -> do
|
||||||
val' <- lock cat val
|
val' <- lock cat val
|
||||||
ret $ mkProd cont val' []
|
ret $ mkProd cont val' []
|
||||||
ResParam _ _ -> ret typePType
|
ResParam _ _ -> ret typePType
|
||||||
ResValue (L _ t) -> ret t
|
ResValue (L _ t) _ -> ret t
|
||||||
ResOverload os tysts -> do
|
ResOverload os tysts -> do
|
||||||
tss <- mapM (\x -> lookupOverloadTypes gr (x,c)) os
|
tss <- mapM (\x -> lookupOverloadTypes gr (x,c)) os
|
||||||
return $ [(tr,ty) | (L _ ty,L _ tr) <- tysts] ++
|
return $ [(tr,ty) | (L _ ty,L _ tr) <- tysts] ++
|
||||||
@@ -148,7 +149,7 @@ lookupOrigInfo gr (m,c) = do
|
|||||||
allOrigInfos :: Grammar -> ModuleName -> [(QIdent,Info)]
|
allOrigInfos :: Grammar -> ModuleName -> [(QIdent,Info)]
|
||||||
allOrigInfos gr m = fromErr [] $ do
|
allOrigInfos gr m = fromErr [] $ do
|
||||||
mo <- lookupModule gr m
|
mo <- lookupModule gr m
|
||||||
return [((m,c),i) | (c,_) <- tree2list (jments mo), Ok (m,i) <- [lookupOrigInfo gr (m,c)]]
|
return [((m,c),i) | (c,_) <- Map.toList (jments mo), Ok (m,i) <- [lookupOrigInfo gr (m,c)]]
|
||||||
|
|
||||||
lookupParamValues :: ErrorMonad m => Grammar -> QIdent -> m [Term]
|
lookupParamValues :: ErrorMonad m => Grammar -> QIdent -> m [Term]
|
||||||
lookupParamValues gr c = do
|
lookupParamValues gr c = do
|
||||||
@@ -176,6 +177,13 @@ allParamValues cnc ptyp =
|
|||||||
-- to normalize records and record types
|
-- to normalize records and record types
|
||||||
sortByFst = sortBy (\ x y -> compare (fst x) (fst y))
|
sortByFst = sortBy (\ x y -> compare (fst x) (fst y))
|
||||||
|
|
||||||
|
lookupParamValueIndex :: ErrorMonad m => Grammar -> QIdent -> m Int
|
||||||
|
lookupParamValueIndex gr c = do
|
||||||
|
(_,info) <- lookupOrigInfo gr c
|
||||||
|
case info of
|
||||||
|
ResValue _ i -> return i
|
||||||
|
_ -> raise $ render (ppQIdent Qualified c <+> "has no parameter index defined")
|
||||||
|
|
||||||
lookupAbsDef :: ErrorMonad m => Grammar -> ModuleName -> Ident -> m (Maybe Int,Maybe [Equation])
|
lookupAbsDef :: ErrorMonad m => Grammar -> ModuleName -> Ident -> m (Maybe Int,Maybe [Equation])
|
||||||
lookupAbsDef gr m c = errIn (render ("looking up absdef of" <+> c)) $ do
|
lookupAbsDef gr m c = errIn (render ("looking up absdef of" <+> c)) $ do
|
||||||
info <- lookupQIdentInfo gr (m,c)
|
info <- lookupQIdentInfo gr (m,c)
|
||||||
@@ -226,7 +234,7 @@ allOpers gr =
|
|||||||
typesIn info = case info of
|
typesIn info = case info of
|
||||||
AbsFun (Just ltyp) _ _ _ -> [ltyp]
|
AbsFun (Just ltyp) _ _ _ -> [ltyp]
|
||||||
ResOper (Just ltyp) _ -> [ltyp]
|
ResOper (Just ltyp) _ -> [ltyp]
|
||||||
ResValue ltyp -> [ltyp]
|
ResValue ltyp _ -> [ltyp]
|
||||||
ResOverload _ tytrs -> [ltyp | (ltyp,_) <- tytrs]
|
ResOverload _ tytrs -> [ltyp | (ltyp,_) <- tytrs]
|
||||||
CncFun (Just (i,ctx,typ)) _ _ _ ->
|
CncFun (Just (i,ctx,typ)) _ _ _ ->
|
||||||
[L NoLoc (mkProdSimple ctx (lock' i typ))]
|
[L NoLoc (mkProdSimple ctx (lock' i typ))]
|
||||||
|
|||||||
@@ -22,14 +22,13 @@ import GF.Data.Operations
|
|||||||
import GF.Data.Str
|
import GF.Data.Str
|
||||||
import GF.Infra.Ident
|
import GF.Infra.Ident
|
||||||
import GF.Grammar.Grammar
|
import GF.Grammar.Grammar
|
||||||
--import GF.Grammar.Values
|
|
||||||
import GF.Grammar.Predef
|
import GF.Grammar.Predef
|
||||||
import GF.Grammar.Printer
|
import GF.Grammar.Printer
|
||||||
|
|
||||||
import Control.Monad.Identity(Identity(..))
|
import Control.Monad.Identity(Identity(..))
|
||||||
import qualified Data.Traversable as T(mapM)
|
import qualified Data.Traversable as T(mapM)
|
||||||
|
import qualified Data.Map as Map
|
||||||
import Control.Monad (liftM, liftM2, liftM3)
|
import Control.Monad (liftM, liftM2, liftM3)
|
||||||
--import Data.Char (isDigit)
|
|
||||||
import Data.List (sortBy,nub)
|
import Data.List (sortBy,nub)
|
||||||
import Data.Monoid
|
import Data.Monoid
|
||||||
import GF.Text.Pretty(render,(<+>),hsep,fsep)
|
import GF.Text.Pretty(render,(<+>),hsep,fsep)
|
||||||
@@ -48,7 +47,7 @@ typeForm t =
|
|||||||
Q c -> ([],c,[])
|
Q c -> ([],c,[])
|
||||||
QC c -> ([],c,[])
|
QC c -> ([],c,[])
|
||||||
Sort c -> ([],(MN identW, c),[])
|
Sort c -> ([],(MN identW, c),[])
|
||||||
_ -> error (render ("no normal form of type" <+> ppTerm Unqualified 0 t))
|
_ -> error (render ("no normal form of type" <+> show t))
|
||||||
|
|
||||||
typeFormCnc :: Type -> (Context, Type)
|
typeFormCnc :: Type -> (Context, Type)
|
||||||
typeFormCnc t =
|
typeFormCnc t =
|
||||||
@@ -608,20 +607,22 @@ sortRec = sortBy ordLabel where
|
|||||||
|
|
||||||
-- | dependency check, detecting circularities and returning topo-sorted list
|
-- | dependency check, detecting circularities and returning topo-sorted list
|
||||||
|
|
||||||
allDependencies :: (ModuleName -> Bool) -> BinTree Ident Info -> [(Ident,[Ident])]
|
allDependencies :: (ModuleName -> Bool) -> Map.Map Ident Info -> [(Ident,[Ident])]
|
||||||
allDependencies ism b =
|
allDependencies ism b =
|
||||||
[(f, nub (concatMap opty (pts i))) | (f,i) <- tree2list b]
|
[(f, nub (concatMap opty (pts i))) | (f,i) <- Map.toList b]
|
||||||
where
|
where
|
||||||
opersIn t = case t of
|
opersIn t = case t of
|
||||||
Q (n,c) | ism n -> [c]
|
Q (n,c) | ism n -> [c]
|
||||||
QC (n,c) | ism n -> [c]
|
QC (n,c) | ism n -> [c]
|
||||||
|
Cn c -> [c]
|
||||||
_ -> collectOp opersIn t
|
_ -> collectOp opersIn t
|
||||||
opty (Just (L _ ty)) = opersIn ty
|
opty (Just (L _ ty)) = opersIn ty
|
||||||
opty _ = []
|
opty _ = []
|
||||||
pts i = case i of
|
pts i = case i of
|
||||||
ResOper pty pt -> [pty,pt]
|
ResOper pty pt -> [pty,pt]
|
||||||
ResOverload _ tyts -> concat [[Just ty, Just tr] | (ty,tr) <- tyts]
|
ResOverload _ tyts -> concat [[Just ty, Just tr] | (ty,tr) <- tyts]
|
||||||
ResParam (Just (L loc ps)) _ -> [Just (L loc t) | (_,cont) <- ps, (_,_,t) <- cont]
|
ResParam (Just (L loc ps)) _ -> [Just (L loc t) | (_,cont,_) <- ps, (_,_,t) <- cont]
|
||||||
|
ResValue pty _ -> [Just pty]
|
||||||
CncCat pty _ _ _ _ -> [pty]
|
CncCat pty _ _ _ _ -> [pty]
|
||||||
CncFun _ pt _ _ -> [pt] ---- (Maybe (Ident,(Context,Type))
|
CncFun _ pt _ _ -> [pt] ---- (Maybe (Ident,(Context,Type))
|
||||||
AbsFun pty _ ptr _ -> [pty] --- ptr is def, which can be mutual
|
AbsFun pty _ ptr _ -> [pty] --- ptr is def, which can be mutual
|
||||||
@@ -634,7 +635,7 @@ topoSortJments (m,mi) = do
|
|||||||
return
|
return
|
||||||
(\cyc -> raise (render ("circular definitions:" <+> fsep (head cyc))))
|
(\cyc -> raise (render ("circular definitions:" <+> fsep (head cyc))))
|
||||||
(topoTest (allDependencies (==m) (jments mi)))
|
(topoTest (allDependencies (==m) (jments mi)))
|
||||||
return (reverse [(i,info) | i <- is, Ok info <- [lookupTree showIdent i (jments mi)]])
|
return (reverse [(i,info) | i <- is, Just info <- [Map.lookup i (jments mi)]])
|
||||||
|
|
||||||
topoSortJments2 :: ErrorMonad m => SourceModule -> m [[(Ident,Info)]]
|
topoSortJments2 :: ErrorMonad m => SourceModule -> m [[(Ident,Info)]]
|
||||||
topoSortJments2 (m,mi) = do
|
topoSortJments2 (m,mi) = do
|
||||||
@@ -644,4 +645,4 @@ topoSortJments2 (m,mi) = do
|
|||||||
<+> fsep (head cyc))))
|
<+> fsep (head cyc))))
|
||||||
(topoTest2 (allDependencies (==m) (jments mi)))
|
(topoTest2 (allDependencies (==m) (jments mi)))
|
||||||
return
|
return
|
||||||
[[(i,info) | i<-is,Ok info<-[lookupTree showIdent i (jments mi)]] | is<-iss]
|
[[(i,info) | i<-is,Just info<-[Map.lookup i (jments mi)]] | is<-iss]
|
||||||
|
|||||||
@@ -24,7 +24,7 @@ import GF.Grammar.Lexer
|
|||||||
import GF.Compile.Update (buildAnyTree)
|
import GF.Compile.Update (buildAnyTree)
|
||||||
import Data.List(intersperse)
|
import Data.List(intersperse)
|
||||||
import Data.Char(isAlphaNum)
|
import Data.Char(isAlphaNum)
|
||||||
import PGF(mkCId)
|
import qualified Data.Map as Map
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -139,7 +139,7 @@ ModHeader
|
|||||||
: ComplMod ModType '=' ModHeaderBody { let { mstat = $1 ;
|
: ComplMod ModType '=' ModHeaderBody { let { mstat = $1 ;
|
||||||
(mtype,id) = $2 ;
|
(mtype,id) = $2 ;
|
||||||
(extends,with,opens) = $4 }
|
(extends,with,opens) = $4 }
|
||||||
in (id, ModInfo mtype mstat noOptions extends with opens [] "" Nothing emptyBinTree) }
|
in (id, ModInfo mtype mstat noOptions extends with opens [] "" Nothing Map.empty) }
|
||||||
|
|
||||||
ComplMod :: { ModuleStatus }
|
ComplMod :: { ModuleStatus }
|
||||||
ComplMod
|
ComplMod
|
||||||
@@ -267,7 +267,7 @@ DataDef
|
|||||||
ParamDef :: { [(Ident,Info)] }
|
ParamDef :: { [(Ident,Info)] }
|
||||||
ParamDef
|
ParamDef
|
||||||
: Posn LhsIdent '=' ListParConstr Posn { ($2, ResParam (Just (mkL $1 $5 [param | L loc param <- $4])) Nothing) :
|
: Posn LhsIdent '=' ListParConstr Posn { ($2, ResParam (Just (mkL $1 $5 [param | L loc param <- $4])) Nothing) :
|
||||||
[(f, ResValue (L loc (mkProdSimple co (Cn $2)))) | L loc (f,co) <- $4] }
|
[(f, ResValue (L loc (mkProdSimple co (Cn $2))) i) | L loc (f,co,i) <- $4] }
|
||||||
| Posn LhsIdent Posn { [($2, ResParam Nothing Nothing)] }
|
| Posn LhsIdent Posn { [($2, ResParam Nothing Nothing)] }
|
||||||
|
|
||||||
OperDef :: { [(Ident,Info)] }
|
OperDef :: { [(Ident,Info)] }
|
||||||
@@ -302,7 +302,7 @@ ListDataConstr
|
|||||||
|
|
||||||
ParConstr :: { L Param }
|
ParConstr :: { L Param }
|
||||||
ParConstr
|
ParConstr
|
||||||
: Posn Ident ListDDecl Posn { mkL $1 $4 ($2,$3) }
|
: Posn Ident ListDDecl Posn { mkL $1 $4 ($2,$3,0) }
|
||||||
|
|
||||||
ListLinDef :: { [(Ident,Info)] }
|
ListLinDef :: { [(Ident,Info)] }
|
||||||
ListLinDef
|
ListLinDef
|
||||||
@@ -624,7 +624,7 @@ ListCFRule
|
|||||||
|
|
||||||
CFRule :: { [BNFCRule] }
|
CFRule :: { [BNFCRule] }
|
||||||
CFRule
|
CFRule
|
||||||
: Ident '.' Ident '::=' ListCFSymbol ';' { [BNFCRule (showIdent $3) $5 (CFObj (mkCId (showIdent $1)) [])]
|
: Ident '.' Ident '::=' ListCFSymbol ';' { [BNFCRule (showIdent $3) $5 (CFObj (showIdent $1) [])]
|
||||||
}
|
}
|
||||||
| Ident '::=' ListCFRHS ';' { let { cat = showIdent $1;
|
| Ident '::=' ListCFRHS ';' { let { cat = showIdent $1;
|
||||||
mkFun cat its =
|
mkFun cat its =
|
||||||
@@ -637,7 +637,7 @@ CFRule
|
|||||||
Terminal c -> filter isAlphaNum c;
|
Terminal c -> filter isAlphaNum c;
|
||||||
NonTerminal (t,_) -> t
|
NonTerminal (t,_) -> t
|
||||||
}
|
}
|
||||||
} in map (\rhs -> BNFCRule cat rhs (CFObj (mkCId (mkFun cat rhs)) [])) $3
|
} in map (\rhs -> BNFCRule cat rhs (CFObj (mkFun cat rhs) [])) $3
|
||||||
}
|
}
|
||||||
| 'coercions' Ident Integer ';' { [BNFCCoercions (showIdent $2) $3]}
|
| 'coercions' Ident Integer ';' { [BNFCCoercions (showIdent $2) $3]}
|
||||||
| 'terminator' NonEmpty Ident String ';' { [BNFCTerminator $2 (showIdent $3) $4] }
|
| 'terminator' NonEmpty Ident String ';' { [BNFCTerminator $2 (showIdent $3) $4] }
|
||||||
@@ -774,7 +774,7 @@ checkInfoType mt jment@(id,info) =
|
|||||||
CncCat pty pd pr ppn _->ifConcrete mt (locPerh pty ++ locPerh pd ++ locPerh pr ++ locPerh ppn)
|
CncCat pty pd pr ppn _->ifConcrete mt (locPerh pty ++ locPerh pd ++ locPerh pr ++ locPerh ppn)
|
||||||
CncFun _ pd ppn _ -> ifConcrete mt (locPerh pd ++ locPerh ppn)
|
CncFun _ pd ppn _ -> ifConcrete mt (locPerh pd ++ locPerh ppn)
|
||||||
ResParam pparam _ -> ifResource mt (locPerh pparam)
|
ResParam pparam _ -> ifResource mt (locPerh pparam)
|
||||||
ResValue ty -> ifResource mt (locL ty)
|
ResValue ty _ -> ifResource mt (locL ty)
|
||||||
ResOper pty pt -> ifOper mt pty pt
|
ResOper pty pt -> ifOper mt pty pt
|
||||||
ResOverload _ xs -> ifResource mt (concat [[loc1,loc2] | (L loc1 _,L loc2 _) <- xs])
|
ResOverload _ xs -> ifResource mt (concat [[loc1,loc2] | (L loc1 _,L loc2 _) <- xs])
|
||||||
where
|
where
|
||||||
|
|||||||
@@ -22,21 +22,17 @@ module GF.Grammar.Printer
|
|||||||
, ppMeta
|
, ppMeta
|
||||||
, getAbs
|
, getAbs
|
||||||
) where
|
) where
|
||||||
import Prelude hiding ((<>)) -- GHC 8.4.1 clash with Text.PrettyPrint
|
|
||||||
|
|
||||||
|
import PGF2 as PGF2
|
||||||
|
import PGF2.Internal as PGF2
|
||||||
import GF.Infra.Ident
|
import GF.Infra.Ident
|
||||||
import GF.Infra.Option
|
import GF.Infra.Option
|
||||||
import GF.Grammar.Values
|
import GF.Grammar.Values
|
||||||
import GF.Grammar.Grammar
|
import GF.Grammar.Grammar
|
||||||
|
|
||||||
import PGF.Internal (ppMeta, ppLit, ppFId, ppFunId, ppSeqId, ppSeq)
|
|
||||||
|
|
||||||
import GF.Text.Pretty
|
import GF.Text.Pretty
|
||||||
import Data.Maybe (isNothing)
|
import Data.Maybe (isNothing)
|
||||||
import Data.List (intersperse)
|
import Data.List (intersperse)
|
||||||
import qualified Data.Map as Map
|
import qualified Data.Map as Map
|
||||||
--import qualified Data.IntMap as IntMap
|
|
||||||
--import qualified Data.Set as Set
|
|
||||||
import qualified Data.Array.IArray as Array
|
import qualified Data.Array.IArray as Array
|
||||||
|
|
||||||
data TermPrintQual
|
data TermPrintQual
|
||||||
@@ -110,8 +106,8 @@ ppJudgement q (id, ResParam pparams _) =
|
|||||||
(case pparams of
|
(case pparams of
|
||||||
Just (L _ ps) -> '=' <+> ppParams q ps
|
Just (L _ ps) -> '=' <+> ppParams q ps
|
||||||
_ -> empty) <+> ';'
|
_ -> empty) <+> ';'
|
||||||
ppJudgement q (id, ResValue pvalue) =
|
ppJudgement q (id, ResValue pvalue i) =
|
||||||
"-- param constructor" <+> id <+> ':' <+>
|
"-- param constructor" <+> "[index" <+> i <> "]" <+> id <+> ':' <+>
|
||||||
(case pvalue of
|
(case pvalue of
|
||||||
(L _ ty) -> ppTerm q 0 ty) <+> ';'
|
(L _ ty) -> ppTerm q 0 ty) <+> ';'
|
||||||
ppJudgement q (id, ResOper ptype pexp) =
|
ppJudgement q (id, ResOper ptype pexp) =
|
||||||
@@ -326,7 +322,7 @@ ppBind (Implicit,v) = braces v
|
|||||||
ppAltern q (x,y) = ppTerm q 0 x <+> '/' <+> ppTerm q 0 y
|
ppAltern q (x,y) = ppTerm q 0 x <+> '/' <+> ppTerm q 0 y
|
||||||
|
|
||||||
ppParams q ps = fsep (intersperse (pp '|') (map (ppParam q) ps))
|
ppParams q ps = fsep (intersperse (pp '|') (map (ppParam q) ps))
|
||||||
ppParam q (id,cxt) = id <+> hsep (map (ppDDecl q) cxt)
|
ppParam q (id,cxt,_) = id <+> hsep (map (ppDDecl q) cxt)
|
||||||
|
|
||||||
ppProduction (Production fid funid args) =
|
ppProduction (Production fid funid args) =
|
||||||
ppFId fid <+> "->" <+> ppFunId funid <>
|
ppFId fid <+> "->" <+> ppFunId funid <>
|
||||||
@@ -363,3 +359,39 @@ getLet (Let l e) = let (ls,e') = getLet e
|
|||||||
in (l:ls,e')
|
in (l:ls,e')
|
||||||
getLet e = ([],e)
|
getLet e = ([],e)
|
||||||
|
|
||||||
|
ppFunId funid = pp 'F' <> pp funid
|
||||||
|
ppSeqId seqid = pp 'S' <> pp seqid
|
||||||
|
|
||||||
|
ppFId fid
|
||||||
|
| fid == PGF2.fidString = pp "CString"
|
||||||
|
| fid == PGF2.fidInt = pp "CInt"
|
||||||
|
| fid == PGF2.fidFloat = pp "CFloat"
|
||||||
|
| fid == PGF2.fidVar = pp "CVar"
|
||||||
|
| fid == PGF2.fidStart = pp "CStart"
|
||||||
|
| otherwise = pp 'C' <> pp fid
|
||||||
|
|
||||||
|
ppMeta :: Int -> Doc
|
||||||
|
ppMeta n
|
||||||
|
| n == 0 = pp '?'
|
||||||
|
| otherwise = pp '?' <> pp n
|
||||||
|
|
||||||
|
ppLit (PGF2.LStr s) = pp (show s)
|
||||||
|
ppLit (PGF2.LInt n) = pp n
|
||||||
|
ppLit (PGF2.LFlt d) = pp d
|
||||||
|
|
||||||
|
ppSeq (seqid,seq) =
|
||||||
|
ppSeqId seqid <+> pp ":=" <+> hsep (map ppSymbol seq)
|
||||||
|
|
||||||
|
ppSymbol (PGF2.SymCat d r) = pp '<' <> pp d <> pp ',' <> pp r <> pp '>'
|
||||||
|
ppSymbol (PGF2.SymLit d r) = pp '{' <> pp d <> pp ',' <> pp r <> pp '}'
|
||||||
|
ppSymbol (PGF2.SymVar d r) = pp '<' <> pp d <> pp ',' <> pp '$' <> pp r <> pp '>'
|
||||||
|
ppSymbol (PGF2.SymKS t) = doubleQuotes (pp t)
|
||||||
|
ppSymbol PGF2.SymNE = pp "nonExist"
|
||||||
|
ppSymbol PGF2.SymBIND = pp "BIND"
|
||||||
|
ppSymbol PGF2.SymSOFT_BIND = pp "SOFT_BIND"
|
||||||
|
ppSymbol PGF2.SymSOFT_SPACE= pp "SOFT_SPACE"
|
||||||
|
ppSymbol PGF2.SymCAPIT = pp "CAPIT"
|
||||||
|
ppSymbol PGF2.SymALL_CAPIT = pp "ALL_CAPIT"
|
||||||
|
ppSymbol (PGF2.SymKP syms alts) = pp "pre" <+> braces (hsep (punctuate (pp ';') (hsep (map ppSymbol syms) : map ppAlt alts)))
|
||||||
|
|
||||||
|
ppAlt (syms,ps) = hsep (map ppSymbol syms) <+> pp '/' <+> hsep (map (doubleQuotes . pp) ps)
|
||||||
|
|||||||
@@ -14,9 +14,3 @@ buildInfo =
|
|||||||
#ifdef SERVER_MODE
|
#ifdef SERVER_MODE
|
||||||
++" server"
|
++" server"
|
||||||
#endif
|
#endif
|
||||||
#ifdef NEW_COMP
|
|
||||||
++" new-comp"
|
|
||||||
#endif
|
|
||||||
#ifdef C_RUNTIME
|
|
||||||
++" c-runtime"
|
|
||||||
#endif
|
|
||||||
|
|||||||
@@ -18,7 +18,6 @@ module GF.Infra.CheckM
|
|||||||
checkIn, checkInModule, checkMap, checkMapRecover,
|
checkIn, checkInModule, checkMap, checkMapRecover,
|
||||||
parallelCheck, accumulateError, commitCheck,
|
parallelCheck, accumulateError, commitCheck,
|
||||||
) where
|
) where
|
||||||
import Prelude hiding ((<>)) -- GHC 8.4.1 clash with Text.PrettyPrint
|
|
||||||
|
|
||||||
import GF.Data.Operations
|
import GF.Data.Operations
|
||||||
--import GF.Infra.Ident
|
--import GF.Infra.Ident
|
||||||
|
|||||||
@@ -13,17 +13,17 @@
|
|||||||
-----------------------------------------------------------------------------
|
-----------------------------------------------------------------------------
|
||||||
|
|
||||||
module GF.Infra.Ident (-- ** Identifiers
|
module GF.Infra.Ident (-- ** Identifiers
|
||||||
ModuleName(..), moduleNameS,
|
ModuleName(..), moduleNameS,
|
||||||
Ident, ident2utf8, showIdent, prefixIdent,
|
Ident, ident2utf8, showIdent, prefixIdent,
|
||||||
-- *** Normal identifiers (returned by the parser)
|
-- *** Normal identifiers (returned by the parser)
|
||||||
identS, identC, identW,
|
identS, identC, identW,
|
||||||
-- *** Special identifiers for internal use
|
-- *** Special identifiers for internal use
|
||||||
identV, identA, identAV,
|
identV, identA, identAV,
|
||||||
argIdent, isArgIdent, getArgIndex,
|
argIdent, isArgIdent, getArgIndex,
|
||||||
varStr, varX, isWildIdent, varIndex,
|
varStr, varX, isWildIdent, varIndex,
|
||||||
-- *** Raw identifiers
|
-- *** Raw identifiers
|
||||||
RawIdent, rawIdentS, rawIdentC, ident2raw, prefixRawIdent,
|
RawIdent, rawIdentS, rawIdentC, ident2raw, prefixRawIdent,
|
||||||
isPrefixOf, showRawIdent
|
isPrefixOf, showRawIdent
|
||||||
) where
|
) where
|
||||||
|
|
||||||
import qualified Data.ByteString.UTF8 as UTF8
|
import qualified Data.ByteString.UTF8 as UTF8
|
||||||
@@ -31,7 +31,7 @@ import qualified Data.ByteString.Char8 as BS(append,isPrefixOf)
|
|||||||
-- Limit use of BS functions to the ones that work correctly on
|
-- Limit use of BS functions to the ones that work correctly on
|
||||||
-- UTF-8-encoded bytestrings!
|
-- UTF-8-encoded bytestrings!
|
||||||
import Data.Char(isDigit)
|
import Data.Char(isDigit)
|
||||||
import PGF.Internal(Binary(..))
|
import Data.Binary(Binary(..))
|
||||||
import GF.Text.Pretty
|
import GF.Text.Pretty
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
-- | Source locations
|
-- | Source locations
|
||||||
module GF.Infra.Location where
|
module GF.Infra.Location where
|
||||||
import Prelude hiding ((<>)) -- GHC 8.4.1 clash with Text.PrettyPrint
|
|
||||||
import GF.Text.Pretty
|
import GF.Text.Pretty
|
||||||
|
|
||||||
-- ** Source locations
|
-- ** Source locations
|
||||||
|
|||||||
@@ -34,17 +34,14 @@ import Data.Maybe
|
|||||||
import GF.Infra.Ident
|
import GF.Infra.Ident
|
||||||
import GF.Infra.GetOpt
|
import GF.Infra.GetOpt
|
||||||
import GF.Grammar.Predef
|
import GF.Grammar.Predef
|
||||||
--import System.Console.GetOpt
|
|
||||||
import System.FilePath
|
import System.FilePath
|
||||||
--import System.IO
|
import PGF2.Internal(Literal(..))
|
||||||
|
|
||||||
import GF.Data.Operations(Err,ErrorMonad(..),liftErr)
|
import GF.Data.Operations(Err,ErrorMonad(..),liftErr)
|
||||||
|
|
||||||
import Data.Set (Set)
|
import Data.Set (Set)
|
||||||
import qualified Data.Set as Set
|
import qualified Data.Set as Set
|
||||||
|
|
||||||
import PGF.Internal(Literal(..))
|
|
||||||
|
|
||||||
usageHeader :: String
|
usageHeader :: String
|
||||||
usageHeader = unlines
|
usageHeader = unlines
|
||||||
["Usage: gf [OPTIONS] [FILE [...]]",
|
["Usage: gf [OPTIONS] [FILE [...]]",
|
||||||
@@ -75,7 +72,6 @@ errors = raise . unlines
|
|||||||
|
|
||||||
data Mode = ModeVersion | ModeHelp
|
data Mode = ModeVersion | ModeHelp
|
||||||
| ModeInteractive | ModeRun
|
| ModeInteractive | ModeRun
|
||||||
| ModeInteractive2 | ModeRun2
|
|
||||||
| ModeCompiler
|
| ModeCompiler
|
||||||
| ModeServer {-port::-}Int
|
| ModeServer {-port::-}Int
|
||||||
deriving (Show,Eq,Ord)
|
deriving (Show,Eq,Ord)
|
||||||
@@ -89,12 +85,9 @@ data Phase = Preproc | Convert | Compile | Link
|
|||||||
data OutputFormat = FmtPGFPretty
|
data OutputFormat = FmtPGFPretty
|
||||||
| FmtCanonicalGF
|
| FmtCanonicalGF
|
||||||
| FmtCanonicalJson
|
| FmtCanonicalJson
|
||||||
| FmtJavaScript
|
|
||||||
| FmtJSON
|
| FmtJSON
|
||||||
| FmtPython
|
|
||||||
| FmtHaskell
|
| FmtHaskell
|
||||||
| FmtJava
|
| FmtJava
|
||||||
| FmtProlog
|
|
||||||
| FmtBNF
|
| FmtBNF
|
||||||
| FmtEBNF
|
| FmtEBNF
|
||||||
| FmtRegular
|
| FmtRegular
|
||||||
@@ -156,7 +149,7 @@ data Flags = Flags {
|
|||||||
optLiteralCats :: Set Ident,
|
optLiteralCats :: Set Ident,
|
||||||
optGFODir :: Maybe FilePath,
|
optGFODir :: Maybe FilePath,
|
||||||
optOutputDir :: Maybe FilePath,
|
optOutputDir :: Maybe FilePath,
|
||||||
optGFLibPath :: Maybe [FilePath],
|
optGFLibPath :: Maybe FilePath,
|
||||||
optDocumentRoot :: Maybe FilePath, -- For --server mode
|
optDocumentRoot :: Maybe FilePath, -- For --server mode
|
||||||
optRecomp :: Recomp,
|
optRecomp :: Recomp,
|
||||||
optProbsFile :: Maybe FilePath,
|
optProbsFile :: Maybe FilePath,
|
||||||
@@ -211,10 +204,9 @@ parseModuleOptions args = do
|
|||||||
then return opts
|
then return opts
|
||||||
else errors $ map ("Non-option among module options: " ++) nonopts
|
else errors $ map ("Non-option among module options: " ++) nonopts
|
||||||
|
|
||||||
fixRelativeLibPaths curr_dir lib_dirs (Options o) = Options (fixPathFlags . o)
|
fixRelativeLibPaths curr_dir lib_dir (Options o) = Options (fixPathFlags . o)
|
||||||
where
|
where
|
||||||
fixPathFlags f@(Flags{optLibraryPath=path}) = f{optLibraryPath=concatMap (\dir -> [parent </> dir
|
fixPathFlags f@(Flags{optLibraryPath=path}) = f{optLibraryPath=concatMap (\dir -> [curr_dir </> dir, lib_dir </> dir]) path}
|
||||||
| parent <- curr_dir : lib_dirs]) path}
|
|
||||||
|
|
||||||
-- Showing options
|
-- Showing options
|
||||||
|
|
||||||
@@ -310,8 +302,6 @@ optDescr =
|
|||||||
Option ['j'] ["jobs"] (OptArg jobs "N") "Compile N modules in parallel with -batch (default 1).",
|
Option ['j'] ["jobs"] (OptArg jobs "N") "Compile N modules in parallel with -batch (default 1).",
|
||||||
Option [] ["interactive"] (NoArg (mode ModeInteractive)) "Run in interactive mode (default).",
|
Option [] ["interactive"] (NoArg (mode ModeInteractive)) "Run in interactive mode (default).",
|
||||||
Option [] ["run"] (NoArg (mode ModeRun)) "Run in interactive mode, showing output only (no other messages).",
|
Option [] ["run"] (NoArg (mode ModeRun)) "Run in interactive mode, showing output only (no other messages).",
|
||||||
Option [] ["cshell"] (NoArg (mode ModeInteractive2)) "Start the C run-time shell.",
|
|
||||||
Option [] ["crun"] (NoArg (mode ModeRun2)) "Start the C run-time shell, showing output only (no other messages).",
|
|
||||||
Option [] ["server"] (OptArg modeServer "port") $
|
Option [] ["server"] (OptArg modeServer "port") $
|
||||||
"Run in HTTP server mode on given port (default "++show defaultPort++").",
|
"Run in HTTP server mode on given port (default "++show defaultPort++").",
|
||||||
Option [] ["document-root"] (ReqArg gfDocuRoot "DIR")
|
Option [] ["document-root"] (ReqArg gfDocuRoot "DIR")
|
||||||
@@ -426,7 +416,7 @@ optDescr =
|
|||||||
literalCat x = set $ \o -> o { optLiteralCats = foldr Set.insert (optLiteralCats o) ((map identS . splitBy (==',')) x) }
|
literalCat x = set $ \o -> o { optLiteralCats = foldr Set.insert (optLiteralCats o) ((map identS . splitBy (==',')) x) }
|
||||||
lexicalCat x = set $ \o -> o { optLexicalCats = foldr Set.insert (optLexicalCats o) (splitBy (==',') x) }
|
lexicalCat x = set $ \o -> o { optLexicalCats = foldr Set.insert (optLexicalCats o) (splitBy (==',') x) }
|
||||||
outDir x = set $ \o -> o { optOutputDir = Just x }
|
outDir x = set $ \o -> o { optOutputDir = Just x }
|
||||||
gfLibPath x = set $ \o -> o { optGFLibPath = Just $ splitInModuleSearchPath x }
|
gfLibPath x = set $ \o -> o { optGFLibPath = Just x }
|
||||||
gfDocuRoot x = set $ \o -> o { optDocumentRoot = Just x }
|
gfDocuRoot x = set $ \o -> o { optDocumentRoot = Just x }
|
||||||
recomp x = set $ \o -> o { optRecomp = x }
|
recomp x = set $ \o -> o { optRecomp = x }
|
||||||
probsFile x = set $ \o -> o { optProbsFile = Just x }
|
probsFile x = set $ \o -> o { optProbsFile = Just x }
|
||||||
@@ -474,12 +464,9 @@ outputFormatsExpl =
|
|||||||
[(("pgf_pretty", FmtPGFPretty),"human-readable pgf"),
|
[(("pgf_pretty", FmtPGFPretty),"human-readable pgf"),
|
||||||
(("canonical_gf", FmtCanonicalGF),"Canonical GF source files"),
|
(("canonical_gf", FmtCanonicalGF),"Canonical GF source files"),
|
||||||
(("canonical_json", FmtCanonicalJson),"Canonical JSON source files"),
|
(("canonical_json", FmtCanonicalJson),"Canonical JSON source files"),
|
||||||
(("js", FmtJavaScript),"JavaScript (whole grammar)"),
|
|
||||||
(("json", FmtJSON),"JSON (whole grammar)"),
|
(("json", FmtJSON),"JSON (whole grammar)"),
|
||||||
(("python", FmtPython),"Python (whole grammar)"),
|
|
||||||
(("haskell", FmtHaskell),"Haskell (abstract syntax)"),
|
(("haskell", FmtHaskell),"Haskell (abstract syntax)"),
|
||||||
(("java", FmtJava),"Java (abstract syntax)"),
|
(("java", FmtJava),"Java (abstract syntax)"),
|
||||||
(("prolog", FmtProlog),"Prolog (whole grammar)"),
|
|
||||||
(("bnf", FmtBNF),"BNF (context-free grammar)"),
|
(("bnf", FmtBNF),"BNF (context-free grammar)"),
|
||||||
(("ebnf", FmtEBNF),"Extended BNF"),
|
(("ebnf", FmtEBNF),"Extended BNF"),
|
||||||
(("regular", FmtRegular),"* regular grammar"),
|
(("regular", FmtRegular),"* regular grammar"),
|
||||||
|
|||||||
@@ -12,9 +12,6 @@ module GF.Infra.SIO(
|
|||||||
newStdGen,print,putStr,putStrLn,
|
newStdGen,print,putStr,putStrLn,
|
||||||
-- ** Specific to GF
|
-- ** Specific to GF
|
||||||
importGrammar,importSource,
|
importGrammar,importSource,
|
||||||
#ifdef C_RUNTIME
|
|
||||||
readPGF2,
|
|
||||||
#endif
|
|
||||||
putStrLnFlush,runInterruptibly,lazySIO,
|
putStrLnFlush,runInterruptibly,lazySIO,
|
||||||
-- * Restricted accesss to arbitrary (potentially unsafe) IO operations
|
-- * Restricted accesss to arbitrary (potentially unsafe) IO operations
|
||||||
-- | If the environment variable GF_RESTRICTED is defined, these
|
-- | If the environment variable GF_RESTRICTED is defined, these
|
||||||
@@ -39,9 +36,6 @@ import qualified System.Random as IO(newStdGen)
|
|||||||
import qualified GF.Infra.UseIO as IO(getLibraryDirectory)
|
import qualified GF.Infra.UseIO as IO(getLibraryDirectory)
|
||||||
import qualified GF.System.Signal as IO(runInterruptibly)
|
import qualified GF.System.Signal as IO(runInterruptibly)
|
||||||
import qualified GF.Command.Importing as GF(importGrammar, importSource)
|
import qualified GF.Command.Importing as GF(importGrammar, importSource)
|
||||||
#ifdef C_RUNTIME
|
|
||||||
import qualified PGF2
|
|
||||||
#endif
|
|
||||||
|
|
||||||
-- * The SIO monad
|
-- * The SIO monad
|
||||||
|
|
||||||
@@ -123,7 +117,3 @@ lazySIO = lift1 lazyIO
|
|||||||
|
|
||||||
importGrammar pgf opts files = lift0 $ GF.importGrammar pgf opts files
|
importGrammar pgf opts files = lift0 $ GF.importGrammar pgf opts files
|
||||||
importSource opts files = lift0 $ GF.importSource opts files
|
importSource opts files = lift0 $ GF.importSource opts files
|
||||||
|
|
||||||
#ifdef C_RUNTIME
|
|
||||||
readPGF2 = lift0 . PGF2.readPGF
|
|
||||||
#endif
|
|
||||||
|
|||||||
@@ -38,7 +38,6 @@ import Control.Monad(when,liftM,foldM)
|
|||||||
import Control.Monad.Trans(MonadIO(..))
|
import Control.Monad.Trans(MonadIO(..))
|
||||||
import Control.Monad.State(StateT,lift)
|
import Control.Monad.State(StateT,lift)
|
||||||
import Control.Exception(evaluate)
|
import Control.Exception(evaluate)
|
||||||
import Data.List (nub)
|
|
||||||
|
|
||||||
--putIfVerb :: MonadIO io => Options -> String -> io ()
|
--putIfVerb :: MonadIO io => Options -> String -> io ()
|
||||||
putIfVerb opts msg = when (verbAtLeast opts Verbose) $ putStrLnE msg
|
putIfVerb opts msg = when (verbAtLeast opts Verbose) $ putStrLnE msg
|
||||||
@@ -52,32 +51,28 @@ type FullPath = String
|
|||||||
gfLibraryPath = "GF_LIB_PATH"
|
gfLibraryPath = "GF_LIB_PATH"
|
||||||
gfGrammarPathVar = "GF_GRAMMAR_PATH"
|
gfGrammarPathVar = "GF_GRAMMAR_PATH"
|
||||||
|
|
||||||
getLibraryDirectory :: MonadIO io => Options -> io [FilePath]
|
getLibraryDirectory :: MonadIO io => Options -> io FilePath
|
||||||
getLibraryDirectory opts =
|
getLibraryDirectory opts =
|
||||||
case flag optGFLibPath opts of
|
case flag optGFLibPath opts of
|
||||||
Just path -> return path
|
Just path -> return path
|
||||||
Nothing -> liftM splitSearchPath $ liftIO (catch (getEnv gfLibraryPath)
|
Nothing -> liftIO $ catch (getEnv gfLibraryPath)
|
||||||
(\ex -> fmap (</> "lib") getDataDir))
|
(\ex -> fmap (</> "lib") getDataDir)
|
||||||
|
|
||||||
getGrammarPath :: MonadIO io => [FilePath] -> io [FilePath]
|
getGrammarPath :: MonadIO io => FilePath -> io [FilePath]
|
||||||
getGrammarPath lib_dirs = liftIO $ do
|
getGrammarPath lib_dir = liftIO $ do
|
||||||
catch (fmap splitSearchPath $ getEnv gfGrammarPathVar)
|
catch (fmap splitSearchPath $ getEnv gfGrammarPathVar)
|
||||||
(\_ -> return $ concat [[lib_dir </> "alltenses", lib_dir </> "prelude"]
|
(\_ -> return [lib_dir </> "alltenses",lib_dir </> "prelude"]) -- e.g. GF_GRAMMAR_PATH
|
||||||
| lib_dir <- lib_dirs ]) -- e.g. GF_GRAMMAR_PATH
|
|
||||||
|
|
||||||
-- | extends the search path with the
|
-- | extends the search path with the
|
||||||
-- 'gfLibraryPath' and 'gfGrammarPathVar'
|
-- 'gfLibraryPath' and 'gfGrammarPathVar'
|
||||||
-- environment variables. Returns only existing paths.
|
-- environment variables. Returns only existing paths.
|
||||||
extendPathEnv :: MonadIO io => Options -> io [FilePath]
|
extendPathEnv :: MonadIO io => Options -> io [FilePath]
|
||||||
extendPathEnv opts = liftIO $ do
|
extendPathEnv opts = liftIO $ do
|
||||||
let opt_path = nub $ flag optLibraryPath opts -- e.g. paths given as options
|
let opt_path = flag optLibraryPath opts -- e.g. paths given as options
|
||||||
lib_dirs <- getLibraryDirectory opts -- e.g. GF_LIB_PATH
|
lib_dir <- getLibraryDirectory opts -- e.g. GF_LIB_PATH
|
||||||
grm_path <- getGrammarPath lib_dirs -- e.g. GF_GRAMMAR_PATH
|
grm_path <- getGrammarPath lib_dir -- e.g. GF_GRAMMAR_PATH
|
||||||
let paths = opt_path ++ lib_dirs ++ grm_path
|
let paths = opt_path ++ [lib_dir] ++ grm_path
|
||||||
when (verbAtLeast opts Verbose) $ putStrLn ("extendPathEnv: opt_path is "++ show opt_path)
|
ps <- liftM concat $ mapM allSubdirs paths
|
||||||
when (verbAtLeast opts Verbose) $ putStrLn ("extendPathEnv: lib_dirs is "++ show lib_dirs)
|
|
||||||
when (verbAtLeast opts Verbose) $ putStrLn ("extendPathEnv: grm_path is "++ show grm_path)
|
|
||||||
ps <- liftM (nub . concat) $ mapM allSubdirs (nub paths)
|
|
||||||
mapM canonicalizePath ps
|
mapM canonicalizePath ps
|
||||||
where
|
where
|
||||||
allSubdirs :: FilePath -> IO [FilePath]
|
allSubdirs :: FilePath -> IO [FilePath]
|
||||||
@@ -85,15 +80,11 @@ extendPathEnv opts = liftIO $ do
|
|||||||
allSubdirs p = case last p of
|
allSubdirs p = case last p of
|
||||||
'*' -> do let path = init p
|
'*' -> do let path = init p
|
||||||
fs <- getSubdirs path
|
fs <- getSubdirs path
|
||||||
let starpaths = [path </> f | f <- fs]
|
return [path </> f | f <- fs]
|
||||||
when (verbAtLeast opts Verbose) $ putStrLn ("extendPathEnv: allSubdirs: * found "++show starpaths)
|
|
||||||
return starpaths
|
|
||||||
_ -> do exists <- doesDirectoryExist p
|
_ -> do exists <- doesDirectoryExist p
|
||||||
if exists
|
if exists
|
||||||
then do
|
then return [p]
|
||||||
when (verbAtLeast opts Verbose) $ putStrLn ("extendPathEnv: allSubdirs: found path "++show p)
|
else do when (verbAtLeast opts Verbose) $ putStrLn ("ignore path "++p)
|
||||||
return [p]
|
|
||||||
else do when (verbAtLeast opts Verbose) $ putStrLn ("extendPathEnv: allSubdirs: ignore path "++ show p)
|
|
||||||
return []
|
return []
|
||||||
|
|
||||||
getSubdirs :: FilePath -> IO [FilePath]
|
getSubdirs :: FilePath -> IO [FilePath]
|
||||||
|
|||||||
@@ -1,11 +1,11 @@
|
|||||||
{-# LANGUAGE CPP, ScopedTypeVariables, FlexibleInstances #-}
|
{-# LANGUAGE CPP, ScopedTypeVariables, FlexibleInstances #-}
|
||||||
-- | GF interactive mode
|
-- | GF interactive mode
|
||||||
module GF.Interactive (mainGFI,mainRunGFI,mainServerGFI) where
|
module GF.Interactive (mainGFI,mainRunGFI,mainServerGFI) where
|
||||||
|
|
||||||
import Prelude hiding (putStrLn,print)
|
import Prelude hiding (putStrLn,print)
|
||||||
import qualified Prelude as P(putStrLn)
|
import qualified Prelude as P(putStrLn)
|
||||||
import GF.Command.Interpreter(CommandEnv(..),mkCommandEnv,interpretCommandLine)
|
import GF.Command.Interpreter(CommandEnv(..),mkCommandEnv,interpretCommandLine)
|
||||||
--import GF.Command.Importing(importSource,importGrammar)
|
import GF.Command.Commands(HasPGF(..),pgfCommands)
|
||||||
import GF.Command.Commands(PGFEnv,HasPGFEnv(..),pgf,pgfEnv,pgfCommands)
|
|
||||||
import GF.Command.CommonCommands(commonCommands,extend)
|
import GF.Command.CommonCommands(commonCommands,extend)
|
||||||
import GF.Command.SourceCommands
|
import GF.Command.SourceCommands
|
||||||
import GF.Command.CommandInfo
|
import GF.Command.CommandInfo
|
||||||
@@ -19,19 +19,13 @@ import GF.Infra.UseIO(ioErrorText,putStrLnE)
|
|||||||
import GF.Infra.SIO
|
import GF.Infra.SIO
|
||||||
import GF.Infra.Option
|
import GF.Infra.Option
|
||||||
import qualified System.Console.Haskeline as Haskeline
|
import qualified System.Console.Haskeline as Haskeline
|
||||||
--import GF.Text.Coding(decodeUnicode,encodeUnicode)
|
|
||||||
|
|
||||||
--import GF.Compile.Coding(codeTerm)
|
import PGF2
|
||||||
|
|
||||||
import PGF
|
|
||||||
import PGF.Internal(abstract,funs,lookStartCat,emptyPGF)
|
|
||||||
|
|
||||||
import Data.Char
|
import Data.Char
|
||||||
import Data.List(isPrefixOf)
|
import Data.List(isPrefixOf)
|
||||||
import qualified Data.Map as Map
|
import qualified Data.Map as Map
|
||||||
import qualified Text.ParserCombinators.ReadP as RP
|
import qualified Text.ParserCombinators.ReadP as RP
|
||||||
--import System.IO(utf8)
|
|
||||||
--import System.CPUTime(getCPUTime)
|
|
||||||
import System.Directory({-getCurrentDirectory,-}getAppUserDataDirectory)
|
import System.Directory({-getCurrentDirectory,-}getAppUserDataDirectory)
|
||||||
import Control.Exception(SomeException,fromException,evaluate,try)
|
import Control.Exception(SomeException,fromException,evaluate,try)
|
||||||
import Control.Monad.State hiding (void)
|
import Control.Monad.State hiding (void)
|
||||||
@@ -280,17 +274,18 @@ importInEnv opts files =
|
|||||||
if flag optRetainResource opts
|
if flag optRetainResource opts
|
||||||
then do src <- lift $ importSource opts files
|
then do src <- lift $ importSource opts files
|
||||||
pgf <- lift . lazySIO $ importPGF pgf0 -- duplicates some work, better to link src
|
pgf <- lift . lazySIO $ importPGF pgf0 -- duplicates some work, better to link src
|
||||||
modify $ \ gfenv -> gfenv {retain=True, pgfenv = (src,pgfEnv pgf)}
|
modify $ \ gfenv -> gfenv {retain=True, pgfenv = (src,pgf)}
|
||||||
else do pgf1 <- lift $ importPGF pgf0
|
else do pgf1 <- lift $ importPGF pgf0
|
||||||
modify $ \ gfenv->gfenv { retain=False,
|
modify $ \ gfenv->gfenv { retain=False,
|
||||||
pgfenv = (emptyGrammar,pgfEnv pgf1) }
|
pgfenv = (emptyGrammar,pgf1) }
|
||||||
where
|
where
|
||||||
importPGF pgf0 =
|
importPGF pgf0 =
|
||||||
do let opts' = addOptions (setOptimization OptCSE False) opts
|
do let opts' = addOptions (setOptimization OptCSE False) opts
|
||||||
pgf1 <- importGrammar pgf0 opts' files
|
pgf1 <- importGrammar pgf0 opts' files
|
||||||
if (verbAtLeast opts Normal)
|
if (verbAtLeast opts Normal)
|
||||||
then putStrLnFlush $
|
then case pgf1 of
|
||||||
unwords $ "\nLanguages:" : map showCId (languages pgf1)
|
Just pgf -> putStrLnFlush $ unwords $ "\nLanguages:" : Map.keys (languages pgf)
|
||||||
|
Nothing -> done
|
||||||
else done
|
else done
|
||||||
return pgf1
|
return pgf1
|
||||||
|
|
||||||
@@ -301,12 +296,12 @@ tryGetLine = do
|
|||||||
Right l -> return l
|
Right l -> return l
|
||||||
|
|
||||||
prompt env
|
prompt env
|
||||||
| retain env || abs == wildCId = "> "
|
| retain env = "> "
|
||||||
| otherwise = showCId abs ++ "> "
|
| otherwise = case multigrammar env of
|
||||||
where
|
Just pgf -> abstractName pgf ++ "> "
|
||||||
abs = abstractName (multigrammar env)
|
Nothing -> "> "
|
||||||
|
|
||||||
type CmdEnv = (Grammar,PGFEnv)
|
type CmdEnv = (Grammar,Maybe PGF)
|
||||||
|
|
||||||
data GFEnv = GFEnv {
|
data GFEnv = GFEnv {
|
||||||
startOpts :: Options,
|
startOpts :: Options,
|
||||||
@@ -318,10 +313,10 @@ data GFEnv = GFEnv {
|
|||||||
|
|
||||||
emptyGFEnv opts = GFEnv opts False emptyCmdEnv emptyCommandEnv []
|
emptyGFEnv opts = GFEnv opts False emptyCmdEnv emptyCommandEnv []
|
||||||
|
|
||||||
emptyCmdEnv = (emptyGrammar,pgfEnv emptyPGF)
|
emptyCmdEnv = (emptyGrammar,Nothing)
|
||||||
|
|
||||||
emptyCommandEnv = mkCommandEnv allCommands
|
emptyCommandEnv = mkCommandEnv allCommands
|
||||||
multigrammar = pgf . snd . pgfenv
|
multigrammar = snd . pgfenv
|
||||||
|
|
||||||
allCommands =
|
allCommands =
|
||||||
extend pgfCommands (helpCommand allCommands:moreCommands)
|
extend pgfCommands (helpCommand allCommands:moreCommands)
|
||||||
@@ -329,24 +324,35 @@ allCommands =
|
|||||||
`Map.union` commonCommands
|
`Map.union` commonCommands
|
||||||
|
|
||||||
instance HasGrammar ShellM where getGrammar = gets (fst . pgfenv)
|
instance HasGrammar ShellM where getGrammar = gets (fst . pgfenv)
|
||||||
instance HasPGFEnv ShellM where getPGFEnv = gets (snd . pgfenv)
|
instance HasPGF ShellM where getPGF = gets (snd . pgfenv)
|
||||||
|
|
||||||
wordCompletion gfenv (left,right) = do
|
wordCompletion gfenv (left,right) = do
|
||||||
case wc_type (reverse left) of
|
case wc_type (reverse left) of
|
||||||
CmplCmd pref
|
CmplCmd pref
|
||||||
-> ret (length pref) [Haskeline.simpleCompletion name | name <- Map.keys (commands cmdEnv), isPrefixOf pref name]
|
-> ret (length pref) [Haskeline.simpleCompletion name | name <- Map.keys (commands cmdEnv), isPrefixOf pref name]
|
||||||
CmplStr (Just (Command _ opts _)) s0
|
CmplStr (Just (Command _ opts _)) s0
|
||||||
-> do mb_state0 <- try (evaluate (initState pgf (optLang opts) (optType opts)))
|
-> case multigrammar gfenv of
|
||||||
case mb_state0 of
|
Just pgf -> let langs = languages pgf
|
||||||
Right state0 -> let (rprefix,rs) = break isSpace (reverse s0)
|
optLang opts = case valStrOpts "lang" "" opts of
|
||||||
s = reverse rs
|
"" -> case Map.minView langs of
|
||||||
prefix = reverse rprefix
|
Nothing -> Nothing
|
||||||
ws = words s
|
Just (concr,_) -> Just concr
|
||||||
in case loop state0 ws of
|
lang -> mplus (Map.lookup lang langs)
|
||||||
Nothing -> ret 0 []
|
(Map.lookup (abstractName pgf ++ lang) langs)
|
||||||
Just state -> let compls = getCompletions state prefix
|
optType opts = let readOpt str = case readType str of
|
||||||
in ret (length prefix) (map (\x -> Haskeline.simpleCompletion x) (Map.keys compls))
|
Just ty -> case checkType pgf ty of
|
||||||
Left (_ :: SomeException) -> ret 0 []
|
Left _ -> Nothing
|
||||||
|
Right ty -> Just ty
|
||||||
|
Nothing -> Nothing
|
||||||
|
in maybeStrOpts "cat" (Just (startCat pgf)) readOpt opts
|
||||||
|
(rprefix,rs) = break isSpace (reverse s0)
|
||||||
|
s = reverse rs
|
||||||
|
prefix = reverse rprefix
|
||||||
|
in case (optLang opts, optType opts) of
|
||||||
|
(Just lang,Just cat) -> let compls = [t | (t,_,_,_) <- complete lang cat s prefix]
|
||||||
|
in ret (length prefix) (map Haskeline.simpleCompletion compls)
|
||||||
|
_ -> ret 0 []
|
||||||
|
Nothing -> ret 0 []
|
||||||
CmplOpt (Just (Command n _ _)) pref
|
CmplOpt (Just (Command n _ _)) pref
|
||||||
-> case Map.lookup n (commands cmdEnv) of
|
-> case Map.lookup n (commands cmdEnv) of
|
||||||
Just inf -> do let flg_compls = [Haskeline.Completion ('-':flg++"=") ('-':flg) False | (flg,_) <- flags inf, isPrefixOf pref flg]
|
Just inf -> do let flg_compls = [Haskeline.Completion ('-':flg++"=") ('-':flg) False | (flg,_) <- flags inf, isPrefixOf pref flg]
|
||||||
@@ -357,23 +363,15 @@ wordCompletion gfenv (left,right) = do
|
|||||||
CmplIdent (Just (Command "i" _ _)) _ -- HACK: file name completion for command i
|
CmplIdent (Just (Command "i" _ _)) _ -- HACK: file name completion for command i
|
||||||
-> Haskeline.completeFilename (left,right)
|
-> Haskeline.completeFilename (left,right)
|
||||||
CmplIdent _ pref
|
CmplIdent _ pref
|
||||||
-> do mb_abs <- try (evaluate (abstract pgf))
|
-> case multigrammar gfenv of
|
||||||
case mb_abs of
|
Just pgf -> ret (length pref) [Haskeline.simpleCompletion name | name <- functions pgf, isPrefixOf pref name]
|
||||||
Right abs -> ret (length pref) [Haskeline.simpleCompletion name | cid <- Map.keys (funs abs), let name = showCId cid, isPrefixOf pref name]
|
Nothing -> ret (length pref) []
|
||||||
Left (_ :: SomeException) -> ret (length pref) []
|
|
||||||
_ -> ret 0 []
|
_ -> ret 0 []
|
||||||
where
|
where
|
||||||
pgf = multigrammar gfenv
|
|
||||||
cmdEnv = commandenv gfenv
|
cmdEnv = commandenv gfenv
|
||||||
optLang opts = valCIdOpts "lang" (head (languages pgf)) opts
|
|
||||||
optType opts =
|
|
||||||
let str = valStrOpts "cat" (showCId $ lookStartCat pgf) opts
|
|
||||||
in case readType str of
|
|
||||||
Just ty -> ty
|
|
||||||
Nothing -> error ("Can't parse '"++str++"' as type")
|
|
||||||
|
|
||||||
loop ps [] = Just ps
|
loop ps [] = Just ps
|
||||||
loop ps (t:ts) = case nextState ps (simpleParseInput t) of
|
loop ps (t:ts) = case error "nextState ps (simpleParseInput t)" of
|
||||||
Left es -> Nothing
|
Left es -> Nothing
|
||||||
Right ps -> loop ps ts
|
Right ps -> loop ps ts
|
||||||
|
|
||||||
|
|||||||
@@ -1,445 +0,0 @@
|
|||||||
{-# LANGUAGE CPP, ScopedTypeVariables, TypeSynonymInstances, FlexibleInstances, FlexibleContexts #-}
|
|
||||||
-- | GF interactive mode (with the C run-time system)
|
|
||||||
module GF.Interactive2 (mainGFI,mainRunGFI{-,mainServerGFI-}) where
|
|
||||||
import Prelude hiding (putStrLn,print)
|
|
||||||
import qualified Prelude as P(putStrLn)
|
|
||||||
import GF.Command.Interpreter(CommandEnv(..),commands,mkCommandEnv,interpretCommandLine)
|
|
||||||
import GF.Command.Commands2(PGFEnv,HasPGFEnv(..),pgf,concs,pgfEnv,emptyPGFEnv,pgfCommands)
|
|
||||||
import GF.Command.CommonCommands
|
|
||||||
import GF.Command.CommandInfo
|
|
||||||
import GF.Command.Help(helpCommand)
|
|
||||||
import GF.Command.Abstract
|
|
||||||
import GF.Command.Parse(readCommandLine,pCommand)
|
|
||||||
import GF.Data.Operations (Err(..),done)
|
|
||||||
import GF.Data.Utilities(whenM,repeatM)
|
|
||||||
|
|
||||||
import GF.Infra.UseIO(ioErrorText,putStrLnE)
|
|
||||||
import GF.Infra.SIO
|
|
||||||
import GF.Infra.Option
|
|
||||||
import qualified System.Console.Haskeline as Haskeline
|
|
||||||
--import GF.Text.Coding(decodeUnicode,encodeUnicode)
|
|
||||||
|
|
||||||
--import GF.Compile.Coding(codeTerm)
|
|
||||||
|
|
||||||
import qualified PGF2 as C
|
|
||||||
import qualified PGF as H
|
|
||||||
|
|
||||||
import Data.Char
|
|
||||||
import Data.List(isPrefixOf)
|
|
||||||
import qualified Data.Map as Map
|
|
||||||
|
|
||||||
import qualified Text.ParserCombinators.ReadP as RP
|
|
||||||
--import System.IO(utf8)
|
|
||||||
--import System.CPUTime(getCPUTime)
|
|
||||||
import System.Directory({-getCurrentDirectory,-}getAppUserDataDirectory)
|
|
||||||
import System.FilePath(takeExtensions)
|
|
||||||
import Control.Exception(SomeException,fromException,try)
|
|
||||||
--import Control.Monad
|
|
||||||
import Control.Monad.State hiding (void)
|
|
||||||
|
|
||||||
import qualified GF.System.Signal as IO(runInterruptibly)
|
|
||||||
{-
|
|
||||||
#ifdef SERVER_MODE
|
|
||||||
import GF.Server(server)
|
|
||||||
#endif
|
|
||||||
-}
|
|
||||||
|
|
||||||
import GF.Command.Messages(welcome)
|
|
||||||
|
|
||||||
-- | Run the GF Shell in quiet mode (@gf -run@).
|
|
||||||
mainRunGFI :: Options -> [FilePath] -> IO ()
|
|
||||||
mainRunGFI opts files = shell (beQuiet opts) files
|
|
||||||
|
|
||||||
beQuiet = addOptions (modifyFlags (\f -> f{optVerbosity=Quiet}))
|
|
||||||
|
|
||||||
-- | Run the interactive GF Shell
|
|
||||||
mainGFI :: Options -> [FilePath] -> IO ()
|
|
||||||
mainGFI opts files = do
|
|
||||||
P.putStrLn welcome
|
|
||||||
P.putStrLn "This shell uses the C run-time system. See help for available commands."
|
|
||||||
shell opts files
|
|
||||||
|
|
||||||
shell opts files = flip evalStateT (emptyGFEnv opts) $
|
|
||||||
do mapStateT runSIO $ importInEnv opts files
|
|
||||||
loop
|
|
||||||
|
|
||||||
{-
|
|
||||||
#ifdef SERVER_MODE
|
|
||||||
-- | Run the GF Server (@gf -server@).
|
|
||||||
-- The 'Int' argument is the port number for the HTTP service.
|
|
||||||
mainServerGFI opts0 port files =
|
|
||||||
server jobs port root (execute1 opts)
|
|
||||||
=<< runSIO (importInEnv (emptyGFEnv opts) opts files)
|
|
||||||
where
|
|
||||||
root = flag optDocumentRoot opts
|
|
||||||
opts = beQuiet opts0
|
|
||||||
jobs = join (flag optJobs opts)
|
|
||||||
#else
|
|
||||||
mainServerGFI opts port files =
|
|
||||||
error "GF has not been compiled with server mode support"
|
|
||||||
#endif
|
|
||||||
-}
|
|
||||||
-- | Read end execute commands until it is time to quit
|
|
||||||
loop :: StateT GFEnv IO ()
|
|
||||||
loop = repeatM readAndExecute1
|
|
||||||
|
|
||||||
-- | Read and execute one command, returning 'True' to continue execution,
|
|
||||||
-- | 'False' when it is time to quit
|
|
||||||
readAndExecute1 :: StateT GFEnv IO Bool
|
|
||||||
readAndExecute1 = mapStateT runSIO . execute1 =<< readCommand
|
|
||||||
|
|
||||||
-- | Read a command
|
|
||||||
readCommand :: StateT GFEnv IO String
|
|
||||||
readCommand =
|
|
||||||
do opts <- gets startOpts
|
|
||||||
case flag optMode opts of
|
|
||||||
ModeRun -> lift tryGetLine
|
|
||||||
_ -> lift . fetchCommand =<< get
|
|
||||||
|
|
||||||
timeIt act =
|
|
||||||
do t1 <- liftSIO $ getCPUTime
|
|
||||||
a <- act
|
|
||||||
t2 <- liftSIO $ getCPUTime
|
|
||||||
return (t2-t1,a)
|
|
||||||
|
|
||||||
-- | Optionally show how much CPU time was used to run an IO action
|
|
||||||
optionallyShowCPUTime :: (Monad m,MonadSIO m) => Options -> m a -> m a
|
|
||||||
optionallyShowCPUTime opts act
|
|
||||||
| not (verbAtLeast opts Normal) = act
|
|
||||||
| otherwise = do (dt,r) <- timeIt act
|
|
||||||
liftSIO $ putStrLnFlush $ show (dt `div` 1000000000) ++ " msec"
|
|
||||||
return r
|
|
||||||
|
|
||||||
type ShellM = StateT GFEnv SIO
|
|
||||||
|
|
||||||
-- | Execute a given command line, returning 'True' to continue execution,
|
|
||||||
-- | 'False' when it is time to quit
|
|
||||||
execute1 :: String -> ShellM Bool
|
|
||||||
execute1 s0 =
|
|
||||||
do modify $ \ gfenv0 -> gfenv0 {history = s0 : history gfenv0}
|
|
||||||
execute1' s0
|
|
||||||
|
|
||||||
-- | Execute a given command line, without adding it to the history
|
|
||||||
execute1' s0 =
|
|
||||||
do opts <- gets startOpts
|
|
||||||
interruptible $ optionallyShowCPUTime opts $
|
|
||||||
case pwords s0 of
|
|
||||||
-- cc, sd, so, ss and dg are now in GF.Commands.SourceCommands
|
|
||||||
-- special commands
|
|
||||||
"q" :_ -> quit
|
|
||||||
"!" :ws -> system_command ws
|
|
||||||
"eh":ws -> execute_history ws
|
|
||||||
"i" :ws -> do import_ ws; continue
|
|
||||||
-- other special commands, working on GFEnv
|
|
||||||
"dc":ws -> define_command ws
|
|
||||||
"dt":ws -> define_tree ws
|
|
||||||
-- ordinary commands
|
|
||||||
_ -> do env <- gets commandenv
|
|
||||||
interpretCommandLine env s0
|
|
||||||
continue
|
|
||||||
where
|
|
||||||
continue,stop :: ShellM Bool
|
|
||||||
continue = return True
|
|
||||||
stop = return False
|
|
||||||
|
|
||||||
interruptible :: ShellM Bool -> ShellM Bool
|
|
||||||
interruptible act =
|
|
||||||
do gfenv <- get
|
|
||||||
mapStateT (
|
|
||||||
either (\e -> printException e >> return (True,gfenv)) return
|
|
||||||
<=< runInterruptibly) act
|
|
||||||
|
|
||||||
-- Special commands:
|
|
||||||
|
|
||||||
quit = do opts <- gets startOpts
|
|
||||||
when (verbAtLeast opts Normal) $ putStrLnE "See you."
|
|
||||||
stop
|
|
||||||
|
|
||||||
system_command ws = do lift $ restrictedSystem $ unwords ws ; continue
|
|
||||||
|
|
||||||
|
|
||||||
{-"eh":w:_ -> do
|
|
||||||
cs <- readFile w >>= return . map words . lines
|
|
||||||
gfenv' <- foldM (flip (process False benv)) gfenv cs
|
|
||||||
loopNewCPU gfenv' -}
|
|
||||||
execute_history [w] =
|
|
||||||
do execute . lines =<< lift (restricted (readFile w))
|
|
||||||
continue
|
|
||||||
where
|
|
||||||
execute :: [String] -> ShellM ()
|
|
||||||
execute [] = done
|
|
||||||
execute (line:lines) = whenM (execute1' line) (execute lines)
|
|
||||||
|
|
||||||
execute_history _ =
|
|
||||||
do putStrLnE "eh command not parsed"
|
|
||||||
continue
|
|
||||||
|
|
||||||
define_command (f:ws) =
|
|
||||||
case readCommandLine (unwords ws) of
|
|
||||||
Just comm ->
|
|
||||||
do modify $
|
|
||||||
\ gfenv ->
|
|
||||||
let env = commandenv gfenv
|
|
||||||
in gfenv {
|
|
||||||
commandenv = env {
|
|
||||||
commandmacros = Map.insert f comm (commandmacros env)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
continue
|
|
||||||
_ -> dc_not_parsed
|
|
||||||
define_command _ = dc_not_parsed
|
|
||||||
|
|
||||||
dc_not_parsed = putStrLnE "command definition not parsed" >> continue
|
|
||||||
|
|
||||||
define_tree (f:ws) =
|
|
||||||
case H.readExpr (unwords ws) of
|
|
||||||
Just exp ->
|
|
||||||
do modify $
|
|
||||||
\ gfenv ->
|
|
||||||
let env = commandenv gfenv
|
|
||||||
in gfenv { commandenv = env {
|
|
||||||
expmacros = Map.insert f exp (expmacros env) } }
|
|
||||||
continue
|
|
||||||
_ -> dt_not_parsed
|
|
||||||
define_tree _ = dt_not_parsed
|
|
||||||
|
|
||||||
dt_not_parsed = putStrLnE "value definition not parsed" >> continue
|
|
||||||
|
|
||||||
pwords s = case words s of
|
|
||||||
w:ws -> getCommandOp w :ws
|
|
||||||
ws -> ws
|
|
||||||
import_ args =
|
|
||||||
do case parseOptions args of
|
|
||||||
Ok (opts',files) -> do
|
|
||||||
opts <- gets startOpts
|
|
||||||
curr_dir <- lift getCurrentDirectory
|
|
||||||
lib_dir <- lift $ getLibraryDirectory (addOptions opts opts')
|
|
||||||
importInEnv (addOptions opts (fixRelativeLibPaths curr_dir lib_dir opts')) files
|
|
||||||
Bad err ->
|
|
||||||
do putStrLnE $ "Command parse error: " ++ err
|
|
||||||
|
|
||||||
-- | Commands that work on 'GFEnv'
|
|
||||||
moreCommands = [
|
|
||||||
("e", emptyCommandInfo {
|
|
||||||
longname = "empty",
|
|
||||||
synopsis = "empty the environment (except the command history)",
|
|
||||||
exec = \ _ _ ->
|
|
||||||
do modify $ \ gfenv -> (emptyGFEnv (startOpts gfenv))
|
|
||||||
{ history=history gfenv }
|
|
||||||
return void
|
|
||||||
}),
|
|
||||||
("ph", emptyCommandInfo {
|
|
||||||
longname = "print_history",
|
|
||||||
synopsis = "print command history",
|
|
||||||
explanation = unlines [
|
|
||||||
"Prints the commands issued during the GF session.",
|
|
||||||
"The result is readable by the eh command.",
|
|
||||||
"The result can be used as a script when starting GF."
|
|
||||||
],
|
|
||||||
examples = [
|
|
||||||
mkEx "ph | wf -file=foo.gfs -- save the history into a file"
|
|
||||||
],
|
|
||||||
exec = \ _ _ ->
|
|
||||||
fmap (fromString . unlines . reverse . drop 1 . history) get
|
|
||||||
}),
|
|
||||||
("r", emptyCommandInfo {
|
|
||||||
longname = "reload",
|
|
||||||
synopsis = "repeat the latest import command",
|
|
||||||
exec = \ _ _ ->
|
|
||||||
do gfenv0 <- get
|
|
||||||
let imports = [(s,ws) | s <- history gfenv0, ("i":ws) <- [pwords s]]
|
|
||||||
case imports of
|
|
||||||
(s,ws):_ -> do
|
|
||||||
putStrLnE $ "repeating latest import: " ++ s
|
|
||||||
import_ ws
|
|
||||||
_ -> do
|
|
||||||
putStrLnE $ "no import in history"
|
|
||||||
return void
|
|
||||||
})
|
|
||||||
]
|
|
||||||
|
|
||||||
|
|
||||||
printException e = maybe (print e) (putStrLn . ioErrorText) (fromException e)
|
|
||||||
|
|
||||||
fetchCommand :: GFEnv -> IO String
|
|
||||||
fetchCommand gfenv = do
|
|
||||||
path <- getAppUserDataDirectory "gf_history"
|
|
||||||
let settings =
|
|
||||||
Haskeline.Settings {
|
|
||||||
Haskeline.complete = wordCompletion gfenv,
|
|
||||||
Haskeline.historyFile = Just path,
|
|
||||||
Haskeline.autoAddHistory = True
|
|
||||||
}
|
|
||||||
res <- IO.runInterruptibly $ Haskeline.runInputT settings (Haskeline.getInputLine (prompt gfenv))
|
|
||||||
case res of
|
|
||||||
Left _ -> return ""
|
|
||||||
Right Nothing -> return "q"
|
|
||||||
Right (Just s) -> return s
|
|
||||||
|
|
||||||
importInEnv :: Options -> [FilePath] -> ShellM ()
|
|
||||||
importInEnv opts files =
|
|
||||||
case files of
|
|
||||||
_ | flag optRetainResource opts ->
|
|
||||||
putStrLnE "Flag -retain is not supported in this shell"
|
|
||||||
[file] | takeExtensions file == ".pgf" -> importPGF file
|
|
||||||
[] -> done
|
|
||||||
_ -> do putStrLnE "Can only import one .pgf file"
|
|
||||||
where
|
|
||||||
importPGF file =
|
|
||||||
do gfenv <- get
|
|
||||||
case multigrammar gfenv of
|
|
||||||
Just _ -> putStrLnE "Discarding previous grammar"
|
|
||||||
_ -> done
|
|
||||||
pgf1 <- lift $ readPGF2 file
|
|
||||||
let gfenv' = gfenv { pgfenv = pgfEnv pgf1 }
|
|
||||||
when (verbAtLeast opts Normal) $
|
|
||||||
let langs = Map.keys . concretes $ gfenv'
|
|
||||||
in putStrLnE . unwords $ "\nLanguages:":langs
|
|
||||||
put gfenv'
|
|
||||||
|
|
||||||
tryGetLine = do
|
|
||||||
res <- try getLine
|
|
||||||
case res of
|
|
||||||
Left (e :: SomeException) -> return "q"
|
|
||||||
Right l -> return l
|
|
||||||
|
|
||||||
prompt env = abs ++ "> "
|
|
||||||
where
|
|
||||||
abs = maybe "" C.abstractName (multigrammar env)
|
|
||||||
|
|
||||||
data GFEnv = GFEnv {
|
|
||||||
startOpts :: Options,
|
|
||||||
--grammar :: (), -- gfo grammar -retain
|
|
||||||
--retain :: (), -- grammar was imported with -retain flag
|
|
||||||
pgfenv :: PGFEnv,
|
|
||||||
commandenv :: CommandEnv ShellM,
|
|
||||||
history :: [String]
|
|
||||||
}
|
|
||||||
|
|
||||||
emptyGFEnv opts = GFEnv opts {-() ()-} emptyPGFEnv emptyCommandEnv []
|
|
||||||
|
|
||||||
emptyCommandEnv = mkCommandEnv allCommands
|
|
||||||
multigrammar = pgf . pgfenv
|
|
||||||
concretes = concs . pgfenv
|
|
||||||
|
|
||||||
allCommands =
|
|
||||||
extend pgfCommands (helpCommand allCommands:moreCommands)
|
|
||||||
`Map.union` commonCommands
|
|
||||||
|
|
||||||
instance HasPGFEnv ShellM where getPGFEnv = gets pgfenv
|
|
||||||
|
|
||||||
-- ** Completion
|
|
||||||
|
|
||||||
wordCompletion gfenv (left,right) = do
|
|
||||||
case wc_type (reverse left) of
|
|
||||||
CmplCmd pref
|
|
||||||
-> ret (length pref) [Haskeline.simpleCompletion name | name <- Map.keys (commands cmdEnv), isPrefixOf pref name]
|
|
||||||
{-
|
|
||||||
CmplStr (Just (Command _ opts _)) s0
|
|
||||||
-> do mb_state0 <- try (evaluate (H.initState pgf (optLang opts) (optType opts)))
|
|
||||||
case mb_state0 of
|
|
||||||
Right state0 -> let (rprefix,rs) = break isSpace (reverse s0)
|
|
||||||
s = reverse rs
|
|
||||||
prefix = reverse rprefix
|
|
||||||
ws = words s
|
|
||||||
in case loop state0 ws of
|
|
||||||
Nothing -> ret 0 []
|
|
||||||
Just state -> let compls = H.getCompletions state prefix
|
|
||||||
in ret (length prefix) (map (\x -> Haskeline.simpleCompletion x) (Map.keys compls))
|
|
||||||
Left (_ :: SomeException) -> ret 0 []
|
|
||||||
-}
|
|
||||||
CmplOpt (Just (Command n _ _)) pref
|
|
||||||
-> case Map.lookup n (commands cmdEnv) of
|
|
||||||
Just inf -> do let flg_compls = [Haskeline.Completion ('-':flg++"=") ('-':flg) False | (flg,_) <- flags inf, isPrefixOf pref flg]
|
|
||||||
opt_compls = [Haskeline.Completion ('-':opt) ('-':opt) True | (opt,_) <- options inf, isPrefixOf pref opt]
|
|
||||||
ret (length pref+1)
|
|
||||||
(flg_compls++opt_compls)
|
|
||||||
Nothing -> ret (length pref) []
|
|
||||||
CmplIdent (Just (Command "i" _ _)) _ -- HACK: file name completion for command i
|
|
||||||
-> Haskeline.completeFilename (left,right)
|
|
||||||
|
|
||||||
CmplIdent _ pref
|
|
||||||
-> case mb_pgf of
|
|
||||||
Just pgf -> ret (length pref)
|
|
||||||
[Haskeline.simpleCompletion name
|
|
||||||
| name <- C.functions pgf,
|
|
||||||
isPrefixOf pref name]
|
|
||||||
_ -> ret (length pref) []
|
|
||||||
|
|
||||||
_ -> ret 0 []
|
|
||||||
where
|
|
||||||
mb_pgf = multigrammar gfenv
|
|
||||||
cmdEnv = commandenv gfenv
|
|
||||||
{-
|
|
||||||
optLang opts = valStrOpts "lang" (head $ Map.keys (concretes cmdEnv)) opts
|
|
||||||
optType opts =
|
|
||||||
let str = valStrOpts "cat" (H.showCId $ H.lookStartCat pgf) opts
|
|
||||||
in case H.readType str of
|
|
||||||
Just ty -> ty
|
|
||||||
Nothing -> error ("Can't parse '"++str++"' as type")
|
|
||||||
|
|
||||||
loop ps [] = Just ps
|
|
||||||
loop ps (t:ts) = case H.nextState ps (H.simpleParseInput t) of
|
|
||||||
Left es -> Nothing
|
|
||||||
Right ps -> loop ps ts
|
|
||||||
-}
|
|
||||||
ret len xs = return (drop len left,xs)
|
|
||||||
|
|
||||||
|
|
||||||
data CompletionType
|
|
||||||
= CmplCmd Ident
|
|
||||||
| CmplStr (Maybe Command) String
|
|
||||||
| CmplOpt (Maybe Command) Ident
|
|
||||||
| CmplIdent (Maybe Command) Ident
|
|
||||||
deriving Show
|
|
||||||
|
|
||||||
wc_type :: String -> CompletionType
|
|
||||||
wc_type = cmd_name
|
|
||||||
where
|
|
||||||
cmd_name cs =
|
|
||||||
let cs1 = dropWhile isSpace cs
|
|
||||||
in go cs1 cs1
|
|
||||||
where
|
|
||||||
go x [] = CmplCmd x
|
|
||||||
go x (c:cs)
|
|
||||||
| isIdent c = go x cs
|
|
||||||
| otherwise = cmd x cs
|
|
||||||
|
|
||||||
cmd x [] = ret CmplIdent x "" 0
|
|
||||||
cmd _ ('|':cs) = cmd_name cs
|
|
||||||
cmd _ (';':cs) = cmd_name cs
|
|
||||||
cmd x ('"':cs) = str x cs cs
|
|
||||||
cmd x ('-':cs) = option x cs cs
|
|
||||||
cmd x (c :cs)
|
|
||||||
| isIdent c = ident x (c:cs) cs
|
|
||||||
| otherwise = cmd x cs
|
|
||||||
|
|
||||||
option x y [] = ret CmplOpt x y 1
|
|
||||||
option x y ('=':cs) = optValue x y cs
|
|
||||||
option x y (c :cs)
|
|
||||||
| isIdent c = option x y cs
|
|
||||||
| otherwise = cmd x cs
|
|
||||||
|
|
||||||
optValue x y ('"':cs) = str x y cs
|
|
||||||
optValue x y cs = cmd x cs
|
|
||||||
|
|
||||||
ident x y [] = ret CmplIdent x y 0
|
|
||||||
ident x y (c:cs)
|
|
||||||
| isIdent c = ident x y cs
|
|
||||||
| otherwise = cmd x cs
|
|
||||||
|
|
||||||
str x y [] = ret CmplStr x y 1
|
|
||||||
str x y ('\"':cs) = cmd x cs
|
|
||||||
str x y ('\\':c:cs) = str x y cs
|
|
||||||
str x y (c:cs) = str x y cs
|
|
||||||
|
|
||||||
ret f x y d = f cmd y
|
|
||||||
where
|
|
||||||
x1 = take (length x - length y - d) x
|
|
||||||
x2 = takeWhile (\c -> isIdent c || isSpace c || c == '-' || c == '=' || c == '"') x1
|
|
||||||
|
|
||||||
cmd = case [x | (x,cs) <- RP.readP_to_S pCommand x2, all isSpace cs] of
|
|
||||||
[x] -> Just x
|
|
||||||
_ -> Nothing
|
|
||||||
|
|
||||||
isIdent c = c == '_' || c == '\'' || isAlphaNum c
|
|
||||||
@@ -2,10 +2,7 @@
|
|||||||
{-# LANGUAGE CPP #-}
|
{-# LANGUAGE CPP #-}
|
||||||
module GF.Main where
|
module GF.Main where
|
||||||
import GF.Compiler
|
import GF.Compiler
|
||||||
import qualified GF.Interactive as GFI1
|
import GF.Interactive
|
||||||
#ifdef C_RUNTIME
|
|
||||||
import qualified GF.Interactive2 as GFI2
|
|
||||||
#endif
|
|
||||||
import GF.Data.ErrM
|
import GF.Data.ErrM
|
||||||
import GF.Infra.Option
|
import GF.Infra.Option
|
||||||
import GF.Infra.UseIO
|
import GF.Infra.UseIO
|
||||||
@@ -47,17 +44,7 @@ mainOpts opts files =
|
|||||||
case flag optMode opts of
|
case flag optMode opts of
|
||||||
ModeVersion -> putStrLn $ "Grammatical Framework (GF) version " ++ showVersion version ++ "\n" ++ buildInfo
|
ModeVersion -> putStrLn $ "Grammatical Framework (GF) version " ++ showVersion version ++ "\n" ++ buildInfo
|
||||||
ModeHelp -> putStrLn helpMessage
|
ModeHelp -> putStrLn helpMessage
|
||||||
ModeServer port -> GFI1.mainServerGFI opts port files
|
ModeServer port -> mainServerGFI opts port files
|
||||||
ModeCompiler -> mainGFC opts files
|
ModeCompiler -> mainGFC opts files
|
||||||
ModeInteractive -> GFI1.mainGFI opts files
|
ModeInteractive -> mainGFI opts files
|
||||||
ModeRun -> GFI1.mainRunGFI opts files
|
ModeRun -> mainRunGFI opts files
|
||||||
#ifdef C_RUNTIME
|
|
||||||
ModeInteractive2 -> GFI2.mainGFI opts files
|
|
||||||
ModeRun2 -> GFI2.mainRunGFI opts files
|
|
||||||
#else
|
|
||||||
ModeInteractive2 -> noCruntime
|
|
||||||
ModeRun2 -> noCruntime
|
|
||||||
where
|
|
||||||
noCruntime = do ePutStrLn "GF configured without C run-time support"
|
|
||||||
exitFailure
|
|
||||||
#endif
|
|
||||||
|
|||||||
@@ -18,13 +18,8 @@ module GF.Quiz (
|
|||||||
morphologyList
|
morphologyList
|
||||||
) where
|
) where
|
||||||
|
|
||||||
import PGF
|
import PGF2
|
||||||
--import PGF.Linearize
|
|
||||||
import GF.Data.Operations
|
import GF.Data.Operations
|
||||||
--import GF.Infra.UseIO
|
|
||||||
--import GF.Infra.Option
|
|
||||||
--import PGF.Probabilistic
|
|
||||||
|
|
||||||
import System.Random
|
import System.Random
|
||||||
import Data.List (nub)
|
import Data.List (nub)
|
||||||
|
|
||||||
@@ -38,7 +33,7 @@ mkQuiz msg tts = do
|
|||||||
teachDialogue qas msg
|
teachDialogue qas msg
|
||||||
|
|
||||||
translationList ::
|
translationList ::
|
||||||
Maybe Expr -> PGF -> Language -> Language -> Type -> Int -> IO [(String,[String])]
|
Maybe Expr -> PGF -> Concr -> Concr -> Type -> Int -> IO [(String,[String])]
|
||||||
translationList mex pgf ig og typ number = do
|
translationList mex pgf ig og typ number = do
|
||||||
gen <- newStdGen
|
gen <- newStdGen
|
||||||
let ts = take number $ case mex of
|
let ts = take number $ case mex of
|
||||||
@@ -46,19 +41,22 @@ translationList mex pgf ig og typ number = do
|
|||||||
Nothing -> generateRandom gen pgf typ
|
Nothing -> generateRandom gen pgf typ
|
||||||
return $ map mkOne $ ts
|
return $ map mkOne $ ts
|
||||||
where
|
where
|
||||||
mkOne t = (norml (linearize pgf ig t),
|
mkOne t = (norml (linearize ig t),
|
||||||
map norml (concatMap lins (homonyms t)))
|
map norml (concatMap lins (homonyms t)))
|
||||||
homonyms = parse pgf ig typ . linearize pgf ig
|
homonyms t =
|
||||||
lins = nub . concatMap (map snd) . tabularLinearizes pgf og
|
case (parse ig typ . linearize ig) t of
|
||||||
|
ParseOk res -> map fst res
|
||||||
|
_ -> []
|
||||||
|
lins = nub . concatMap (map snd) . tabularLinearizeAll og
|
||||||
|
|
||||||
morphologyList ::
|
morphologyList ::
|
||||||
Maybe Expr -> PGF -> Language -> Type -> Int -> IO [(String,[String])]
|
Maybe Expr -> PGF -> Concr -> Type -> Int -> IO [(String,[String])]
|
||||||
morphologyList mex pgf ig typ number = do
|
morphologyList mex pgf ig typ number = do
|
||||||
gen <- newStdGen
|
gen <- newStdGen
|
||||||
let ts = take (max 1 number) $ case mex of
|
let ts = take (max 1 number) $ case mex of
|
||||||
Just ex -> generateRandomFrom gen pgf ex
|
Just ex -> generateRandomFrom gen pgf ex
|
||||||
Nothing -> generateRandom gen pgf typ
|
Nothing -> generateRandom gen pgf typ
|
||||||
let ss = map (tabularLinearizes pgf ig) ts
|
let ss = map (tabularLinearizeAll ig) ts
|
||||||
let size = length (head (head ss))
|
let size = length (head (head ss))
|
||||||
let forms = take number $ randomRs (0,size-1) gen
|
let forms = take number $ randomRs (0,size-1) gen
|
||||||
return [(snd (head pws0) +++ fst (pws0 !! i), ws) |
|
return [(snd (head pws0) +++ fst (pws0 !! i), ws) |
|
||||||
|
|||||||
@@ -3,7 +3,6 @@
|
|||||||
module GF.Server(server) where
|
module GF.Server(server) where
|
||||||
import Data.List(partition,stripPrefix,isInfixOf)
|
import Data.List(partition,stripPrefix,isInfixOf)
|
||||||
import qualified Data.Map as M
|
import qualified Data.Map as M
|
||||||
import Control.Applicative -- for GHC<7.10
|
|
||||||
import Control.Monad(when)
|
import Control.Monad(when)
|
||||||
import Control.Monad.State(StateT(..),get,gets,put)
|
import Control.Monad.State(StateT(..),get,gets,put)
|
||||||
import Control.Monad.Error(ErrorT(..),Error(..))
|
import Control.Monad.Error(ErrorT(..),Error(..))
|
||||||
@@ -34,7 +33,7 @@ import Network.Shed.Httpd(initServer,Request(..),Response(..),noCache)
|
|||||||
--import qualified Network.FastCGI as FCGI -- from hackage direct-fastcgi
|
--import qualified Network.FastCGI as FCGI -- from hackage direct-fastcgi
|
||||||
import Network.CGI(handleErrors,liftIO)
|
import Network.CGI(handleErrors,liftIO)
|
||||||
import CGIUtils(handleCGIErrors)--,outputJSONP,stderrToFile
|
import CGIUtils(handleCGIErrors)--,outputJSONP,stderrToFile
|
||||||
import Text.JSON(JSValue(..),Result(..),valFromObj,encode,decode,showJSON,makeObj)
|
import Text.JSON(encode,showJSON,makeObj)
|
||||||
--import System.IO.Silently(hCapture)
|
--import System.IO.Silently(hCapture)
|
||||||
import System.Process(readProcessWithExitCode)
|
import System.Process(readProcessWithExitCode)
|
||||||
import System.Exit(ExitCode(..))
|
import System.Exit(ExitCode(..))
|
||||||
@@ -43,7 +42,6 @@ import GF.Infra.UseIO(readBinaryFile,writeBinaryFile,ePutStrLn)
|
|||||||
import GF.Infra.SIO(captureSIO)
|
import GF.Infra.SIO(captureSIO)
|
||||||
import GF.Data.Utilities(apSnd,mapSnd)
|
import GF.Data.Utilities(apSnd,mapSnd)
|
||||||
import qualified PGFService as PS
|
import qualified PGFService as PS
|
||||||
import qualified ExampleService as ES
|
|
||||||
import Data.Version(showVersion)
|
import Data.Version(showVersion)
|
||||||
import Paths_gf(getDataDir,version)
|
import Paths_gf(getDataDir,version)
|
||||||
import GF.Infra.BuildInfo (buildInfo)
|
import GF.Infra.BuildInfo (buildInfo)
|
||||||
@@ -171,7 +169,6 @@ handle logLn documentroot state0 cache execute1 stateVar
|
|||||||
(_ ,_ ,".pgf") -> do --debug $ "PGF service: "++path
|
(_ ,_ ,".pgf") -> do --debug $ "PGF service: "++path
|
||||||
wrapCGI $ PS.cgiMain' cache path
|
wrapCGI $ PS.cgiMain' cache path
|
||||||
(dir,"grammars.cgi",_ ) -> grammarList dir (decoded qs)
|
(dir,"grammars.cgi",_ ) -> grammarList dir (decoded qs)
|
||||||
(dir ,"exb.fcgi" ,_ ) -> wrapCGI $ ES.cgiMain' root dir (PS.pgfCache cache)
|
|
||||||
_ -> serveStaticFile rpath path
|
_ -> serveStaticFile rpath path
|
||||||
where path = translatePath rpath
|
where path = translatePath rpath
|
||||||
_ -> return $ resp400 upath
|
_ -> return $ resp400 upath
|
||||||
@@ -180,7 +177,7 @@ handle logLn documentroot state0 cache execute1 stateVar
|
|||||||
|
|
||||||
translatePath rpath = root</>rpath -- hmm, check for ".."
|
translatePath rpath = root</>rpath -- hmm, check for ".."
|
||||||
|
|
||||||
versionInfo (c1,c2) =
|
versionInfo c =
|
||||||
html200 . unlines $
|
html200 . unlines $
|
||||||
"<!DOCTYPE html>":
|
"<!DOCTYPE html>":
|
||||||
"<meta name = \"viewport\" content = \"width = device-width\">":
|
"<meta name = \"viewport\" content = \"width = device-width\">":
|
||||||
@@ -188,8 +185,7 @@ handle logLn documentroot state0 cache execute1 stateVar
|
|||||||
"":
|
"":
|
||||||
("<h2>"++hdr++"</h2>"):
|
("<h2>"++hdr++"</h2>"):
|
||||||
(zipWith (++) ("<p>":repeat "<br>") buildinfo)++
|
(zipWith (++) ("<p>":repeat "<br>") buildinfo)++
|
||||||
sh "Haskell run-time system" c1++
|
sh "Run-time system" c
|
||||||
sh "C run-time system" c2
|
|
||||||
where
|
where
|
||||||
hdr:buildinfo = lines gf_version
|
hdr:buildinfo = lines gf_version
|
||||||
rel = makeRelative documentroot
|
rel = makeRelative documentroot
|
||||||
@@ -284,17 +280,13 @@ handle logLn documentroot state0 cache execute1 stateVar
|
|||||||
skip_empty = filter (not.null.snd)
|
skip_empty = filter (not.null.snd)
|
||||||
|
|
||||||
jsonList = jsonList' return
|
jsonList = jsonList' return
|
||||||
jsonListLong ext = jsonList' (mapM (addTime ext)) ext
|
jsonListLong = jsonList' (mapM addTime)
|
||||||
jsonList' details ext = fmap (json200) (details =<< ls_ext "." ext)
|
jsonList' details ext = fmap (json200) (details =<< ls_ext "." ext)
|
||||||
|
|
||||||
addTime ext path =
|
addTime path =
|
||||||
do t <- getModificationTime path
|
do t <- getModificationTime path
|
||||||
if ext==".json"
|
return $ makeObj ["path".=path,"time".=format t]
|
||||||
then addComment (time t) <$> liftIO (try $ getComment path)
|
|
||||||
else return . makeObj $ time t
|
|
||||||
where
|
where
|
||||||
addComment t = makeObj . either (const t) (\c->t++["comment".=c])
|
|
||||||
time t = ["path".=path,"time".=format t]
|
|
||||||
format = formatTime defaultTimeLocale rfc822DateFormat
|
format = formatTime defaultTimeLocale rfc822DateFormat
|
||||||
|
|
||||||
rm path | takeExtension path `elem` ok_to_delete =
|
rm path | takeExtension path `elem` ok_to_delete =
|
||||||
@@ -336,11 +328,6 @@ handle logLn documentroot state0 cache execute1 stateVar
|
|||||||
do paths <- getDirectoryContents dir
|
do paths <- getDirectoryContents dir
|
||||||
return [path | path<-paths, takeExtension path==ext]
|
return [path | path<-paths, takeExtension path==ext]
|
||||||
|
|
||||||
getComment path =
|
|
||||||
do Ok (JSObject obj) <- decode <$> readFile path
|
|
||||||
Ok cmnt <- return (valFromObj "comment" obj)
|
|
||||||
return (cmnt::String)
|
|
||||||
|
|
||||||
-- * Dynamic content
|
-- * Dynamic content
|
||||||
|
|
||||||
jsonresult cwd dir cmd (ecode,stdout,stderr) files =
|
jsonresult cwd dir cmd (ecode,stdout,stderr) files =
|
||||||
|
|||||||
@@ -14,7 +14,6 @@ import qualified Data.Map as Map
|
|||||||
import Data.Set (Set)
|
import Data.Set (Set)
|
||||||
import qualified Data.Set as Set
|
import qualified Data.Set as Set
|
||||||
|
|
||||||
import PGF.Internal
|
|
||||||
import GF.Data.Utilities
|
import GF.Data.Utilities
|
||||||
import GF.Grammar.CFG
|
import GF.Grammar.CFG
|
||||||
--import GF.Speech.PGFToCFG
|
--import GF.Speech.PGFToCFG
|
||||||
|
|||||||
@@ -7,15 +7,12 @@
|
|||||||
-----------------------------------------------------------------------------
|
-----------------------------------------------------------------------------
|
||||||
|
|
||||||
module GF.Speech.GSL (gslPrinter) where
|
module GF.Speech.GSL (gslPrinter) where
|
||||||
import Prelude hiding ((<>)) -- GHC 8.4.1 clash with Text.PrettyPrint
|
|
||||||
|
|
||||||
--import GF.Data.Utilities
|
|
||||||
import GF.Grammar.CFG
|
import GF.Grammar.CFG
|
||||||
import GF.Speech.SRG
|
import GF.Speech.SRG
|
||||||
import GF.Speech.RegExp
|
import GF.Speech.RegExp
|
||||||
import GF.Infra.Option
|
import GF.Infra.Option
|
||||||
--import GF.Infra.Ident
|
import PGF2
|
||||||
import PGF
|
|
||||||
|
|
||||||
import Data.Char (toUpper,toLower)
|
import Data.Char (toUpper,toLower)
|
||||||
import Data.List (partition)
|
import Data.List (partition)
|
||||||
@@ -24,7 +21,7 @@ import GF.Text.Pretty
|
|||||||
width :: Int
|
width :: Int
|
||||||
width = 75
|
width = 75
|
||||||
|
|
||||||
gslPrinter :: Options -> PGF -> CId -> String
|
gslPrinter :: Options -> PGF -> Concr -> String
|
||||||
gslPrinter opts pgf cnc = renderStyle st $ prGSL $ makeNonLeftRecursiveSRG opts pgf cnc
|
gslPrinter opts pgf cnc = renderStyle st $ prGSL $ makeNonLeftRecursiveSRG opts pgf cnc
|
||||||
where st = style { lineLength = width }
|
where st = style { lineLength = width }
|
||||||
|
|
||||||
|
|||||||
@@ -11,7 +11,6 @@
|
|||||||
-----------------------------------------------------------------------------
|
-----------------------------------------------------------------------------
|
||||||
|
|
||||||
module GF.Speech.JSGF (jsgfPrinter) where
|
module GF.Speech.JSGF (jsgfPrinter) where
|
||||||
import Prelude hiding ((<>)) -- GHC 8.4.1 clash with Text.PrettyPrint
|
|
||||||
|
|
||||||
--import GF.Data.Utilities
|
--import GF.Data.Utilities
|
||||||
import GF.Infra.Option
|
import GF.Infra.Option
|
||||||
@@ -19,7 +18,7 @@ import GF.Grammar.CFG
|
|||||||
import GF.Speech.RegExp
|
import GF.Speech.RegExp
|
||||||
import GF.Speech.SISR
|
import GF.Speech.SISR
|
||||||
import GF.Speech.SRG
|
import GF.Speech.SRG
|
||||||
import PGF
|
import PGF2
|
||||||
|
|
||||||
import Data.Char
|
import Data.Char
|
||||||
import Data.List
|
import Data.List
|
||||||
@@ -31,8 +30,8 @@ width :: Int
|
|||||||
width = 75
|
width = 75
|
||||||
|
|
||||||
jsgfPrinter :: Options
|
jsgfPrinter :: Options
|
||||||
-> PGF
|
-> PGF
|
||||||
-> CId -> String
|
-> Concr -> String
|
||||||
jsgfPrinter opts pgf cnc = renderStyle st $ prJSGF sisr $ makeNonLeftRecursiveSRG opts pgf cnc
|
jsgfPrinter opts pgf cnc = renderStyle st $ prJSGF sisr $ makeNonLeftRecursiveSRG opts pgf cnc
|
||||||
where st = style { lineLength = width }
|
where st = style { lineLength = width }
|
||||||
sisr = flag optSISR opts
|
sisr = flag optSISR opts
|
||||||
|
|||||||
@@ -6,60 +6,54 @@
|
|||||||
----------------------------------------------------------------------
|
----------------------------------------------------------------------
|
||||||
module GF.Speech.PGFToCFG (bnfPrinter, pgfToCFG) where
|
module GF.Speech.PGFToCFG (bnfPrinter, pgfToCFG) where
|
||||||
|
|
||||||
import PGF(showCId)
|
import PGF2
|
||||||
import PGF.Internal as PGF
|
import PGF2.Internal
|
||||||
--import GF.Infra.Ident
|
|
||||||
import GF.Grammar.CFG hiding (Symbol)
|
import GF.Grammar.CFG hiding (Symbol)
|
||||||
|
|
||||||
import Data.Array.IArray as Array
|
|
||||||
--import Data.List
|
|
||||||
import Data.Map (Map)
|
import Data.Map (Map)
|
||||||
import qualified Data.Map as Map
|
import qualified Data.Map as Map
|
||||||
import qualified Data.IntMap as IntMap
|
import qualified Data.IntMap as IntMap
|
||||||
--import Data.Maybe
|
|
||||||
import Data.Set (Set)
|
import Data.Set (Set)
|
||||||
import qualified Data.Set as Set
|
import qualified Data.Set as Set
|
||||||
|
|
||||||
bnfPrinter :: PGF -> CId -> String
|
bnfPrinter :: PGF -> Concr -> String
|
||||||
bnfPrinter = toBNF id
|
bnfPrinter = toBNF id
|
||||||
|
|
||||||
toBNF :: (CFG -> CFG) -> PGF -> CId -> String
|
toBNF :: (CFG -> CFG) -> PGF -> Concr -> String
|
||||||
toBNF f pgf cnc = prCFG $ f $ pgfToCFG pgf cnc
|
toBNF f pgf cnc = prCFG $ f $ pgfToCFG pgf cnc
|
||||||
|
|
||||||
type Profile = [Int]
|
type Profile = [Int]
|
||||||
|
|
||||||
pgfToCFG :: PGF
|
pgfToCFG :: PGF -> Concr -> CFG
|
||||||
-> CId -- ^ Concrete syntax name
|
pgfToCFG pgf cnc = mkCFG start_cat extCats (startRules ++ concatMap ruleToCFRule rules)
|
||||||
-> CFG
|
|
||||||
pgfToCFG pgf lang = mkCFG (showCId (lookStartCat pgf)) extCats (startRules ++ concatMap ruleToCFRule rules)
|
|
||||||
where
|
where
|
||||||
cnc = lookConcr pgf lang
|
(_,start_cat,_) = unType (startCat pgf)
|
||||||
|
|
||||||
rules :: [(FId,Production)]
|
rules :: [(FId,Production)]
|
||||||
rules = [(fcat,prod) | (fcat,set) <- IntMap.toList (PGF.productions cnc)
|
rules = [(fcat,prod) | fcat <- [0..concrTotalCats cnc],
|
||||||
, prod <- Set.toList set]
|
prod <- concrProductions cnc fcat]
|
||||||
|
|
||||||
fcatCats :: Map FId Cat
|
fcatCats :: Map FId Cat
|
||||||
fcatCats = Map.fromList [(fc, showCId c ++ "_" ++ show i)
|
fcatCats = Map.fromList [(fc, c ++ "_" ++ show i)
|
||||||
| (c,CncCat s e lbls) <- Map.toList (cnccats cnc),
|
| (c,s,e,lbls) <- concrCategories cnc,
|
||||||
(fc,i) <- zip (range (s,e)) [1..]]
|
(fc,i) <- zip [s..e] [1..]]
|
||||||
|
|
||||||
fcatCat :: FId -> Cat
|
fcatCat :: FId -> Cat
|
||||||
fcatCat c = Map.findWithDefault ("Unknown_" ++ show c) c fcatCats
|
fcatCat c = Map.findWithDefault ("Unknown_" ++ show c) c fcatCats
|
||||||
|
|
||||||
fcatToCat :: FId -> LIndex -> Cat
|
fcatToCat :: FId -> Int -> Cat
|
||||||
fcatToCat c l = fcatCat c ++ row
|
fcatToCat c l = fcatCat c ++ row
|
||||||
where row = if catLinArity c == 1 then "" else "_" ++ show l
|
where row = if catLinArity c == 1 then "" else "_" ++ show l
|
||||||
|
|
||||||
-- gets the number of fields in the lincat for the given category
|
-- gets the number of fields in the lincat for the given category
|
||||||
catLinArity :: FId -> Int
|
catLinArity :: FId -> Int
|
||||||
catLinArity c = maximum (1:[rangeSize (bounds rhs) | (CncFun _ rhs, _) <- topdownRules c])
|
catLinArity c = maximum (1:[length rhs | ((_,rhs), _) <- topdownRules c])
|
||||||
|
|
||||||
topdownRules cat = f cat []
|
topdownRules cat = f cat []
|
||||||
where
|
where
|
||||||
f cat rules = maybe rules (Set.foldr g rules) (IntMap.lookup cat (productions cnc))
|
f cat rules = foldr g rules (concrProductions cnc cat)
|
||||||
|
|
||||||
g (PApply funid args) rules = (cncfuns cnc ! funid,args) : rules
|
g (PApply funid args) rules = (concrFunction cnc funid,args) : rules
|
||||||
g (PCoerce cat) rules = f cat rules
|
g (PCoerce cat) rules = f cat rules
|
||||||
|
|
||||||
|
|
||||||
@@ -67,26 +61,26 @@ pgfToCFG pgf lang = mkCFG (showCId (lookStartCat pgf)) extCats (startRules ++ co
|
|||||||
extCats = Set.fromList $ map ruleLhs startRules
|
extCats = Set.fromList $ map ruleLhs startRules
|
||||||
|
|
||||||
startRules :: [CFRule]
|
startRules :: [CFRule]
|
||||||
startRules = [Rule (showCId c) [NonTerminal (fcatToCat fc r)] (CFRes 0)
|
startRules = [Rule c [NonTerminal (fcatToCat fc r)] (CFRes 0)
|
||||||
| (c,CncCat s e lbls) <- Map.toList (cnccats cnc),
|
| (c,s,e,lbls) <- concrCategories cnc,
|
||||||
fc <- range (s,e), not (isPredefFId fc),
|
fc <- [s..e], not (isPredefFId fc),
|
||||||
r <- [0..catLinArity fc-1]]
|
r <- [0..catLinArity fc-1]]
|
||||||
|
|
||||||
ruleToCFRule :: (FId,Production) -> [CFRule]
|
ruleToCFRule :: (FId,Production) -> [CFRule]
|
||||||
ruleToCFRule (c,PApply funid args) =
|
ruleToCFRule (c,PApply funid args) =
|
||||||
[Rule (fcatToCat c l) (mkRhs row) (profilesToTerm [fixProfile row n | n <- [0..length args-1]])
|
[Rule (fcatToCat c l) (mkRhs row) (profilesToTerm [fixProfile row n | n <- [0..length args-1]])
|
||||||
| (l,seqid) <- Array.assocs rhs
|
| (l,seqid) <- zip [0..] rhs
|
||||||
, let row = sequences cnc ! seqid
|
, let row = concrSequence cnc seqid
|
||||||
, not (containsLiterals row)]
|
, not (containsLiterals row)]
|
||||||
where
|
where
|
||||||
CncFun f rhs = cncfuns cnc ! funid
|
(f, rhs) = concrFunction cnc funid
|
||||||
|
|
||||||
mkRhs :: Array DotPos Symbol -> [CFSymbol]
|
mkRhs :: [Symbol] -> [CFSymbol]
|
||||||
mkRhs = concatMap symbolToCFSymbol . Array.elems
|
mkRhs = concatMap symbolToCFSymbol
|
||||||
|
|
||||||
containsLiterals :: Array DotPos Symbol -> Bool
|
containsLiterals :: [Symbol] -> Bool
|
||||||
containsLiterals row = not (null ([n | SymLit n _ <- Array.elems row] ++
|
containsLiterals row = not (null ([n | SymLit n _ <- row] ++
|
||||||
[n | SymVar n _ <- Array.elems row]))
|
[n | SymVar n _ <- row]))
|
||||||
|
|
||||||
symbolToCFSymbol :: Symbol -> [CFSymbol]
|
symbolToCFSymbol :: Symbol -> [CFSymbol]
|
||||||
symbolToCFSymbol (SymCat n l) = [let PArg _ fid = args!!n in NonTerminal (fcatToCat fid l)]
|
symbolToCFSymbol (SymCat n l) = [let PArg _ fid = args!!n in NonTerminal (fcatToCat fid l)]
|
||||||
@@ -102,10 +96,10 @@ pgfToCFG pgf lang = mkCFG (showCId (lookStartCat pgf)) extCats (startRules ++ co
|
|||||||
symbolToCFSymbol SymALL_CAPIT = [Terminal "&|"]
|
symbolToCFSymbol SymALL_CAPIT = [Terminal "&|"]
|
||||||
symbolToCFSymbol SymNE = []
|
symbolToCFSymbol SymNE = []
|
||||||
|
|
||||||
fixProfile :: Array DotPos Symbol -> Int -> Profile
|
fixProfile :: [Symbol] -> Int -> Profile
|
||||||
fixProfile row i = [k | (k,j) <- nts, j == i]
|
fixProfile row i = [k | (k,j) <- nts, j == i]
|
||||||
where
|
where
|
||||||
nts = zip [0..] [j | nt <- Array.elems row, j <- getPos nt]
|
nts = zip [0..] [j | nt <- row, j <- getPos nt]
|
||||||
|
|
||||||
getPos (SymCat j _) = [j]
|
getPos (SymCat j _) = [j]
|
||||||
getPos (SymLit j _) = [j]
|
getPos (SymLit j _) = [j]
|
||||||
@@ -113,9 +107,10 @@ pgfToCFG pgf lang = mkCFG (showCId (lookStartCat pgf)) extCats (startRules ++ co
|
|||||||
|
|
||||||
profilesToTerm :: [Profile] -> CFTerm
|
profilesToTerm :: [Profile] -> CFTerm
|
||||||
profilesToTerm ps = CFObj f (zipWith profileToTerm argTypes ps)
|
profilesToTerm ps = CFObj f (zipWith profileToTerm argTypes ps)
|
||||||
where (argTypes,_) = catSkeleton $ lookType (abstract pgf) f
|
where Just (hypos,_,_) = fmap unType (functionType pgf f)
|
||||||
|
argTypes = [cat | (_,_,ty) <- hypos, let (_,cat,_) = unType ty]
|
||||||
|
|
||||||
profileToTerm :: CId -> Profile -> CFTerm
|
profileToTerm :: Fun -> Profile -> CFTerm
|
||||||
profileToTerm t [] = CFMeta t
|
profileToTerm t [] = CFMeta t
|
||||||
profileToTerm _ xs = CFRes (last xs) -- FIXME: unify
|
profileToTerm _ xs = CFRes (last xs) -- FIXME: unify
|
||||||
ruleToCFRule (c,PCoerce c') =
|
ruleToCFRule (c,PCoerce c') =
|
||||||
|
|||||||
@@ -11,12 +11,12 @@ import GF.Grammar.CFG
|
|||||||
import GF.Speech.CFGToFA
|
import GF.Speech.CFGToFA
|
||||||
import GF.Speech.PGFToCFG
|
import GF.Speech.PGFToCFG
|
||||||
import GF.Speech.RegExp
|
import GF.Speech.RegExp
|
||||||
import PGF
|
import PGF2
|
||||||
|
|
||||||
regexpPrinter :: PGF -> CId -> String
|
regexpPrinter :: PGF -> Concr -> String
|
||||||
regexpPrinter pgf cnc = (++"\n") $ prRE id $ dfa2re $ cfgToFA $ pgfToCFG pgf cnc
|
regexpPrinter pgf cnc = (++"\n") $ prRE id $ dfa2re $ cfgToFA $ pgfToCFG pgf cnc
|
||||||
|
|
||||||
multiRegexpPrinter :: PGF -> CId -> String
|
multiRegexpPrinter :: PGF -> Concr -> String
|
||||||
multiRegexpPrinter pgf cnc = prREs $ mfa2res $ cfgToMFA $ pgfToCFG pgf cnc
|
multiRegexpPrinter pgf cnc = prREs $ mfa2res $ cfgToMFA $ pgfToCFG pgf cnc
|
||||||
|
|
||||||
prREs :: [(String,RE CFSymbol)] -> String
|
prREs :: [(String,RE CFSymbol)] -> String
|
||||||
|
|||||||
@@ -10,13 +10,9 @@ module GF.Speech.SISR (SISRFormat(..), SISRTag, prSISR,
|
|||||||
|
|
||||||
import Data.List
|
import Data.List
|
||||||
|
|
||||||
--import GF.Data.Utilities
|
|
||||||
--import GF.Infra.Ident
|
|
||||||
import GF.Infra.Option (SISRFormat(..))
|
import GF.Infra.Option (SISRFormat(..))
|
||||||
import GF.Grammar.CFG
|
import GF.Grammar.CFG
|
||||||
import GF.Speech.SRG (SRGNT)
|
import GF.Speech.SRG (SRGNT)
|
||||||
import PGF(showCId)
|
|
||||||
|
|
||||||
import qualified GF.JavaScript.AbsJS as JS
|
import qualified GF.JavaScript.AbsJS as JS
|
||||||
import qualified GF.JavaScript.PrintJS as JS
|
import qualified GF.JavaScript.PrintJS as JS
|
||||||
|
|
||||||
@@ -50,12 +46,12 @@ catSISR t (c,i) fmt
|
|||||||
profileFinalSISR :: CFTerm -> SISRFormat -> SISRTag
|
profileFinalSISR :: CFTerm -> SISRFormat -> SISRTag
|
||||||
profileFinalSISR term fmt = [JS.DExpr $ fmtOut fmt `ass` f term]
|
profileFinalSISR term fmt = [JS.DExpr $ fmtOut fmt `ass` f term]
|
||||||
where
|
where
|
||||||
f (CFObj n ts) = tree (showCId n) (map f ts)
|
f (CFObj n ts) = tree n (map f ts)
|
||||||
f (CFAbs v x) = JS.EFun [var v] [JS.SReturn (f x)]
|
f (CFAbs v x) = JS.EFun [var v] [JS.SReturn (f x)]
|
||||||
f (CFApp x y) = JS.ECall (f x) [f y]
|
f (CFApp x y) = JS.ECall (f x) [f y]
|
||||||
f (CFRes i) = JS.EIndex (JS.EVar args) (JS.EInt (fromIntegral i))
|
f (CFRes i) = JS.EIndex (JS.EVar args) (JS.EInt (fromIntegral i))
|
||||||
f (CFVar v) = JS.EVar (var v)
|
f (CFVar v) = JS.EVar (var v)
|
||||||
f (CFMeta typ) = obj [("name",JS.EStr "?"), ("type",JS.EStr (showCId typ))]
|
f (CFMeta typ) = obj [("name",JS.EStr "?"), ("type",JS.EStr typ)]
|
||||||
|
|
||||||
fmtOut SISR_WD20030401 = JS.EVar (JS.Ident "$")
|
fmtOut SISR_WD20030401 = JS.EVar (JS.Ident "$")
|
||||||
fmtOut SISR_1_0 = JS.EVar (JS.Ident "out")
|
fmtOut SISR_1_0 = JS.EVar (JS.Ident "out")
|
||||||
|
|||||||
@@ -16,17 +16,14 @@ module GF.Speech.SLF (slfPrinter,slfGraphvizPrinter,
|
|||||||
import GF.Data.Utilities
|
import GF.Data.Utilities
|
||||||
import GF.Grammar.CFG
|
import GF.Grammar.CFG
|
||||||
import GF.Speech.FiniteState
|
import GF.Speech.FiniteState
|
||||||
--import GF.Speech.CFG
|
|
||||||
import GF.Speech.CFGToFA
|
import GF.Speech.CFGToFA
|
||||||
import GF.Speech.PGFToCFG
|
import GF.Speech.PGFToCFG
|
||||||
import qualified GF.Data.Graphviz as Dot
|
import qualified GF.Data.Graphviz as Dot
|
||||||
import PGF
|
import PGF2
|
||||||
--import PGF.CId
|
|
||||||
|
|
||||||
import Control.Monad
|
import Control.Monad
|
||||||
import qualified Control.Monad.State as STM
|
import qualified Control.Monad.State as STM
|
||||||
import Data.Char (toUpper)
|
import Data.Char (toUpper)
|
||||||
--import Data.List
|
|
||||||
import Data.Maybe
|
import Data.Maybe
|
||||||
|
|
||||||
data SLFs = SLFs [(String,SLF)] SLF
|
data SLFs = SLFs [(String,SLF)] SLF
|
||||||
@@ -43,7 +40,7 @@ data SLFEdge = SLFEdge { eId :: Int, eStart :: Int, eEnd :: Int }
|
|||||||
|
|
||||||
type SLF_FA = FA State (Maybe CFSymbol) ()
|
type SLF_FA = FA State (Maybe CFSymbol) ()
|
||||||
|
|
||||||
mkFAs :: PGF -> CId -> (SLF_FA, [(String,SLF_FA)])
|
mkFAs :: PGF -> Concr -> (SLF_FA, [(String,SLF_FA)])
|
||||||
mkFAs pgf cnc = (slfStyleFA main, [(c,slfStyleFA n) | (c,n) <- subs])
|
mkFAs pgf cnc = (slfStyleFA main, [(c,slfStyleFA n) | (c,n) <- subs])
|
||||||
where MFA start subs = {- renameSubs $ -} cfgToMFA $ pgfToCFG pgf cnc
|
where MFA start subs = {- renameSubs $ -} cfgToMFA $ pgfToCFG pgf cnc
|
||||||
main = let (fa,s,f) = newFA_ in newTransition s f (NonTerminal start) fa
|
main = let (fa,s,f) = newFA_ in newTransition s f (NonTerminal start) fa
|
||||||
@@ -64,7 +61,7 @@ renameSubs (MFA start subs) = MFA (newName start) subs'
|
|||||||
-- * SLF graphviz printing (without sub-networks)
|
-- * SLF graphviz printing (without sub-networks)
|
||||||
--
|
--
|
||||||
|
|
||||||
slfGraphvizPrinter :: PGF -> CId -> String
|
slfGraphvizPrinter :: PGF -> Concr -> String
|
||||||
slfGraphvizPrinter pgf cnc
|
slfGraphvizPrinter pgf cnc
|
||||||
= prFAGraphviz $ gvFA $ slfStyleFA $ cfgToFA' $ pgfToCFG pgf cnc
|
= prFAGraphviz $ gvFA $ slfStyleFA $ cfgToFA' $ pgfToCFG pgf cnc
|
||||||
where
|
where
|
||||||
@@ -74,7 +71,7 @@ slfGraphvizPrinter pgf cnc
|
|||||||
-- * SLF graphviz printing (with sub-networks)
|
-- * SLF graphviz printing (with sub-networks)
|
||||||
--
|
--
|
||||||
|
|
||||||
slfSubGraphvizPrinter :: PGF -> CId -> String
|
slfSubGraphvizPrinter :: PGF -> Concr -> String
|
||||||
slfSubGraphvizPrinter pgf cnc = Dot.prGraphviz g
|
slfSubGraphvizPrinter pgf cnc = Dot.prGraphviz g
|
||||||
where (main, subs) = mkFAs pgf cnc
|
where (main, subs) = mkFAs pgf cnc
|
||||||
g = STM.evalState (liftM2 Dot.addSubGraphs ss m) [0..]
|
g = STM.evalState (liftM2 Dot.addSubGraphs ss m) [0..]
|
||||||
@@ -100,7 +97,7 @@ gvSLFFA n fa =
|
|||||||
-- * SLF printing (without sub-networks)
|
-- * SLF printing (without sub-networks)
|
||||||
--
|
--
|
||||||
|
|
||||||
slfPrinter :: PGF -> CId -> String
|
slfPrinter :: PGF -> Concr -> String
|
||||||
slfPrinter pgf cnc
|
slfPrinter pgf cnc
|
||||||
= prSLF $ automatonToSLF mkSLFNode $ slfStyleFA $ cfgToFA' $ pgfToCFG pgf cnc
|
= prSLF $ automatonToSLF mkSLFNode $ slfStyleFA $ cfgToFA' $ pgfToCFG pgf cnc
|
||||||
|
|
||||||
@@ -109,7 +106,7 @@ slfPrinter pgf cnc
|
|||||||
--
|
--
|
||||||
|
|
||||||
-- | Make a network with subnetworks in SLF
|
-- | Make a network with subnetworks in SLF
|
||||||
slfSubPrinter :: PGF -> CId -> String
|
slfSubPrinter :: PGF -> Concr -> String
|
||||||
slfSubPrinter pgf cnc = prSLFs slfs
|
slfSubPrinter pgf cnc = prSLFs slfs
|
||||||
where
|
where
|
||||||
(main,subs) = mkFAs pgf cnc
|
(main,subs) = mkFAs pgf cnc
|
||||||
|
|||||||
@@ -17,21 +17,15 @@ module GF.Speech.SRG (SRG(..), SRGRule(..), SRGAlt(..), SRGItem, SRGSymbol
|
|||||||
, lookupFM_
|
, lookupFM_
|
||||||
) where
|
) where
|
||||||
|
|
||||||
--import GF.Data.Operations
|
import PGF2
|
||||||
import GF.Data.Utilities
|
import GF.Data.Utilities
|
||||||
--import GF.Infra.Ident
|
|
||||||
import GF.Infra.Option
|
import GF.Infra.Option
|
||||||
import GF.Grammar.CFG
|
import GF.Grammar.CFG
|
||||||
import GF.Speech.PGFToCFG
|
import GF.Speech.PGFToCFG
|
||||||
--import GF.Data.Relation
|
|
||||||
--import GF.Speech.FiniteState
|
|
||||||
import GF.Speech.RegExp
|
import GF.Speech.RegExp
|
||||||
import GF.Speech.CFGToFA
|
import GF.Speech.CFGToFA
|
||||||
--import GF.Infra.Option
|
|
||||||
import PGF
|
|
||||||
|
|
||||||
import Data.List
|
import Data.List
|
||||||
--import Data.Maybe (fromMaybe, maybeToList)
|
|
||||||
import Data.Map (Map)
|
import Data.Map (Map)
|
||||||
import qualified Data.Map as Map
|
import qualified Data.Map as Map
|
||||||
import Data.Set (Set)
|
import Data.Set (Set)
|
||||||
@@ -62,16 +56,16 @@ type SRGSymbol = Symbol SRGNT Token
|
|||||||
-- | An SRG non-terminal. Category name and its number in the profile.
|
-- | An SRG non-terminal. Category name and its number in the profile.
|
||||||
type SRGNT = (Cat, Int)
|
type SRGNT = (Cat, Int)
|
||||||
|
|
||||||
ebnfPrinter :: Options -> PGF -> CId -> String
|
ebnfPrinter :: Options -> PGF -> Concr -> String
|
||||||
ebnfPrinter opts pgf cnc = prSRG opts $ makeSRG opts pgf cnc
|
ebnfPrinter opts pgf cnc = prSRG opts $ makeSRG opts pgf cnc
|
||||||
|
|
||||||
-- | Create a compact filtered non-left-recursive SRG.
|
-- | Create a compact filtered non-left-recursive SRG.
|
||||||
makeNonLeftRecursiveSRG :: Options -> PGF -> CId -> SRG
|
makeNonLeftRecursiveSRG :: Options -> PGF -> Concr -> SRG
|
||||||
makeNonLeftRecursiveSRG opts = makeSRG opts'
|
makeNonLeftRecursiveSRG opts = makeSRG opts'
|
||||||
where
|
where
|
||||||
opts' = setDefaultCFGTransform opts CFGNoLR True
|
opts' = setDefaultCFGTransform opts CFGNoLR True
|
||||||
|
|
||||||
makeSRG :: Options -> PGF -> CId -> SRG
|
makeSRG :: Options -> PGF -> Concr -> SRG
|
||||||
makeSRG opts = mkSRG cfgToSRG preprocess
|
makeSRG opts = mkSRG cfgToSRG preprocess
|
||||||
where
|
where
|
||||||
cfgToSRG cfg = [cfRulesToSRGRule rs | (_,rs) <- allRulesGrouped cfg]
|
cfgToSRG cfg = [cfRulesToSRGRule rs | (_,rs) <- allRulesGrouped cfg]
|
||||||
@@ -97,7 +91,7 @@ stats g = "Categories: " ++ show (countCats g)
|
|||||||
-}
|
-}
|
||||||
makeNonRecursiveSRG :: Options
|
makeNonRecursiveSRG :: Options
|
||||||
-> PGF
|
-> PGF
|
||||||
-> CId -- ^ Concrete syntax name.
|
-> Concr
|
||||||
-> SRG
|
-> SRG
|
||||||
makeNonRecursiveSRG opts = mkSRG cfgToSRG id
|
makeNonRecursiveSRG opts = mkSRG cfgToSRG id
|
||||||
where
|
where
|
||||||
@@ -105,17 +99,17 @@ makeNonRecursiveSRG opts = mkSRG cfgToSRG id
|
|||||||
where
|
where
|
||||||
MFA _ dfas = cfgToMFA cfg
|
MFA _ dfas = cfgToMFA cfg
|
||||||
dfaToSRGItem = mapRE dummySRGNT . minimizeRE . dfa2re
|
dfaToSRGItem = mapRE dummySRGNT . minimizeRE . dfa2re
|
||||||
dummyCFTerm = CFMeta (mkCId "dummy")
|
dummyCFTerm = CFMeta "dummy"
|
||||||
dummySRGNT = mapSymbol (\c -> (c,0)) id
|
dummySRGNT = mapSymbol (\c -> (c,0)) id
|
||||||
|
|
||||||
mkSRG :: (CFG -> [SRGRule]) -> (CFG -> CFG) -> PGF -> CId -> SRG
|
mkSRG :: (CFG -> [SRGRule]) -> (CFG -> CFG) -> PGF -> Concr -> SRG
|
||||||
mkSRG mkRules preprocess pgf cnc =
|
mkSRG mkRules preprocess pgf cnc =
|
||||||
SRG { srgName = showCId cnc,
|
SRG { srgName = concreteName cnc,
|
||||||
srgStartCat = cfgStartCat cfg,
|
srgStartCat = cfgStartCat cfg,
|
||||||
srgExternalCats = cfgExternalCats cfg,
|
srgExternalCats = cfgExternalCats cfg,
|
||||||
srgLanguage = languageCode pgf cnc,
|
srgLanguage = languageCode cnc,
|
||||||
srgRules = mkRules cfg }
|
srgRules = mkRules cfg }
|
||||||
where cfg = renameCats (showCId cnc) $ preprocess $ pgfToCFG pgf cnc
|
where cfg = renameCats (concreteName cnc) $ preprocess $ pgfToCFG pgf cnc
|
||||||
|
|
||||||
-- | Renames all external cats C to C_cat, and all internal cats C_X (where X is any string),
|
-- | Renames all external cats C to C_cat, and all internal cats C_X (where X is any string),
|
||||||
-- to C_N where N is an integer.
|
-- to C_N where N is an integer.
|
||||||
|
|||||||
@@ -18,7 +18,6 @@
|
|||||||
-----------------------------------------------------------------------------
|
-----------------------------------------------------------------------------
|
||||||
|
|
||||||
module GF.Speech.SRGS_ABNF (srgsAbnfPrinter, srgsAbnfNonRecursivePrinter) where
|
module GF.Speech.SRGS_ABNF (srgsAbnfPrinter, srgsAbnfNonRecursivePrinter) where
|
||||||
import Prelude hiding ((<>)) -- GHC 8.4.1 clash with Text.PrettyPrint
|
|
||||||
|
|
||||||
--import GF.Data.Utilities
|
--import GF.Data.Utilities
|
||||||
import GF.Infra.Option
|
import GF.Infra.Option
|
||||||
@@ -26,23 +25,21 @@ import GF.Grammar.CFG
|
|||||||
import GF.Speech.SISR as SISR
|
import GF.Speech.SISR as SISR
|
||||||
import GF.Speech.SRG
|
import GF.Speech.SRG
|
||||||
import GF.Speech.RegExp
|
import GF.Speech.RegExp
|
||||||
import PGF (PGF, CId)
|
import PGF2 (PGF,Concr)
|
||||||
|
|
||||||
--import Data.Char
|
|
||||||
import Data.List
|
import Data.List
|
||||||
import Data.Maybe
|
import Data.Maybe
|
||||||
import GF.Text.Pretty
|
import GF.Text.Pretty
|
||||||
--import Debug.Trace
|
|
||||||
|
|
||||||
width :: Int
|
width :: Int
|
||||||
width = 75
|
width = 75
|
||||||
|
|
||||||
srgsAbnfPrinter :: Options
|
srgsAbnfPrinter :: Options
|
||||||
-> PGF -> CId -> String
|
-> PGF -> Concr -> String
|
||||||
srgsAbnfPrinter opts pgf cnc = showDoc $ prABNF sisr $ makeNonLeftRecursiveSRG opts pgf cnc
|
srgsAbnfPrinter opts pgf cnc = showDoc $ prABNF sisr $ makeNonLeftRecursiveSRG opts pgf cnc
|
||||||
where sisr = flag optSISR opts
|
where sisr = flag optSISR opts
|
||||||
|
|
||||||
srgsAbnfNonRecursivePrinter :: Options -> PGF -> CId -> String
|
srgsAbnfNonRecursivePrinter :: Options -> PGF -> Concr -> String
|
||||||
srgsAbnfNonRecursivePrinter opts pgf cnc = showDoc $ prABNF Nothing $ makeNonRecursiveSRG opts pgf cnc
|
srgsAbnfNonRecursivePrinter opts pgf cnc = showDoc $ prABNF Nothing $ makeNonRecursiveSRG opts pgf cnc
|
||||||
|
|
||||||
showDoc = renderStyle (style { lineLength = width })
|
showDoc = renderStyle (style { lineLength = width })
|
||||||
|
|||||||
@@ -13,7 +13,7 @@ import GF.Grammar.CFG
|
|||||||
import GF.Speech.RegExp
|
import GF.Speech.RegExp
|
||||||
import GF.Speech.SISR as SISR
|
import GF.Speech.SISR as SISR
|
||||||
import GF.Speech.SRG
|
import GF.Speech.SRG
|
||||||
import PGF (PGF, CId, Token)
|
import PGF2 (PGF, Concr)
|
||||||
|
|
||||||
--import Control.Monad
|
--import Control.Monad
|
||||||
--import Data.Char (toUpper,toLower)
|
--import Data.Char (toUpper,toLower)
|
||||||
@@ -22,11 +22,11 @@ import Data.Maybe
|
|||||||
--import qualified Data.Map as Map
|
--import qualified Data.Map as Map
|
||||||
|
|
||||||
srgsXmlPrinter :: Options
|
srgsXmlPrinter :: Options
|
||||||
-> PGF -> CId -> String
|
-> PGF -> Concr -> String
|
||||||
srgsXmlPrinter opts pgf cnc = prSrgsXml sisr $ makeNonLeftRecursiveSRG opts pgf cnc
|
srgsXmlPrinter opts pgf cnc = prSrgsXml sisr $ makeNonLeftRecursiveSRG opts pgf cnc
|
||||||
where sisr = flag optSISR opts
|
where sisr = flag optSISR opts
|
||||||
|
|
||||||
srgsXmlNonRecursivePrinter :: Options -> PGF -> CId -> String
|
srgsXmlNonRecursivePrinter :: Options -> PGF -> Concr -> String
|
||||||
srgsXmlNonRecursivePrinter opts pgf cnc = prSrgsXml Nothing $ makeNonRecursiveSRG opts pgf cnc
|
srgsXmlNonRecursivePrinter opts pgf cnc = prSrgsXml Nothing $ makeNonRecursiveSRG opts pgf cnc
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -6,15 +6,8 @@
|
|||||||
-----------------------------------------------------------------------------
|
-----------------------------------------------------------------------------
|
||||||
module GF.Speech.VoiceXML (grammar2vxml) where
|
module GF.Speech.VoiceXML (grammar2vxml) where
|
||||||
|
|
||||||
--import GF.Data.Operations
|
|
||||||
--import GF.Data.Str (sstrV)
|
|
||||||
--import GF.Data.Utilities
|
|
||||||
import GF.Data.XML
|
import GF.Data.XML
|
||||||
--import GF.Infra.Ident
|
import PGF2
|
||||||
import PGF
|
|
||||||
import PGF.Internal
|
|
||||||
|
|
||||||
--import Control.Monad (liftM)
|
|
||||||
import Data.List (intersperse) -- isPrefixOf, find
|
import Data.List (intersperse) -- isPrefixOf, find
|
||||||
import qualified Data.Map as Map
|
import qualified Data.Map as Map
|
||||||
import Data.Maybe (fromMaybe)
|
import Data.Maybe (fromMaybe)
|
||||||
@@ -22,59 +15,45 @@ import Data.Maybe (fromMaybe)
|
|||||||
--import Debug.Trace
|
--import Debug.Trace
|
||||||
|
|
||||||
-- | the main function
|
-- | the main function
|
||||||
grammar2vxml :: PGF -> CId -> String
|
grammar2vxml :: PGF -> Concr -> String
|
||||||
grammar2vxml pgf cnc = showsXMLDoc (skel2vxml name language start skel qs) ""
|
grammar2vxml pgf cnc = showsXMLDoc (skel2vxml name mb_language start skel qs) ""
|
||||||
where skel = pgfSkeleton pgf
|
where skel = pgfSkeleton pgf
|
||||||
name = showCId cnc
|
name = concreteName cnc
|
||||||
qs = catQuestions pgf cnc (map fst skel)
|
qs = catQuestions cnc (map fst skel)
|
||||||
language = languageCode pgf cnc
|
mb_language = languageCode cnc
|
||||||
start = lookStartCat pgf
|
(_,start,_) = unType (startCat pgf)
|
||||||
|
|
||||||
--
|
--
|
||||||
-- * VSkeleton: a simple description of the abstract syntax.
|
-- * VSkeleton: a simple description of the abstract syntax.
|
||||||
--
|
--
|
||||||
|
|
||||||
type Skeleton = [(CId, [(CId, [CId])])]
|
type Skeleton = [(Cat, [(Fun, [Cat])])]
|
||||||
|
|
||||||
pgfSkeleton :: PGF -> Skeleton
|
pgfSkeleton :: PGF -> Skeleton
|
||||||
pgfSkeleton pgf = [(c,[(f,fst (catSkeleton (lookType (abstract pgf) f))) | (_,f) <- fs])
|
pgfSkeleton pgf = [(c,[(f,[cat | (_,_,ty) <- hypos, let (_,cat,_) = unType ty]) | f <- functionsByCat pgf c, Just (hypos,_,_) <- [fmap unType (functionType pgf f)]])
|
||||||
| (c,(_,fs,_)) <- Map.toList (cats (abstract pgf))]
|
| c <- categories pgf]
|
||||||
|
|
||||||
--
|
--
|
||||||
-- * Questions to ask
|
-- * Questions to ask
|
||||||
--
|
--
|
||||||
|
|
||||||
type CatQuestions = [(CId,String)]
|
type CatQuestions = [(Cat,String)]
|
||||||
|
|
||||||
catQuestions :: PGF -> CId -> [CId] -> CatQuestions
|
catQuestions :: Concr -> [Cat] -> CatQuestions
|
||||||
catQuestions pgf cnc cats = [(c,catQuestion pgf cnc c) | c <- cats]
|
catQuestions cnc cats = [(c,catQuestion cnc c) | c <- cats]
|
||||||
|
|
||||||
catQuestion :: PGF -> CId -> CId -> String
|
catQuestion :: Concr -> Cat -> String
|
||||||
catQuestion pgf cnc cat = showPrintName pgf cnc cat
|
catQuestion cnc cat = fromMaybe cat (printName cnc cat)
|
||||||
|
|
||||||
|
getCatQuestion :: Cat -> CatQuestions -> String
|
||||||
{-
|
|
||||||
lin :: StateGrammar -> String -> Err String
|
|
||||||
lin gr fun = do
|
|
||||||
tree <- string2treeErr gr fun
|
|
||||||
let ls = map unt $ linTree2strings noMark g c tree
|
|
||||||
case ls of
|
|
||||||
[] -> fail $ "No linearization of " ++ fun
|
|
||||||
l:_ -> return l
|
|
||||||
where c = cncId gr
|
|
||||||
g = stateGrammarST gr
|
|
||||||
unt = formatAsText
|
|
||||||
-}
|
|
||||||
|
|
||||||
getCatQuestion :: CId -> CatQuestions -> String
|
|
||||||
getCatQuestion c qs =
|
getCatQuestion c qs =
|
||||||
fromMaybe (error "No question for category " ++ showCId c) (lookup c qs)
|
fromMaybe (error "No question for category " ++ c) (lookup c qs)
|
||||||
|
|
||||||
--
|
--
|
||||||
-- * Generate VoiceXML
|
-- * Generate VoiceXML
|
||||||
--
|
--
|
||||||
|
|
||||||
skel2vxml :: String -> Maybe String -> CId -> Skeleton -> CatQuestions -> XML
|
skel2vxml :: String -> Maybe String -> Cat -> Skeleton -> CatQuestions -> XML
|
||||||
skel2vxml name language start skel qs =
|
skel2vxml name language start skel qs =
|
||||||
vxml language ([startForm] ++ concatMap (uncurry (catForms gr qs)) skel)
|
vxml language ([startForm] ++ concatMap (uncurry (catForms gr qs)) skel)
|
||||||
where
|
where
|
||||||
@@ -86,12 +65,12 @@ grammarURI :: String -> String
|
|||||||
grammarURI name = name ++ ".grxml"
|
grammarURI name = name ++ ".grxml"
|
||||||
|
|
||||||
|
|
||||||
catForms :: String -> CatQuestions -> CId -> [(CId, [CId])] -> [XML]
|
catForms :: String -> CatQuestions -> Cat -> [(Fun, [Cat])] -> [XML]
|
||||||
catForms gr qs cat fs =
|
catForms gr qs cat fs =
|
||||||
comments [showCId cat ++ " category."]
|
comments [cat ++ " category."]
|
||||||
++ [cat2form gr qs cat fs]
|
++ [cat2form gr qs cat fs]
|
||||||
|
|
||||||
cat2form :: String -> CatQuestions -> CId -> [(CId, [CId])] -> XML
|
cat2form :: String -> CatQuestions -> Cat -> [(Fun, [Cat])] -> XML
|
||||||
cat2form gr qs cat fs =
|
cat2form gr qs cat fs =
|
||||||
form (catFormId cat) $
|
form (catFormId cat) $
|
||||||
[var "old" Nothing,
|
[var "old" Nothing,
|
||||||
@@ -104,22 +83,22 @@ cat2form gr qs cat fs =
|
|||||||
++ concatMap (uncurry (fun2sub gr cat)) fs
|
++ concatMap (uncurry (fun2sub gr cat)) fs
|
||||||
++ [block [return_ ["term"]{-]-}]]
|
++ [block [return_ ["term"]{-]-}]]
|
||||||
|
|
||||||
fun2sub :: String -> CId -> CId -> [CId] -> [XML]
|
fun2sub :: String -> Cat -> Fun -> [Cat] -> [XML]
|
||||||
fun2sub gr cat fun args =
|
fun2sub gr cat fun args =
|
||||||
comments [showCId fun ++ " : ("
|
comments [fun ++ " : ("
|
||||||
++ concat (intersperse ", " (map showCId args))
|
++ concat (intersperse ", " args)
|
||||||
++ ") " ++ showCId cat] ++ ss
|
++ ") " ++ cat] ++ ss
|
||||||
where
|
where
|
||||||
ss = zipWith mkSub [0..] args
|
ss = zipWith mkSub [0..] args
|
||||||
mkSub n t = subdialog s [("src","#"++catFormId t),
|
mkSub n t = subdialog s [("src","#"++catFormId t),
|
||||||
("cond","term.name == "++string (showCId fun))]
|
("cond","term.name == "++string fun)]
|
||||||
[param "old" v,
|
[param "old" v,
|
||||||
filled [] [assign v (s++".term")]]
|
filled [] [assign v (s++".term")]]
|
||||||
where s = showCId fun ++ "_" ++ show n
|
where s = fun ++ "_" ++ show n
|
||||||
v = "term.args["++show n++"]"
|
v = "term.args["++show n++"]"
|
||||||
|
|
||||||
catFormId :: CId -> String
|
catFormId :: Cat -> String
|
||||||
catFormId c = showCId c ++ "_cat"
|
catFormId c = c ++ "_cat"
|
||||||
|
|
||||||
|
|
||||||
--
|
--
|
||||||
|
|||||||
@@ -15,7 +15,6 @@ stringOp good name = case name of
|
|||||||
"lexgreek" -> Just $ appLexer lexAGreek
|
"lexgreek" -> Just $ appLexer lexAGreek
|
||||||
"lexgreek2" -> Just $ appLexer lexAGreek2
|
"lexgreek2" -> Just $ appLexer lexAGreek2
|
||||||
"words" -> Just $ appLexer words
|
"words" -> Just $ appLexer words
|
||||||
"bind" -> Just $ appUnlexer (unwords . bindTok)
|
|
||||||
"unchars" -> Just $ appUnlexer concat
|
"unchars" -> Just $ appUnlexer concat
|
||||||
"unlextext" -> Just $ appUnlexer (unlexText . unquote . bindTok)
|
"unlextext" -> Just $ appUnlexer (unlexText . unquote . bindTok)
|
||||||
"unlexcode" -> Just $ appUnlexer unlexCode
|
"unlexcode" -> Just $ appUnlexer unlexCode
|
||||||
|
|||||||
@@ -39,7 +39,6 @@ allTransliterations = Map.fromList [
|
|||||||
("amharic",transAmharic),
|
("amharic",transAmharic),
|
||||||
("ancientgreek", transAncientGreek),
|
("ancientgreek", transAncientGreek),
|
||||||
("arabic", transArabic),
|
("arabic", transArabic),
|
||||||
("arabic_unvocalized", transArabicUnvoc),
|
|
||||||
("devanagari", transDevanagari),
|
("devanagari", transDevanagari),
|
||||||
("greek", transGreek),
|
("greek", transGreek),
|
||||||
("hebrew", transHebrew),
|
("hebrew", transHebrew),
|
||||||
@@ -179,13 +178,6 @@ transArabic = mkTransliteration "Arabic" allTrans allCodes where
|
|||||||
allCodes = [0x0621..0x062f] ++ [0x0630..0x063a] ++
|
allCodes = [0x0621..0x062f] ++ [0x0630..0x063a] ++
|
||||||
[0x0641..0x064f] ++ [0x0650..0x0657] ++ [0x0671,0x061f]
|
[0x0641..0x064f] ++ [0x0650..0x0657] ++ [0x0671,0x061f]
|
||||||
|
|
||||||
|
|
||||||
transArabicUnvoc :: Transliteration
|
|
||||||
transArabicUnvoc = transArabic{
|
|
||||||
invisible_chars = ["a","u","i","v2","o","V+","V-","a:"],
|
|
||||||
printname = "unvocalized Arabic"
|
|
||||||
}
|
|
||||||
|
|
||||||
transPersian :: Transliteration
|
transPersian :: Transliteration
|
||||||
transPersian = (mkTransliteration "Persian/Farsi" allTrans allCodes)
|
transPersian = (mkTransliteration "Persian/Farsi" allTrans allCodes)
|
||||||
{invisible_chars = ["a","u","i"]} where
|
{invisible_chars = ["a","u","i"]} where
|
||||||
|
|||||||
@@ -17,7 +17,7 @@ import GF.Grammar.Printer(ppParams,ppTerm,getAbs,TermPrintQual(..))
|
|||||||
import GF.Grammar.Parser(runP,pModDef)
|
import GF.Grammar.Parser(runP,pModDef)
|
||||||
import GF.Grammar.Lexer(Posn(..))
|
import GF.Grammar.Lexer(Posn(..))
|
||||||
import GF.Data.ErrM
|
import GF.Data.ErrM
|
||||||
import PGF.Internal(Literal(LStr))
|
import PGF2.Internal(Literal(LStr))
|
||||||
|
|
||||||
import SimpleEditor.Syntax as S
|
import SimpleEditor.Syntax as S
|
||||||
import SimpleEditor.JSON
|
import SimpleEditor.JSON
|
||||||
@@ -119,7 +119,7 @@ convCncJment (name,jment) =
|
|||||||
case jment of
|
case jment of
|
||||||
ResParam ops _ ->
|
ResParam ops _ ->
|
||||||
return $ Pa $ Param i (maybe "" (render . ppParams q . unLoc) ops)
|
return $ Pa $ Param i (maybe "" (render . ppParams q . unLoc) ops)
|
||||||
ResValue _ -> return Ignored
|
ResValue _ _ -> return Ignored
|
||||||
CncCat (Just (L _ typ)) Nothing Nothing pprn _ -> -- ignores printname !!
|
CncCat (Just (L _ typ)) Nothing Nothing pprn _ -> -- ignores printname !!
|
||||||
return $ LC $ Lincat i (render $ ppTerm q 0 typ)
|
return $ LC $ Lincat i (render $ ppTerm q 0 typ)
|
||||||
ResOper oltyp (Just lterm) -> return $ Op $ Oper lhs rhs
|
ResOper oltyp (Just lterm) -> return $ Op $ Oper lhs rhs
|
||||||
|
|||||||
@@ -1,553 +0,0 @@
|
|||||||
module ExampleDemo (Environ,initial,getNext, provideExample, testThis,mkFuncWithArg,searchGoodTree,isMeta)
|
|
||||||
where
|
|
||||||
|
|
||||||
import PGF
|
|
||||||
--import System.IO
|
|
||||||
import Data.List
|
|
||||||
--import Control.Monad
|
|
||||||
import qualified Data.Map as Map
|
|
||||||
--import qualified Data.IntMap as IntMap
|
|
||||||
import qualified Data.Set as Set
|
|
||||||
import Data.Maybe
|
|
||||||
--import System.Environment (getArgs)
|
|
||||||
import System.Random (RandomGen) --newStdGen
|
|
||||||
|
|
||||||
|
|
||||||
type MyType = CId -- name of the categories from the program
|
|
||||||
type ConcType = CId -- categories from the resource grammar, that we parse on
|
|
||||||
type MyFunc = CId -- functions that we need to implement
|
|
||||||
--type FuncWithArg = ((MyFunc, MyType), Expr) -- function with arguments
|
|
||||||
type InterInstr = [String] -- lincats that were generated but not written to the file
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
data FuncWithArg = FuncWithArg
|
|
||||||
{getName :: MyFunc, -- name of the function to generate
|
|
||||||
getType :: MyType, -- return type of the function
|
|
||||||
getTypeArgs :: [MyType] -- types of arguments
|
|
||||||
}
|
|
||||||
deriving (Show,Eq,Ord)
|
|
||||||
|
|
||||||
-- we assume that it's for English for the moment
|
|
||||||
|
|
||||||
|
|
||||||
type TypeMap = Map.Map MyType ConcType -- mapping found from a file
|
|
||||||
|
|
||||||
type ConcMap = Map.Map MyFunc Expr -- concrete expression after parsing
|
|
||||||
|
|
||||||
data Environ = Env {getTypeMap :: TypeMap, -- mapping between a category in the grammar and a concrete type from RGL
|
|
||||||
getConcMap :: ConcMap, -- concrete expression after parsing
|
|
||||||
getSigs :: Map.Map MyType [FuncWithArg], -- functions for which we have the concrete syntax already with args
|
|
||||||
getAll :: [FuncWithArg] -- all the functions with arguments
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
getNext :: Environ -> Environ -> ([MyFunc],[MyFunc])
|
|
||||||
getNext env example_env =
|
|
||||||
let sgs = getSigs env
|
|
||||||
allfuncs = getAll env
|
|
||||||
names = Set.fromList $ map getName $ concat $ Map.elems sgs
|
|
||||||
exampleable = filter (\x -> (isJust $ getNameExpr x env)
|
|
||||||
&&
|
|
||||||
(not $ Set.member x names) -- maybe drop this if you want to also rewrite from examples...
|
|
||||||
) $ map getName allfuncs
|
|
||||||
testeable = filter (\x -> (isJust $ getNameExpr x env )
|
|
||||||
&&
|
|
||||||
(Set.member x names)
|
|
||||||
) $ map getName allfuncs
|
|
||||||
|
|
||||||
in (exampleable,testeable)
|
|
||||||
|
|
||||||
|
|
||||||
provideExample :: RandomGen gen => gen -> Environ -> MyFunc -> PGF -> PGF -> Language -> Maybe (Expr,String)
|
|
||||||
provideExample gen env myfunc parsePGF pgfFile lang =
|
|
||||||
fmap giveExample $ getNameExpr myfunc env
|
|
||||||
where
|
|
||||||
giveExample e_ =
|
|
||||||
let newexpr = head $ generateRandomFromDepth gen pgfFile e_ (Just 5) -- change here with the new random generator
|
|
||||||
ty = getType $ head $ filter (\x -> getName x == myfunc) $ getAll env
|
|
||||||
embeddedExpr = maybe "" (\x -> ", as in: " ++ q (linearize pgfFile lang x)) (embedInStart (getAll env) (Map.fromList [(ty,e_)]))
|
|
||||||
lexpr = linearize pgfFile lang newexpr
|
|
||||||
q s = sq++s++sq
|
|
||||||
sq = "\""
|
|
||||||
in (newexpr,q lexpr ++ embeddedExpr)
|
|
||||||
-- question, you need the IO monad for the random generator, how to do otherwise ??
|
|
||||||
-- question can you make the expression bold/italic - somehow distinguishable from the rest ?
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
testThis :: Environ -> MyFunc -> PGF -> Language -> Maybe String
|
|
||||||
testThis env myfunc parsePGF lang =
|
|
||||||
fmap (linearize parsePGF lang . mapToResource env . llin env) $
|
|
||||||
getNameExpr myfunc env
|
|
||||||
|
|
||||||
|
|
||||||
-- we assume that even the functions linearized by the user will still be in getSigs along with their linearization
|
|
||||||
|
|
||||||
|
|
||||||
-- fill in the blancs of an expression that we want to linearize for testing purposes
|
|
||||||
---------------------------------------------------------------------------
|
|
||||||
|
|
||||||
llin :: Environ -> Expr -> Expr
|
|
||||||
llin env expr =
|
|
||||||
let
|
|
||||||
(id,args) = fromJust $ unApp expr
|
|
||||||
--cexpr = fromJust $ Map.lookup id (getConcMap env)
|
|
||||||
in
|
|
||||||
if any isMeta args
|
|
||||||
then let
|
|
||||||
sigs = concat $ Map.elems $ getSigs env
|
|
||||||
tys = findExprWhich sigs id
|
|
||||||
in replaceConcArg 1 tys expr env
|
|
||||||
else mkApp id $ map (llin env) args
|
|
||||||
|
|
||||||
|
|
||||||
-- argument of the meta variable to replace, list of arguments left, expression to replace, environment, current replace expression
|
|
||||||
replaceConcArg :: Int -> [MyType] -> Expr -> Environ -> Expr
|
|
||||||
replaceConcArg i [] expr env = expr
|
|
||||||
replaceConcArg i (t:ts) expr env = -- TO DO : insert randomness here !!
|
|
||||||
let ss = fromJust $ Map.lookup t $ getSigs env
|
|
||||||
args = filter (null . getTypeArgs) ss
|
|
||||||
finArg = if null args then let l = last ss in llin env (mkApp (getName l) [mkMeta j | j <- [1..(length $ getTypeArgs l)]])
|
|
||||||
else mkApp (getName $ last args) []
|
|
||||||
in
|
|
||||||
let newe = replaceOne i finArg expr
|
|
||||||
in replaceConcArg (i+1) ts newe env
|
|
||||||
|
|
||||||
-- replace a certain metavariable with a certain expression in another expression - return updated expression
|
|
||||||
replaceOne :: Int -> Expr -> Expr -> Expr
|
|
||||||
replaceOne i erep expr =
|
|
||||||
if isMeta expr && ((fromJust $ unMeta expr) == i)
|
|
||||||
then erep
|
|
||||||
else if isMeta expr then expr
|
|
||||||
else let (id,args) = fromJust $ unApp expr
|
|
||||||
in
|
|
||||||
mkApp id $ map (replaceOne i erep) args
|
|
||||||
|
|
||||||
|
|
||||||
findExprWhich :: [FuncWithArg] -> MyFunc -> [MyType]
|
|
||||||
findExprWhich lst f = getTypeArgs $ head $ filter (\x -> getName x == f) lst
|
|
||||||
|
|
||||||
|
|
||||||
mapToResource :: Environ -> Expr -> Expr
|
|
||||||
mapToResource env expr =
|
|
||||||
let (id,args) = maybe (error $ "tried to unwrap " ++ showExpr [] expr) (\x -> x) (unApp expr)
|
|
||||||
cmap = getConcMap env
|
|
||||||
cexp = maybe (error $ "didn't find " ++ showCId id ++ " in "++ show cmap) (\x -> x) (Map.lookup id cmap)
|
|
||||||
in
|
|
||||||
if null args then cexp
|
|
||||||
else let newargs = map (mapToResource env) args
|
|
||||||
in replaceAllArgs cexp 1 newargs
|
|
||||||
where
|
|
||||||
replaceAllArgs expr i [] = expr
|
|
||||||
replaceAllArgs expr i (x:xs) = replaceAllArgs (replaceOne i x expr) (i+1) xs
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
-----------------------------------------------
|
|
||||||
|
|
||||||
-- embed expression in another one from the start category
|
|
||||||
|
|
||||||
embedInStart :: [FuncWithArg] -> Map.Map MyType Expr -> Maybe Expr
|
|
||||||
embedInStart fss cs =
|
|
||||||
let currset = Map.toList cs
|
|
||||||
nextset = Map.fromList $ concat [ if elem myt (getTypeArgs farg)
|
|
||||||
then connectWithArg (myt,exp) farg else []
|
|
||||||
| (myt,exp) <- currset, farg <- fss]
|
|
||||||
nextmap = Map.union cs nextset
|
|
||||||
maybeExpr = Map.lookup startCateg nextset
|
|
||||||
in if isNothing maybeExpr then
|
|
||||||
if Map.size nextmap == Map.size cs then Nothing --error $ "could't build " ++ show startCateg ++ "with " ++ show fss
|
|
||||||
else embedInStart fss nextmap
|
|
||||||
else return $ fromJust maybeExpr
|
|
||||||
where
|
|
||||||
connectWithArg (myt,exp) farg =
|
|
||||||
let ind = head $ elemIndices myt (getTypeArgs farg)
|
|
||||||
in [(getType farg, mkApp (getName farg) $ [mkMeta i | i <- [1..ind]] ++ [exp] ++ [mkMeta i | i <- [(ind + 1)..((length $ getTypeArgs farg) - 1)]])]
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
-----------------------------------------------
|
|
||||||
{-
|
|
||||||
updateConcMap :: Environ -> MyFunc -> Expr -> Environ
|
|
||||||
updateConcMap env myf expr =
|
|
||||||
Env (getTypeMap env) (Map.insert myf expr (getConcMap env)) (getSigs env) (getAll env)
|
|
||||||
|
|
||||||
|
|
||||||
updateInterInstr :: Environ -> MyType -> FuncWithArg -> Environ
|
|
||||||
updateInterInstr env myt myf =
|
|
||||||
let ii = getSigs env
|
|
||||||
newInterInstr =
|
|
||||||
maybe (Map.insert myt [myf] ii) (\x -> Map.insert myt (myf:x) ii) $ Map.lookup myt ii
|
|
||||||
in Env (getTypeMap env) (getConcMap env) newInterInstr (getAll env)
|
|
||||||
|
|
||||||
|
|
||||||
putSignatures :: Environ -> [FuncWithArg] -> Environ
|
|
||||||
putSignatures env fss =
|
|
||||||
Env (getTypeMap env) (getConcMap env) (mkSigs fss) (getAll env)
|
|
||||||
|
|
||||||
|
|
||||||
updateEnv :: Environ -> FuncWithArg -> MyType -> Expr -> Environ
|
|
||||||
updateEnv env myf myt expr =
|
|
||||||
let ii = getSigs env
|
|
||||||
nn = getName myf
|
|
||||||
newInterInstr =
|
|
||||||
maybe (Map.insert myt [myf] ii) (\x -> Map.insert myt (myf:x) ii) $ Map.lookup myt ii
|
|
||||||
in Env (getTypeMap env) (Map.insert nn expr (getConcMap env)) newInterInstr (getAll env)
|
|
||||||
-}
|
|
||||||
|
|
||||||
mkSigs :: [FuncWithArg] -> Map.Map MyType [FuncWithArg]
|
|
||||||
mkSigs fss = Map.fromListWith (++) $ zip (map getType fss) (map (\x -> [x]) fss)
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
{------------------------------------
|
|
||||||
lang :: String
|
|
||||||
lang = "Eng"
|
|
||||||
|
|
||||||
|
|
||||||
parseLang :: Language
|
|
||||||
parseLang = fromJust $ readLanguage "ParseEng"
|
|
||||||
|
|
||||||
|
|
||||||
parsePGFfile :: String
|
|
||||||
parsePGFfile = "ParseEngAbs.pgf"
|
|
||||||
------------------------------------}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
searchGoodTree :: Environ -> Expr -> [Expr] -> IO (Maybe (Expr,Expr))
|
|
||||||
searchGoodTree env expr [] = return Nothing
|
|
||||||
searchGoodTree env expr (e:es) =
|
|
||||||
do val <- debugReplaceArgs expr e env
|
|
||||||
maybe (searchGoodTree env expr es) (\x -> return $ Just (x,e)) val
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
getNameExpr :: MyFunc -> Environ -> Maybe Expr
|
|
||||||
getNameExpr myfunc env =
|
|
||||||
let allfunc = filter (\x -> getName x == myfunc) $ getAll env
|
|
||||||
in
|
|
||||||
if null allfunc then Nothing
|
|
||||||
else getExpr (head allfunc) env
|
|
||||||
|
|
||||||
-- find an expression to generate where we have all the other elements available
|
|
||||||
getExpr :: FuncWithArg -> Environ -> Maybe Expr
|
|
||||||
getExpr farg env =
|
|
||||||
let tys = getTypeArgs farg
|
|
||||||
ctx = getSigs env
|
|
||||||
lst = getConcTypes ctx tys 1
|
|
||||||
in if (all isJust lst) then Just $ mkApp (getName farg) (map fromJust lst)
|
|
||||||
else Nothing
|
|
||||||
where getConcTypes context [] i = []
|
|
||||||
getConcTypes context (ty:types) i =
|
|
||||||
let pos = Map.lookup ty context
|
|
||||||
in
|
|
||||||
if isNothing pos || (null $ fromJust pos) then [Nothing]
|
|
||||||
else
|
|
||||||
let mm = last $ fromJust pos
|
|
||||||
mmargs = getTypeArgs mm
|
|
||||||
newi = i + length mmargs - 1
|
|
||||||
lst = getConcTypes (Map.insert ty (init $ (fromJust pos)) context) types (newi+1)
|
|
||||||
in
|
|
||||||
if (all isJust lst) then -- i..newi
|
|
||||||
(Just $ mkApp (getName mm) [mkMeta j | j <- [1..(length mmargs)]]) : lst
|
|
||||||
else [Nothing]
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
-- only covers simple expressions with meta variables, not the rest...
|
|
||||||
isGeneralizationOf :: Expr -> Expr -> Bool
|
|
||||||
isGeneralizationOf genExpr testExpr =
|
|
||||||
if isMeta genExpr then True
|
|
||||||
else if isMeta testExpr then False
|
|
||||||
else let genUnwrap = unApp genExpr
|
|
||||||
testUnwrap = unApp testExpr
|
|
||||||
in if isNothing genUnwrap || isNothing testUnwrap then False -- see if you can generalize here
|
|
||||||
else let (gencid, genargs) = fromJust genUnwrap
|
|
||||||
(testcid, testargs) = fromJust testUnwrap
|
|
||||||
in
|
|
||||||
(gencid == testcid) && (length genargs == length testargs)
|
|
||||||
&& (and [isGeneralizationOf g t | (g,t) <- (zip genargs testargs)])
|
|
||||||
|
|
||||||
{-do lst <- getConcTypes context types (i+1)
|
|
||||||
return $ mkMeta i : lst -}
|
|
||||||
|
|
||||||
debugReplaceArgs :: Expr -> Expr -> Environ -> IO (Maybe Expr)
|
|
||||||
debugReplaceArgs aexpr cexpr env =
|
|
||||||
if isNothing $ unApp aexpr then return Nothing
|
|
||||||
else if any isNothing $ map unApp $ snd $ fromJust $ unApp aexpr then return Nothing
|
|
||||||
else
|
|
||||||
let args = map (fst.fromJust.unApp) $ snd $ fromJust $ unApp aexpr
|
|
||||||
concExprs = map (\x -> fromJust $ Map.lookup x $ getConcMap env) args
|
|
||||||
in startReplace 1 cexpr concExprs
|
|
||||||
where
|
|
||||||
startReplace i cex [] = return $ Just cex
|
|
||||||
startReplace i cex (a:as) = do val <- debugReplaceConc cex i a
|
|
||||||
maybe ( --do putStrLn $ "didn't find "++ showExpr [] a ++ " in " ++showExpr [] cexpr
|
|
||||||
return Nothing)
|
|
||||||
(\x -> --do putStrLn $ "found it, the current expression is "++ showExpr [] x
|
|
||||||
startReplace (i+1) x as)
|
|
||||||
val
|
|
||||||
|
|
||||||
debugReplaceConc :: Expr -> Int -> Expr -> IO (Maybe Expr)
|
|
||||||
debugReplaceConc expr i e =
|
|
||||||
let (newe,isThere) = searchArg expr
|
|
||||||
in if isThere then return $ Just newe else return $ Nothing
|
|
||||||
where
|
|
||||||
searchArg e_ =
|
|
||||||
if isGeneralizationOf e e_ then (mkMeta i, True)
|
|
||||||
else maybe (e_,False) (\(cid,args) -> let repargs = map searchArg args
|
|
||||||
in (mkApp cid (map fst repargs), or $ map snd repargs)) $ unApp e_
|
|
||||||
|
|
||||||
|
|
||||||
{-
|
|
||||||
-- replaceArgs : Original expression to parse (from abstract syntax) -> Concrete expression (parsed)
|
|
||||||
replaceArgs :: Expr -> Expr -> Environ -> Maybe Expr
|
|
||||||
replaceArgs aexpr cexpr env =
|
|
||||||
if isNothing $ unApp aexpr then error $ "could't unwrap this "++ show aexpr
|
|
||||||
else if any isNothing $ map unApp $ snd $ fromJust $ unApp aexpr then error $ "couldn't unwrap more this : "++ show aexpr
|
|
||||||
else
|
|
||||||
let args = map (fst.fromJust.unApp) $ snd $ fromJust $ unApp aexpr
|
|
||||||
concExprs = map (\x -> fromJust $ Map.lookup x $ getConcMap env) args
|
|
||||||
in startReplace 1 cexpr concExprs
|
|
||||||
where
|
|
||||||
startReplace i cex [] = return cex
|
|
||||||
startReplace i cex (a:as) = maybe Nothing (\x -> startReplace (i+1) x as) $ replaceConc cex i a
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
replaceConc :: Expr -> Int -> Expr -> Maybe Expr
|
|
||||||
replaceConc expr i e =
|
|
||||||
let (newe,isThere) = searchArg expr
|
|
||||||
in if isThere then return newe else Nothing
|
|
||||||
where
|
|
||||||
searchArg e_ =
|
|
||||||
if isGeneralizationOf e e_ then (mkMeta i, True)
|
|
||||||
else maybe (e_,False) (\(cid,args) -> let repargs = map searchArg args
|
|
||||||
in (mkApp cid (map fst repargs), or $ map snd repargs)) $ unApp e_
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
writeResults :: Environ -> String -> IO ()
|
|
||||||
writeResults env fileName =
|
|
||||||
let cmap = getConcMap env
|
|
||||||
lincats = unlines $ map (\(x,y) -> "lincat " ++ showCId x ++ " = " ++ showCId y ++ " ; " ) $ Map.toList $ getTypeMap env
|
|
||||||
sigs = unlines $ map
|
|
||||||
(\x -> let n = getName x
|
|
||||||
no = length $ getTypeArgs x
|
|
||||||
oargs = unwords $ ("lin " ++ showCId n) : ["o"++show i | i <- [1..no]]
|
|
||||||
in (oargs ++ " = " ++ (simpleReplace $ showExpr [] $ fromJust $ Map.lookup n cmap) ++ " ; ")) $ concat $ Map.elems $ getSigs env
|
|
||||||
in
|
|
||||||
writeFile fileName ("\n" ++ lincats ++ "\n\n" ++ sigs)
|
|
||||||
|
|
||||||
|
|
||||||
simpleReplace :: String -> String
|
|
||||||
simpleReplace [] = []
|
|
||||||
simpleReplace ('?':xs) = 'o' : simpleReplace xs
|
|
||||||
simpleReplace (x:xs) = x : simpleReplace xs
|
|
||||||
-}
|
|
||||||
|
|
||||||
isMeta :: Expr -> Bool
|
|
||||||
isMeta = isJust.unMeta
|
|
||||||
|
|
||||||
-- works with utf-8 characters also, as it seems
|
|
||||||
|
|
||||||
|
|
||||||
mkFuncWithArg :: ((CId,CId),[CId]) -> FuncWithArg
|
|
||||||
mkFuncWithArg ((c1,c2),cids) = FuncWithArg c1 c2 cids
|
|
||||||
|
|
||||||
|
|
||||||
---------------------------------------------------------------------------------
|
|
||||||
|
|
||||||
initial :: TypeMap -> ConcMap -> [FuncWithArg] -> [FuncWithArg] -> Environ
|
|
||||||
initial tm cm fss allfs = Env tm cm (mkSigs fss) allfs
|
|
||||||
{-
|
|
||||||
testInit :: [FuncWithArg] -> Environ
|
|
||||||
testInit allfs = initial lTypes Map.empty [] allfs
|
|
||||||
|
|
||||||
lTypes = Map.fromList [(mkCId "Comment", mkCId "S"),(mkCId "Item", mkCId "NP"), (mkCId "Kind", mkCId "CN"), (mkCId "Quality", mkCId "AP")]
|
|
||||||
-}
|
|
||||||
startCateg = mkCId "Comment"
|
|
||||||
-- question about either to give the startcat or not ...
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
----------------------------------------------------------------------------------------------------------
|
|
||||||
{-
|
|
||||||
main =
|
|
||||||
do args <- getArgs
|
|
||||||
case args of
|
|
||||||
[pgfFile] ->
|
|
||||||
do pgf <- readPGF pgfFile
|
|
||||||
parsePGF <- readPGF parsePGFfile
|
|
||||||
fsWithArg <- forExample pgf
|
|
||||||
let funcsWithArg = map (map mkFuncWithArg) fsWithArg
|
|
||||||
let morpho = buildMorpho parsePGF parseLang
|
|
||||||
let fss = concat funcsWithArg
|
|
||||||
let fileName = takeWhile (/='.') pgfFile ++ lang ++ ".gf"
|
|
||||||
env <- start parsePGF pgf morpho (testInit fss) fss
|
|
||||||
putStrLn $ "Should I write the results to a file ? yes/no"
|
|
||||||
ans <-getLine
|
|
||||||
if ans == "yes" then do writeResults env fileName
|
|
||||||
putStrLn $ "Wrote file " ++ fileName
|
|
||||||
else return ()
|
|
||||||
_ -> fail "usage : Testing <path-to-pgf> "
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
start :: PGF -> PGF -> Morpho -> Environ -> [FuncWithArg] -> IO Environ
|
|
||||||
start parsePGF pgfFile morpho env lst =
|
|
||||||
do putStrLn "Do you want examples from another language ? (no/concrete syntax name otherwise)"
|
|
||||||
ans1 <- getLine
|
|
||||||
putStrLn "Do you want testing mode ? (yes/no)"
|
|
||||||
ans2 <- getLine
|
|
||||||
case (ans1,ans2) of
|
|
||||||
("no","no") -> do putStrLn "no extra language, just the abstract syntax tree"
|
|
||||||
interact env lst False Nothing
|
|
||||||
(_,"no") -> interact env lst False (readLanguage ans1)
|
|
||||||
("no","yes") -> do putStrLn "no extra language, just the abstract syntax tree"
|
|
||||||
interact env lst True Nothing
|
|
||||||
(_,"yes") -> interact env lst True (readLanguage ans1)
|
|
||||||
("no",_) -> do putStrLn "no extra language, just the abstract syntax tree"
|
|
||||||
putStrLn $ "I assume you don't want the testing mode ... "
|
|
||||||
interact env lst False Nothing
|
|
||||||
(_,_) -> do putStrLn $ "I assume you don't want the testing mode ... "
|
|
||||||
interact env lst False (readLanguage ans1)
|
|
||||||
where
|
|
||||||
|
|
||||||
interact environ [] func _ = return environ
|
|
||||||
interact environ (farg:fargs) boo otherLang =
|
|
||||||
do
|
|
||||||
maybeEnv <- basicInter farg otherLang environ boo
|
|
||||||
if isNothing maybeEnv then return environ
|
|
||||||
else interact (fromJust maybeEnv) fargs boo otherLang
|
|
||||||
|
|
||||||
basicInter farg js environ False =
|
|
||||||
let e_ = getExpr farg environ in
|
|
||||||
if isNothing e_ then return $ Just environ
|
|
||||||
else parseAndBuild farg js environ (getType farg) e_ Nothing
|
|
||||||
basicInter farg js environ True =
|
|
||||||
let (e_,e_test) = get2Expr farg environ in
|
|
||||||
if isNothing e_ then return $ Just environ
|
|
||||||
else if isNothing e_test then do putStrLn $ "not enough arguments "++ (showCId $ getName farg)
|
|
||||||
parseAndBuild farg js environ (getType farg) e_ Nothing
|
|
||||||
else parseAndBuild farg js environ (getType farg) e_ e_test
|
|
||||||
|
|
||||||
-- . head . generateRandomFrom gen2 pgfFile
|
|
||||||
parseAndBuild farg js environ ty e_ e_test =
|
|
||||||
do let expr = fromJust e_
|
|
||||||
gen1 <- newStdGen
|
|
||||||
gen2 <- newStdGen
|
|
||||||
let newexpr = head $ generateRandomFrom gen1 pgfFile expr
|
|
||||||
let embeddedExpr = maybe "***" (showExpr [] ) (embedInStart (getAll environ) (Map.fromList [(ty,expr)]))
|
|
||||||
let lexpr = if isNothing js then "" else "\n-- " ++ linearize pgfFile (fromJust js) newexpr ++ " --"
|
|
||||||
putStrLn $ "Give an example for " ++ (showExpr [] expr)
|
|
||||||
++ lexpr ++ "and now"
|
|
||||||
++ "\n\nas in " ++ embeddedExpr ++ "\n\n"
|
|
||||||
--
|
|
||||||
ex <- getLine
|
|
||||||
if (ex == ":q") then return Nothing
|
|
||||||
else
|
|
||||||
let ctype = fromJust $ Map.lookup (getType farg) (getTypeMap environ) in
|
|
||||||
do env' <- decypher farg ex expr environ (fromJust $ readType $ showCId ctype) e_test
|
|
||||||
return (Just env')
|
|
||||||
|
|
||||||
decypher farg ex expr environ ty e_test =
|
|
||||||
--do putStrLn $ "We need to parse " ++ ex ++ " as " ++ show ctype
|
|
||||||
let pTrees = parse parsePGF (fromJust $ readLanguage "ParseEng") ty ex in
|
|
||||||
pickTree farg expr environ ex e_test pTrees
|
|
||||||
|
|
||||||
-- putStrLn $ "And now for testing, \n is this also correct yes/no \n ## " ++ (linearize parsePGF parseLang $ mapToResource newenv $ llin newenv e_test) ++ " ##"
|
|
||||||
|
|
||||||
-- select the right tree among the options given by the parser
|
|
||||||
pickTree farg expr environ ex e_test [] =
|
|
||||||
let miswords = morphoMissing morpho (words ex)
|
|
||||||
in
|
|
||||||
if null miswords then do putStrLn $ "all words known, but some syntactic construction is not covered by the grammar..."
|
|
||||||
return environ
|
|
||||||
else do putStrLn $ "the following words are unknown, please add them to the lexicon: " ++ show miswords
|
|
||||||
return environ
|
|
||||||
pickTree farg expr environ ex e_test [tree] =
|
|
||||||
do val <- searchGoodTree environ expr [tree] -- maybe order here after the probabilities for better precision
|
|
||||||
maybe (do putStrLn $ "none of the trees is consistent with the rest of the grammar, please check arguments "
|
|
||||||
return environ)
|
|
||||||
(\(x,newtree) -> let newenv = updateEnv environ farg (getType farg) x in
|
|
||||||
do putStrLn $ "the result is "++showExpr [] x
|
|
||||||
newtestenv <- testTest newenv e_test -- question ? should it belong there - there is just one possibility of a tree...
|
|
||||||
return newenv) val
|
|
||||||
pickTree farg expr environ ex e_test parseTrees =
|
|
||||||
do putStrLn $ "There is more than one possibility, do you want to choose the right tree yourself ? yes/no "
|
|
||||||
putStr " >"
|
|
||||||
ans <- getLine
|
|
||||||
if ans == "yes" then do pTree <- chooseRightTree parseTrees
|
|
||||||
processTree farg environ expr pTree e_test
|
|
||||||
else processTree farg environ expr parseTrees e_test
|
|
||||||
|
|
||||||
-- introduce testing function, if it doesn't work, then reparse, take that tree
|
|
||||||
testTree envv e_test = return envv -- TO DO - add testing here
|
|
||||||
|
|
||||||
testTest envv Nothing = return envv
|
|
||||||
testTest envv (Just exxpr) = testTree envv exxpr
|
|
||||||
|
|
||||||
|
|
||||||
-- allows the user to pick his own tree
|
|
||||||
chooseRightTree trees = return trees -- TO DO - add something clever here
|
|
||||||
|
|
||||||
-- selects the tree from where one can abstract over the original arguments
|
|
||||||
processTree farg environ expr lsTrees e_test =
|
|
||||||
let trmes = if length lsTrees == 1 then "the tree is not consistent " else "none of the trees is consistent " in
|
|
||||||
do val <- searchGoodTree environ expr lsTrees
|
|
||||||
maybe (do putStrLn $ trmes ++ "with the rest of the grammar, please check arguments! "
|
|
||||||
return environ)
|
|
||||||
(\(x,newtree) -> let newenv = updateEnv environ farg (getType farg) x in
|
|
||||||
do putStrLn $ "the result is "++showExpr [] x
|
|
||||||
newtestenv <- testTest newenv e_test
|
|
||||||
return newenv) val
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
-------------------------------
|
|
||||||
|
|
||||||
get2Expr :: FuncWithArg -> Environ -> (Maybe Expr, Maybe Expr)
|
|
||||||
get2Expr farg env =
|
|
||||||
let tys = getTypeArgs farg
|
|
||||||
ctx = getSigs env
|
|
||||||
(lst1,lst2) = getConcTypes2 ctx tys 1
|
|
||||||
arg1 = if (all isJust lst1) then Just $ mkApp (getName farg) (map fromJust lst1) else Nothing
|
|
||||||
arg2 = if (all isJust lst2) then Just $ mkApp (getName farg) (map fromJust lst2) else Nothing
|
|
||||||
in if arg1 == arg2 then (arg1, Nothing)
|
|
||||||
else (arg1,arg2)
|
|
||||||
where
|
|
||||||
getConcTypes2 context [] i = ([],[])
|
|
||||||
getConcTypes2 context (ty:types) i =
|
|
||||||
let pos = Map.lookup ty context
|
|
||||||
in
|
|
||||||
if isNothing pos || (null $ fromJust pos) then ([Nothing],[Nothing])
|
|
||||||
else
|
|
||||||
let (mm,tt) = (last $ fromJust pos, head $ fromJust pos)
|
|
||||||
mmargs = getTypeArgs mm
|
|
||||||
newi = i + length mmargs - 1
|
|
||||||
(lst1,lst2) = getConcTypes2 (Map.insert ty (init (fromJust pos)) context) types (newi+1)
|
|
||||||
ttargs = getTypeArgs tt
|
|
||||||
newtti = i + length ttargs - 1
|
|
||||||
fstArg = if (all isJust lst1) then -- i..newi
|
|
||||||
(Just $ mkApp (getName mm) [mkMeta j | j <- [1..(length mmargs)]]) : lst1
|
|
||||||
else [Nothing]
|
|
||||||
sndArg = if (all isJust lst2) then
|
|
||||||
(Just $ mkApp (getName tt) [mkMeta j | j <- [1..(length ttargs)]]) : lst2
|
|
||||||
else [Nothing]
|
|
||||||
in
|
|
||||||
(fstArg,sndArg)
|
|
||||||
|
|
||||||
|
|
||||||
-}
|
|
||||||
|
|
||||||
@@ -1,128 +0,0 @@
|
|||||||
module ExampleService(cgiMain,cgiMain',newPGFCache) where
|
|
||||||
import System.Random(newStdGen)
|
|
||||||
import System.FilePath((</>),makeRelative)
|
|
||||||
import Data.Map(fromList)
|
|
||||||
import Data.Char(isDigit)
|
|
||||||
import Data.Maybe(fromJust)
|
|
||||||
import qualified Codec.Binary.UTF8.String as UTF8 (decodeString)
|
|
||||||
import PGF
|
|
||||||
import GF.Compile.ToAPI
|
|
||||||
import Network.CGI
|
|
||||||
import Text.JSON
|
|
||||||
import CGIUtils
|
|
||||||
import Cache
|
|
||||||
import qualified ExampleDemo as E
|
|
||||||
|
|
||||||
newPGFCache = newCache readPGF
|
|
||||||
|
|
||||||
|
|
||||||
cgiMain :: Cache PGF -> CGI CGIResult
|
|
||||||
cgiMain = handleErrors . handleCGIErrors . cgiMain' "." "."
|
|
||||||
|
|
||||||
cgiMain' root cwd cache =
|
|
||||||
do command <- getInp "command"
|
|
||||||
environ <- parseEnviron =<< getInp "state"
|
|
||||||
case command of
|
|
||||||
"possibilities" -> doPossibilities environ
|
|
||||||
"provide_example" -> doProvideExample root cwd cache environ
|
|
||||||
"abstract_example" -> doAbstractExample cwd cache environ
|
|
||||||
"test_function" -> doTestFunction cwd cache environ
|
|
||||||
_ -> throwCGIError 400 ("Unknown command: "++command) []
|
|
||||||
|
|
||||||
doPossibilities environ =
|
|
||||||
do example_environ <- parseEnviron =<< getInp "example_state"
|
|
||||||
outputJSONP (E.getNext environ example_environ)
|
|
||||||
|
|
||||||
doProvideExample root cwd cache environ =
|
|
||||||
do Just lang <- readInput "lang"
|
|
||||||
fun <- getCId "fun"
|
|
||||||
parsePGF <- readParsePGF cwd cache
|
|
||||||
let adjpath path = root</>makeRelative "/" (makeRelative root cwd</>path)
|
|
||||||
pgf <- liftIO . readCache cache . adjpath =<< getInp "grammar"
|
|
||||||
gen <- liftIO newStdGen
|
|
||||||
let Just (e,s) = E.provideExample gen environ fun parsePGF pgf lang
|
|
||||||
res = (showExpr [] e,s)
|
|
||||||
liftIO $ logError $ "proveExample ... = "++show res
|
|
||||||
outputJSONP res
|
|
||||||
|
|
||||||
doAbstractExample cwd cache environ =
|
|
||||||
do example <- getInp "input"
|
|
||||||
Just params <- readInput "params"
|
|
||||||
absstr <- getInp "abstract"
|
|
||||||
Just abs <- return $ readExpr absstr
|
|
||||||
liftIO $ logError $ "abstract = "++showExpr [] abs
|
|
||||||
Just cat <- readInput "cat"
|
|
||||||
let t = mkType [] cat []
|
|
||||||
parsePGF <- readParsePGF cwd cache
|
|
||||||
let lang:_ = languages parsePGF
|
|
||||||
ae <- liftIO $ abstractExample parsePGF environ lang t abs example
|
|
||||||
outputJSONP (fmap (\(e,_)->(exprToAPI (instExpMeta params e),e)) ae)
|
|
||||||
|
|
||||||
abstractExample parsePGF env lang cat abs example =
|
|
||||||
E.searchGoodTree env abs (parse parsePGF lang cat example)
|
|
||||||
|
|
||||||
doTestFunction cwd cache environ =
|
|
||||||
do fun <- getCId "fun"
|
|
||||||
parsePGF <- readParsePGF cwd cache
|
|
||||||
let lang:_ = languages parsePGF
|
|
||||||
Just txt <- return (E.testThis environ fun parsePGF lang)
|
|
||||||
outputJSONP txt
|
|
||||||
|
|
||||||
getCId :: String -> CGI CId
|
|
||||||
getCId name = maybe err return =<< fmap readCId (getInp name)
|
|
||||||
where err = throwCGIError 400 ("Bad "++name) []
|
|
||||||
{-
|
|
||||||
getLimit :: CGI Int
|
|
||||||
getLimit = maybe err return =<< readInput "limit"
|
|
||||||
where err = throwCGIError 400 "Missing/bad limit" []
|
|
||||||
-}
|
|
||||||
|
|
||||||
readParsePGF cwd cache =
|
|
||||||
do parsepgf <- getInp "parser"
|
|
||||||
liftIO $ readCache cache (cwd</>parsepgf)
|
|
||||||
|
|
||||||
parseEnviron s = do state <- liftIO $ readIO s
|
|
||||||
return $ environ state
|
|
||||||
|
|
||||||
getInp name = maybe err (return . UTF8.decodeString) =<< getInput name
|
|
||||||
where err = throwCGIError 400 ("Missing parameter: "++name) []
|
|
||||||
|
|
||||||
|
|
||||||
instance JSON CId where
|
|
||||||
showJSON = showJSON . show
|
|
||||||
readJSON = (readResult =<<) . readJSON
|
|
||||||
|
|
||||||
instance JSON Expr where
|
|
||||||
showJSON = showJSON . showExpr []
|
|
||||||
readJSON = (m2r . readExpr =<<) . readJSON
|
|
||||||
|
|
||||||
m2r = maybe (Error "read failed") Ok
|
|
||||||
|
|
||||||
readResult s = case reads s of
|
|
||||||
(x,r):_ | lex r==[("","")] -> Ok x
|
|
||||||
_ -> Error "read failed"
|
|
||||||
|
|
||||||
--------------------------------------------------------------------------------
|
|
||||||
-- cat lincat fun lin fun cat cat
|
|
||||||
environ :: ([(CId, CId)],[(CId, Expr)],[((CId, CId), [CId])]) -> E.Environ
|
|
||||||
environ (lincats,lins0,funs) =
|
|
||||||
E.initial (fromList lincats) concmap fs allfs
|
|
||||||
where
|
|
||||||
concmap = fromList lins
|
|
||||||
allfs = map E.mkFuncWithArg funs
|
|
||||||
fs = [E.mkFuncWithArg f | f@((fn,_),_)<-funs, fn `elem` cns]
|
|
||||||
cns = map fst lins
|
|
||||||
lins = filter (not . E.isMeta .snd) lins0
|
|
||||||
|
|
||||||
|
|
||||||
instExpMeta :: [CId] -> Expr -> Expr
|
|
||||||
instExpMeta ps = fromJust . readExpr . instMeta ps . showExpr []
|
|
||||||
|
|
||||||
instMeta :: [CId] -> String -> String
|
|
||||||
instMeta ps s =
|
|
||||||
case break (=='?') s of
|
|
||||||
(s1,'?':s2) ->
|
|
||||||
case span isDigit s2 of
|
|
||||||
(s21@(_:_),s22) -> s1++show (ps!!(read s21-1))++instMeta ps s22
|
|
||||||
("",s22) -> s1++'?':instMeta ps s22
|
|
||||||
(_,_) -> s
|
|
||||||
@@ -1,15 +0,0 @@
|
|||||||
{-# LANGUAGE CPP #-}
|
|
||||||
import Control.Concurrent(forkIO)
|
|
||||||
import Network.FastCGI(runFastCGI,runFastCGIConcurrent')
|
|
||||||
import ExampleService(cgiMain,newPGFCache)
|
|
||||||
|
|
||||||
main = do --stderrToFile logFile
|
|
||||||
fcgiMain =<< newPGFCache
|
|
||||||
|
|
||||||
|
|
||||||
fcgiMain cache =
|
|
||||||
#ifndef mingw32_HOST_OS
|
|
||||||
runFastCGIConcurrent' forkIO 100 (cgiMain cache)
|
|
||||||
#else
|
|
||||||
runFastCGI (cgiMain cache)
|
|
||||||
#endif
|
|
||||||
@@ -1,25 +0,0 @@
|
|||||||
Name: gf-exb
|
|
||||||
Version: 1.0
|
|
||||||
Cabal-version: >= 1.8
|
|
||||||
Build-type: Simple
|
|
||||||
License: GPL
|
|
||||||
Synopsis: Example-based grammar writing for the Grammatical Framework
|
|
||||||
|
|
||||||
executable exb.fcgi
|
|
||||||
main-is: exb-fcgi.hs
|
|
||||||
Hs-source-dirs: . ../server ../compiler ../runtime/haskell
|
|
||||||
other-modules: ExampleService ExampleDemo
|
|
||||||
FastCGIUtils Cache GF.Compile.ToAPI
|
|
||||||
-- and a lot more...
|
|
||||||
ghc-options: -threaded
|
|
||||||
if impl(ghc>=7.0)
|
|
||||||
ghc-options: -rtsopts
|
|
||||||
|
|
||||||
build-depends: base >=4.2 && <5, json, cgi, fastcgi, random,
|
|
||||||
containers, old-time, directory, bytestring, utf8-string,
|
|
||||||
pretty, array, mtl, fst, filepath
|
|
||||||
|
|
||||||
if os(windows)
|
|
||||||
ghc-options: -optl-mwindows
|
|
||||||
else
|
|
||||||
build-depends: unix
|
|
||||||
@@ -1,20 +0,0 @@
|
|||||||
|
|
||||||
Editor improvements for example-based grammar writing:
|
|
||||||
+ Remove the same language from the example language menu
|
|
||||||
+ Send the other language environment to getNext
|
|
||||||
- Compile a new .pgf automatically when needed
|
|
||||||
- Update buttons automatically when functions are added or removed
|
|
||||||
- Switch over to using AbsParadigmsEng.pgf instead of the old exprToAPI function
|
|
||||||
|
|
||||||
Editor support for guided construction of linearization functions
|
|
||||||
- enter api expressions by parsing them with AbsParadigmsEng.pgf in minibar
|
|
||||||
- replace simpleParseInput with one that accepts quoted string literals
|
|
||||||
- use lexcode/unlexcode in minibar
|
|
||||||
- better support for literals in minibar (completion info from the PGF
|
|
||||||
library should indicate if literals are acceptable)
|
|
||||||
|
|
||||||
Server support for example-based grammar writing:
|
|
||||||
- Change getNext to use info from the example language
|
|
||||||
- Random generator restricted to defined functions
|
|
||||||
|
|
||||||
- More testing
|
|
||||||
@@ -88,14 +88,13 @@ libpgf_la_SOURCES = \
|
|||||||
pgf/graphviz.c \
|
pgf/graphviz.c \
|
||||||
pgf/aligner.c \
|
pgf/aligner.c \
|
||||||
pgf/pgf.c \
|
pgf/pgf.c \
|
||||||
pgf/pgf.h
|
pgf/pgf.h \
|
||||||
libpgf_la_LDFLAGS = -no-undefined
|
libpgf_la_LDFLAGS = "-no-undefined"
|
||||||
libpgf_la_LIBADD = libgu.la
|
libpgf_la_LIBADD = libgu.la
|
||||||
|
|
||||||
libsg_la_SOURCES = \
|
libsg_la_SOURCES = \
|
||||||
sg/sqlite3Btree.c \
|
sg/sqlite3Btree.c \
|
||||||
sg/sg.c
|
sg/sg.c
|
||||||
libsg_la_LDFLAGS = -no-undefined
|
|
||||||
libsg_la_LIBADD = libgu.la libpgf.la
|
libsg_la_LIBADD = libgu.la libpgf.la
|
||||||
|
|
||||||
bin_PROGRAMS =
|
bin_PROGRAMS =
|
||||||
|
|||||||
@@ -23,14 +23,6 @@
|
|||||||
|
|
||||||
#define restrict __restrict
|
#define restrict __restrict
|
||||||
|
|
||||||
#elif defined(__MINGW32__)
|
|
||||||
|
|
||||||
#define GU_API_DECL
|
|
||||||
#define GU_API
|
|
||||||
|
|
||||||
#define GU_INTERNAL_DECL
|
|
||||||
#define GU_INTERNAL
|
|
||||||
|
|
||||||
#else
|
#else
|
||||||
|
|
||||||
#define GU_API_DECL
|
#define GU_API_DECL
|
||||||
@@ -38,9 +30,7 @@
|
|||||||
|
|
||||||
#define GU_INTERNAL_DECL __attribute__ ((visibility ("hidden")))
|
#define GU_INTERNAL_DECL __attribute__ ((visibility ("hidden")))
|
||||||
#define GU_INTERNAL __attribute__ ((visibility ("hidden")))
|
#define GU_INTERNAL __attribute__ ((visibility ("hidden")))
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// end MSVC workaround
|
// end MSVC workaround
|
||||||
|
|
||||||
#include <stddef.h>
|
#include <stddef.h>
|
||||||
|
|||||||
@@ -7,6 +7,9 @@
|
|||||||
|
|
||||||
typedef struct GuMapData GuMapData;
|
typedef struct GuMapData GuMapData;
|
||||||
|
|
||||||
|
#define SKIP_DELETED 1
|
||||||
|
#define SKIP_NONE 2
|
||||||
|
|
||||||
struct GuMapData {
|
struct GuMapData {
|
||||||
uint8_t* keys;
|
uint8_t* keys;
|
||||||
uint8_t* values;
|
uint8_t* values;
|
||||||
@@ -19,6 +22,7 @@ struct GuMap {
|
|||||||
GuHasher* hasher;
|
GuHasher* hasher;
|
||||||
size_t key_size;
|
size_t key_size;
|
||||||
size_t value_size;
|
size_t value_size;
|
||||||
|
size_t cell_size; // cell_size = GU_MAX(value_size,sizeof(uint8_t))
|
||||||
const void* default_value;
|
const void* default_value;
|
||||||
GuMapData data;
|
GuMapData data;
|
||||||
|
|
||||||
@@ -30,9 +34,7 @@ gu_map_finalize(GuFinalizer* fin)
|
|||||||
{
|
{
|
||||||
GuMap* map = gu_container(fin, GuMap, fin);
|
GuMap* map = gu_container(fin, GuMap, fin);
|
||||||
gu_mem_buf_free(map->data.keys);
|
gu_mem_buf_free(map->data.keys);
|
||||||
if (map->value_size) {
|
gu_mem_buf_free(map->data.values);
|
||||||
gu_mem_buf_free(map->data.values);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static const GuWord gu_map_empty_key = 0;
|
static const GuWord gu_map_empty_key = 0;
|
||||||
@@ -68,7 +70,7 @@ gu_map_entry_is_free(GuMap* map, GuMapData* data, size_t idx)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static bool
|
static bool
|
||||||
gu_map_lookup(GuMap* map, const void* key, size_t* idx_out)
|
gu_map_lookup(GuMap* map, const void* key, uint8_t del, size_t* idx_out)
|
||||||
{
|
{
|
||||||
size_t n = map->data.n_entries;
|
size_t n = map->data.n_entries;
|
||||||
if (map->hasher == gu_addr_hasher) {
|
if (map->hasher == gu_addr_hasher) {
|
||||||
@@ -78,13 +80,17 @@ gu_map_lookup(GuMap* map, const void* key, size_t* idx_out)
|
|||||||
while (true) {
|
while (true) {
|
||||||
const void* entry_key =
|
const void* entry_key =
|
||||||
((const void**)map->data.keys)[idx];
|
((const void**)map->data.keys)[idx];
|
||||||
|
|
||||||
if (entry_key == NULL && map->data.zero_idx != idx) {
|
if (entry_key == NULL && map->data.zero_idx != idx) {
|
||||||
*idx_out = idx;
|
if (map->data.values[idx * map->cell_size] != del) { //skip deleted
|
||||||
return false;
|
*idx_out = idx;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
} else if (entry_key == key) {
|
} else if (entry_key == key) {
|
||||||
*idx_out = idx;
|
*idx_out = idx;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
idx = (idx + offset) % n;
|
idx = (idx + offset) % n;
|
||||||
}
|
}
|
||||||
} else if (map->hasher == gu_word_hasher) {
|
} else if (map->hasher == gu_word_hasher) {
|
||||||
@@ -156,33 +162,18 @@ gu_map_resize(GuMap* map, size_t req_entries)
|
|||||||
size_t key_size = map->key_size;
|
size_t key_size = map->key_size;
|
||||||
size_t key_alloc = 0;
|
size_t key_alloc = 0;
|
||||||
data->keys = gu_mem_buf_alloc(req_entries * key_size, &key_alloc);
|
data->keys = gu_mem_buf_alloc(req_entries * key_size, &key_alloc);
|
||||||
|
memset(data->keys, 0, key_alloc);
|
||||||
|
|
||||||
size_t value_size = map->value_size;
|
|
||||||
size_t value_alloc = 0;
|
size_t value_alloc = 0;
|
||||||
if (value_size) {
|
size_t cell_size = map->cell_size;
|
||||||
data->values = gu_mem_buf_alloc(req_entries * value_size,
|
data->values = gu_mem_buf_alloc(req_entries * cell_size, &value_alloc);
|
||||||
&value_alloc);
|
memset(data->values, 0, value_alloc);
|
||||||
memset(data->values, 0, value_alloc);
|
|
||||||
}
|
|
||||||
|
|
||||||
data->n_entries = gu_twin_prime_inf(value_size ?
|
|
||||||
GU_MIN(key_alloc / key_size,
|
|
||||||
value_alloc / value_size)
|
|
||||||
: key_alloc / key_size);
|
|
||||||
if (map->hasher == gu_addr_hasher) {
|
|
||||||
for (size_t i = 0; i < data->n_entries; i++) {
|
|
||||||
((const void**)data->keys)[i] = NULL;
|
|
||||||
}
|
|
||||||
} else if (map->hasher == gu_string_hasher) {
|
|
||||||
for (size_t i = 0; i < data->n_entries; i++) {
|
|
||||||
((GuString*)data->keys)[i] = NULL;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
memset(data->keys, 0, key_alloc);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
data->n_entries = gu_twin_prime_inf(
|
||||||
|
GU_MIN(key_alloc / key_size,
|
||||||
|
value_alloc / cell_size));
|
||||||
gu_assert(data->n_entries > data->n_occupied);
|
gu_assert(data->n_entries > data->n_occupied);
|
||||||
|
|
||||||
data->n_occupied = 0;
|
data->n_occupied = 0;
|
||||||
data->zero_idx = SIZE_MAX;
|
data->zero_idx = SIZE_MAX;
|
||||||
|
|
||||||
@@ -196,16 +187,14 @@ gu_map_resize(GuMap* map, size_t req_entries)
|
|||||||
} else if (map->hasher == gu_string_hasher) {
|
} else if (map->hasher == gu_string_hasher) {
|
||||||
old_key = (void*) *(GuString*)old_key;
|
old_key = (void*) *(GuString*)old_key;
|
||||||
}
|
}
|
||||||
void* old_value = &old_data.values[i * value_size];
|
void* old_value = &old_data.values[i * cell_size];
|
||||||
|
|
||||||
memcpy(gu_map_insert(map, old_key),
|
memcpy(gu_map_insert(map, old_key),
|
||||||
old_value, map->value_size);
|
old_value, map->value_size);
|
||||||
}
|
}
|
||||||
|
|
||||||
gu_mem_buf_free(old_data.keys);
|
gu_mem_buf_free(old_data.keys);
|
||||||
if (value_size) {
|
gu_mem_buf_free(old_data.values);
|
||||||
gu_mem_buf_free(old_data.values);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -226,9 +215,9 @@ GU_API void*
|
|||||||
gu_map_find(GuMap* map, const void* key)
|
gu_map_find(GuMap* map, const void* key)
|
||||||
{
|
{
|
||||||
size_t idx;
|
size_t idx;
|
||||||
bool found = gu_map_lookup(map, key, &idx);
|
bool found = gu_map_lookup(map, key, SKIP_DELETED, &idx);
|
||||||
if (found) {
|
if (found) {
|
||||||
return &map->data.values[idx * map->value_size];
|
return &map->data.values[idx * map->cell_size];
|
||||||
}
|
}
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
@@ -244,7 +233,7 @@ GU_API const void*
|
|||||||
gu_map_find_key(GuMap* map, const void* key)
|
gu_map_find_key(GuMap* map, const void* key)
|
||||||
{
|
{
|
||||||
size_t idx;
|
size_t idx;
|
||||||
bool found = gu_map_lookup(map, key, &idx);
|
bool found = gu_map_lookup(map, key, SKIP_DELETED, &idx);
|
||||||
if (found) {
|
if (found) {
|
||||||
return &map->data.keys[idx * map->key_size];
|
return &map->data.keys[idx * map->key_size];
|
||||||
}
|
}
|
||||||
@@ -255,17 +244,17 @@ GU_API bool
|
|||||||
gu_map_has(GuMap* ht, const void* key)
|
gu_map_has(GuMap* ht, const void* key)
|
||||||
{
|
{
|
||||||
size_t idx;
|
size_t idx;
|
||||||
return gu_map_lookup(ht, key, &idx);
|
return gu_map_lookup(ht, key, SKIP_DELETED, &idx);
|
||||||
}
|
}
|
||||||
|
|
||||||
GU_API void*
|
GU_API void*
|
||||||
gu_map_insert(GuMap* map, const void* key)
|
gu_map_insert(GuMap* map, const void* key)
|
||||||
{
|
{
|
||||||
size_t idx;
|
size_t idx;
|
||||||
bool found = gu_map_lookup(map, key, &idx);
|
bool found = gu_map_lookup(map, key, SKIP_NONE, &idx);
|
||||||
if (!found) {
|
if (!found) {
|
||||||
if (gu_map_maybe_resize(map)) {
|
if (gu_map_maybe_resize(map)) {
|
||||||
found = gu_map_lookup(map, key, &idx);
|
found = gu_map_lookup(map, key, SKIP_NONE, &idx);
|
||||||
gu_assert(!found);
|
gu_assert(!found);
|
||||||
}
|
}
|
||||||
if (map->hasher == gu_addr_hasher) {
|
if (map->hasher == gu_addr_hasher) {
|
||||||
@@ -277,7 +266,7 @@ gu_map_insert(GuMap* map, const void* key)
|
|||||||
key, map->key_size);
|
key, map->key_size);
|
||||||
}
|
}
|
||||||
if (map->default_value) {
|
if (map->default_value) {
|
||||||
memcpy(&map->data.values[idx * map->value_size],
|
memcpy(&map->data.values[idx * map->cell_size],
|
||||||
map->default_value, map->value_size);
|
map->default_value, map->value_size);
|
||||||
}
|
}
|
||||||
if (gu_map_entry_is_free(map, &map->data, idx)) {
|
if (gu_map_entry_is_free(map, &map->data, idx)) {
|
||||||
@@ -286,7 +275,32 @@ gu_map_insert(GuMap* map, const void* key)
|
|||||||
}
|
}
|
||||||
map->data.n_occupied++;
|
map->data.n_occupied++;
|
||||||
}
|
}
|
||||||
return &map->data.values[idx * map->value_size];
|
return &map->data.values[idx * map->cell_size];
|
||||||
|
}
|
||||||
|
|
||||||
|
GU_API void
|
||||||
|
gu_map_delete(GuMap* map, const void* key)
|
||||||
|
{
|
||||||
|
size_t idx;
|
||||||
|
bool found = gu_map_lookup(map, key, SKIP_NONE, &idx);
|
||||||
|
if (found) {
|
||||||
|
if (map->hasher == gu_addr_hasher) {
|
||||||
|
((const void**)map->data.keys)[idx] = NULL;
|
||||||
|
} else if (map->hasher == gu_string_hasher) {
|
||||||
|
((GuString*)map->data.keys)[idx] = NULL;
|
||||||
|
} else {
|
||||||
|
memset(&map->data.keys[idx * map->key_size],
|
||||||
|
0, map->key_size);
|
||||||
|
}
|
||||||
|
map->data.values[idx * map->cell_size] = SKIP_DELETED;
|
||||||
|
|
||||||
|
if (gu_map_buf_is_zero(&map->data.keys[idx * map->key_size],
|
||||||
|
map->key_size)) {
|
||||||
|
map->data.zero_idx = SIZE_MAX;
|
||||||
|
}
|
||||||
|
|
||||||
|
map->data.n_occupied--;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
GU_API void
|
GU_API void
|
||||||
@@ -297,7 +311,7 @@ gu_map_iter(GuMap* map, GuMapItor* itor, GuExn* err)
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
const void* key = &map->data.keys[i * map->key_size];
|
const void* key = &map->data.keys[i * map->key_size];
|
||||||
void* value = &map->data.values[i * map->value_size];
|
void* value = &map->data.values[i * map->cell_size];
|
||||||
if (map->hasher == gu_addr_hasher) {
|
if (map->hasher == gu_addr_hasher) {
|
||||||
key = *(const void* const*) key;
|
key = *(const void* const*) key;
|
||||||
} else if (map->hasher == gu_string_hasher) {
|
} else if (map->hasher == gu_string_hasher) {
|
||||||
@@ -307,47 +321,30 @@ gu_map_iter(GuMap* map, GuMapItor* itor, GuExn* err)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
typedef struct {
|
GU_API bool
|
||||||
GuEnum en;
|
gu_map_next(GuMap* map, size_t* pi, void** pkey, void* pvalue)
|
||||||
GuMap* ht;
|
|
||||||
size_t i;
|
|
||||||
GuMapKeyValue x;
|
|
||||||
} GuMapEnum;
|
|
||||||
|
|
||||||
static void
|
|
||||||
gu_map_enum_next(GuEnum* self, void* to, GuPool* pool)
|
|
||||||
{
|
{
|
||||||
*((GuMapKeyValue**) to) = NULL;
|
while (*pi < map->data.n_entries) {
|
||||||
|
if (gu_map_entry_is_free(map, &map->data, *pi)) {
|
||||||
size_t i;
|
(*pi)++;
|
||||||
GuMapEnum* en = (GuMapEnum*) self;
|
|
||||||
for (i = en->i; i < en->ht->data.n_entries; i++) {
|
|
||||||
if (gu_map_entry_is_free(en->ht, &en->ht->data, i)) {
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
en->x.key = &en->ht->data.keys[i * en->ht->key_size];
|
|
||||||
en->x.value = &en->ht->data.values[i * en->ht->value_size];
|
*pkey = &map->data.keys[*pi * map->key_size];
|
||||||
if (en->ht->hasher == gu_addr_hasher) {
|
if (map->hasher == gu_addr_hasher) {
|
||||||
en->x.key = *(const void* const*) en->x.key;
|
*pkey = *(void**) *pkey;
|
||||||
} else if (en->ht->hasher == gu_string_hasher) {
|
} else if (map->hasher == gu_string_hasher) {
|
||||||
en->x.key = *(GuString*) en->x.key;
|
*pkey = *(void**) *pkey;
|
||||||
}
|
}
|
||||||
|
|
||||||
*((GuMapKeyValue**) to) = &en->x;
|
memcpy(pvalue, &map->data.values[*pi * map->cell_size],
|
||||||
break;
|
map->value_size);
|
||||||
}
|
|
||||||
|
|
||||||
en->i = i+1;
|
|
||||||
}
|
|
||||||
|
|
||||||
GU_API GuEnum*
|
(*pi)++;
|
||||||
gu_map_enum(GuMap* ht, GuPool* pool)
|
return true;
|
||||||
{
|
}
|
||||||
GuMapEnum* en = gu_new(GuMapEnum, pool);
|
|
||||||
en->en.next = gu_map_enum_next;
|
return false;
|
||||||
en->ht = ht;
|
|
||||||
en->i = 0;
|
|
||||||
return &en->en;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
GU_API size_t
|
GU_API size_t
|
||||||
@@ -363,8 +360,6 @@ gu_map_count(GuMap* map)
|
|||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
|
|
||||||
static const uint8_t gu_map_no_values[1] = { 0 };
|
|
||||||
|
|
||||||
GU_API GuMap*
|
GU_API GuMap*
|
||||||
gu_make_map(size_t key_size, GuHasher* hasher,
|
gu_make_map(size_t key_size, GuHasher* hasher,
|
||||||
size_t value_size, const void* default_value,
|
size_t value_size, const void* default_value,
|
||||||
@@ -375,7 +370,7 @@ gu_make_map(size_t key_size, GuHasher* hasher,
|
|||||||
.n_occupied = 0,
|
.n_occupied = 0,
|
||||||
.n_entries = 0,
|
.n_entries = 0,
|
||||||
.keys = NULL,
|
.keys = NULL,
|
||||||
.values = value_size ? NULL : (uint8_t*) gu_map_no_values,
|
.values = NULL,
|
||||||
.zero_idx = SIZE_MAX
|
.zero_idx = SIZE_MAX
|
||||||
};
|
};
|
||||||
GuMap* map = gu_new(GuMap, pool);
|
GuMap* map = gu_new(GuMap, pool);
|
||||||
@@ -384,6 +379,7 @@ gu_make_map(size_t key_size, GuHasher* hasher,
|
|||||||
map->data = data;
|
map->data = data;
|
||||||
map->key_size = key_size;
|
map->key_size = key_size;
|
||||||
map->value_size = value_size;
|
map->value_size = value_size;
|
||||||
|
map->cell_size = GU_MAX(value_size,sizeof(uint8_t));
|
||||||
map->fin.fn = gu_map_finalize;
|
map->fin.fn = gu_map_finalize;
|
||||||
gu_pool_finally(pool, &map->fin);
|
gu_pool_finally(pool, &map->fin);
|
||||||
|
|
||||||
|
|||||||
@@ -62,6 +62,9 @@ gu_map_has(GuMap* ht, const void* key);
|
|||||||
GU_API_DECL void*
|
GU_API_DECL void*
|
||||||
gu_map_insert(GuMap* ht, const void* key);
|
gu_map_insert(GuMap* ht, const void* key);
|
||||||
|
|
||||||
|
GU_API_DECL void
|
||||||
|
gu_map_delete(GuMap* ht, const void* key);
|
||||||
|
|
||||||
#define gu_map_put(MAP, KEYP, V, VAL) \
|
#define gu_map_put(MAP, KEYP, V, VAL) \
|
||||||
GU_BEGIN \
|
GU_BEGIN \
|
||||||
V* gu_map_put_p_ = gu_map_insert((MAP), (KEYP)); \
|
V* gu_map_put_p_ = gu_map_insert((MAP), (KEYP)); \
|
||||||
@@ -71,13 +74,8 @@ gu_map_insert(GuMap* ht, const void* key);
|
|||||||
GU_API_DECL void
|
GU_API_DECL void
|
||||||
gu_map_iter(GuMap* ht, GuMapItor* itor, GuExn* err);
|
gu_map_iter(GuMap* ht, GuMapItor* itor, GuExn* err);
|
||||||
|
|
||||||
typedef struct {
|
GU_API bool
|
||||||
const void* key;
|
gu_map_next(GuMap* map, size_t* pi, void** pkey, void* pvalue);
|
||||||
void* value;
|
|
||||||
} GuMapKeyValue;
|
|
||||||
|
|
||||||
GU_API_DECL GuEnum*
|
|
||||||
gu_map_enum(GuMap* ht, GuPool* pool);
|
|
||||||
|
|
||||||
typedef GuMap GuIntMap;
|
typedef GuMap GuIntMap;
|
||||||
|
|
||||||
|
|||||||
@@ -344,8 +344,9 @@ struct PgfCCat {
|
|||||||
PgfCncFuns* linrefs;
|
PgfCncFuns* linrefs;
|
||||||
size_t n_synprods;
|
size_t n_synprods;
|
||||||
PgfProductionSeq* prods;
|
PgfProductionSeq* prods;
|
||||||
float viterbi_prob;
|
prob_t viterbi_prob;
|
||||||
int fid;
|
int fid;
|
||||||
|
int chunk_count;
|
||||||
PgfItemConts* conts;
|
PgfItemConts* conts;
|
||||||
struct PgfAnswers* answers;
|
struct PgfAnswers* answers;
|
||||||
GuFinalizer fin[0];
|
GuFinalizer fin[0];
|
||||||
|
|||||||
@@ -30,8 +30,8 @@ pgf_expr_unwrap(PgfExpr expr)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
PGF_API int
|
static PgfExprTag
|
||||||
pgf_expr_arity(PgfExpr expr)
|
pgf_expr_arity(PgfExpr expr, int *arity)
|
||||||
{
|
{
|
||||||
int n = 0;
|
int n = 0;
|
||||||
while (true) {
|
while (true) {
|
||||||
@@ -44,10 +44,9 @@ pgf_expr_arity(PgfExpr expr)
|
|||||||
n = n + 1;
|
n = n + 1;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case PGF_EXPR_FUN:
|
|
||||||
return n;
|
|
||||||
default:
|
default:
|
||||||
return -1;
|
*arity = n;
|
||||||
|
return i.tag;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -55,8 +54,8 @@ pgf_expr_arity(PgfExpr expr)
|
|||||||
PGF_API PgfApplication*
|
PGF_API PgfApplication*
|
||||||
pgf_expr_unapply(PgfExpr expr, GuPool* pool)
|
pgf_expr_unapply(PgfExpr expr, GuPool* pool)
|
||||||
{
|
{
|
||||||
int arity = pgf_expr_arity(expr);
|
int arity;
|
||||||
if (arity < 0) {
|
if (pgf_expr_arity(expr, &arity) != PGF_EXPR_FUN) {
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
PgfApplication* appl = gu_new_flex(pool, PgfApplication, args, arity);
|
PgfApplication* appl = gu_new_flex(pool, PgfApplication, args, arity);
|
||||||
@@ -68,13 +67,38 @@ pgf_expr_unapply(PgfExpr expr, GuPool* pool)
|
|||||||
appl->args[n] = app->arg;
|
appl->args[n] = app->arg;
|
||||||
expr = app->fun;
|
expr = app->fun;
|
||||||
}
|
}
|
||||||
PgfExpr e = pgf_expr_unwrap(expr);
|
appl->efun = pgf_expr_unwrap(expr);
|
||||||
gu_assert(gu_variant_tag(e) == PGF_EXPR_FUN);
|
gu_assert(gu_variant_tag(appl->efun) == PGF_EXPR_FUN);
|
||||||
PgfExprFun* fun = gu_variant_data(e);
|
PgfExprFun* fun = gu_variant_data(appl->efun);
|
||||||
appl->fun = fun->fun;
|
appl->fun = fun->fun;
|
||||||
return appl;
|
return appl;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
PGF_API PgfApplication*
|
||||||
|
pgf_expr_unapply_ex(PgfExpr expr, GuPool* pool)
|
||||||
|
{
|
||||||
|
int arity;
|
||||||
|
pgf_expr_arity(expr, &arity);
|
||||||
|
|
||||||
|
PgfApplication* appl = gu_new_flex(pool, PgfApplication, args, arity);
|
||||||
|
appl->n_args = arity;
|
||||||
|
for (int n = arity - 1; n >= 0; n--) {
|
||||||
|
PgfExpr e = pgf_expr_unwrap(expr);
|
||||||
|
gu_assert(gu_variant_tag(e) == PGF_EXPR_APP);
|
||||||
|
PgfExprApp* app = gu_variant_data(e);
|
||||||
|
appl->args[n] = app->arg;
|
||||||
|
expr = app->fun;
|
||||||
|
}
|
||||||
|
appl->efun = pgf_expr_unwrap(expr);
|
||||||
|
if (gu_variant_tag(appl->efun) == PGF_EXPR_FUN) {
|
||||||
|
PgfExprFun* fun = gu_variant_data(appl->efun);
|
||||||
|
appl->fun = fun->fun;
|
||||||
|
} else {
|
||||||
|
appl->fun = NULL;
|
||||||
|
}
|
||||||
|
return appl;
|
||||||
|
}
|
||||||
|
|
||||||
PGF_API PgfExpr
|
PGF_API PgfExpr
|
||||||
pgf_expr_apply(PgfApplication* app, GuPool* pool)
|
pgf_expr_apply(PgfApplication* app, GuPool* pool)
|
||||||
{
|
{
|
||||||
@@ -675,6 +699,17 @@ pgf_expr_parser_binds(PgfExprParser* parser)
|
|||||||
return binds;
|
return binds;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
PGF_API GuString
|
||||||
|
pgf_expr_parser_ident(PgfExprParser* parser)
|
||||||
|
{
|
||||||
|
GuString ident = NULL;
|
||||||
|
if (parser->token_tag == PGF_TOKEN_IDENT) {
|
||||||
|
ident = gu_string_copy(gu_string_buf_data(parser->token_value), parser->expr_pool);
|
||||||
|
pgf_expr_parser_token(parser, true);
|
||||||
|
}
|
||||||
|
return ident;
|
||||||
|
}
|
||||||
|
|
||||||
PGF_API PgfExpr
|
PGF_API PgfExpr
|
||||||
pgf_expr_parser_expr(PgfExprParser* parser, bool mark)
|
pgf_expr_parser_expr(PgfExprParser* parser, bool mark)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -126,12 +126,10 @@ typedef struct {
|
|||||||
PgfExpr expr;
|
PgfExpr expr;
|
||||||
} PgfExprProb;
|
} PgfExprProb;
|
||||||
|
|
||||||
PGF_API_DECL int
|
|
||||||
pgf_expr_arity(PgfExpr expr);
|
|
||||||
|
|
||||||
typedef struct PgfApplication PgfApplication;
|
typedef struct PgfApplication PgfApplication;
|
||||||
|
|
||||||
struct PgfApplication {
|
struct PgfApplication {
|
||||||
|
PgfExpr efun;
|
||||||
PgfCId fun;
|
PgfCId fun;
|
||||||
int n_args;
|
int n_args;
|
||||||
PgfExpr args[];
|
PgfExpr args[];
|
||||||
@@ -140,6 +138,9 @@ struct PgfApplication {
|
|||||||
PGF_API_DECL PgfApplication*
|
PGF_API_DECL PgfApplication*
|
||||||
pgf_expr_unapply(PgfExpr expr, GuPool* pool);
|
pgf_expr_unapply(PgfExpr expr, GuPool* pool);
|
||||||
|
|
||||||
|
PGF_API_DECL PgfApplication*
|
||||||
|
pgf_expr_unapply_ex(PgfExpr expr, GuPool* pool);
|
||||||
|
|
||||||
PGF_API_DECL PgfExpr
|
PGF_API_DECL PgfExpr
|
||||||
pgf_expr_apply(PgfApplication*, GuPool* pool);
|
pgf_expr_apply(PgfApplication*, GuPool* pool);
|
||||||
|
|
||||||
|
|||||||
@@ -175,9 +175,8 @@ redo:;
|
|||||||
gu_buf_get(buf, PgfProductionApply*, index);
|
gu_buf_get(buf, PgfProductionApply*, index);
|
||||||
gu_assert(n_args == gu_seq_length(papply->args));
|
gu_assert(n_args == gu_seq_length(papply->args));
|
||||||
|
|
||||||
capp->abs_id = papply->fun->absfun->name;
|
capp->fun = papply->fun;
|
||||||
capp->fun = papply->fun;
|
capp->fid = 0;
|
||||||
capp->fid = 0;
|
|
||||||
capp->n_args = n_args;
|
capp->n_args = n_args;
|
||||||
|
|
||||||
for (size_t i = 0; i < n_args; i++) {
|
for (size_t i = 0; i < n_args; i++) {
|
||||||
@@ -223,10 +222,10 @@ redo:;
|
|||||||
static PgfCncTree
|
static PgfCncTree
|
||||||
pgf_cnc_resolve_def(PgfCnc* cnc,
|
pgf_cnc_resolve_def(PgfCnc* cnc,
|
||||||
size_t n_vars, PgfPrintContext* context,
|
size_t n_vars, PgfPrintContext* context,
|
||||||
PgfCId abs_id, PgfCCat* ccat, GuString s, GuPool* pool)
|
PgfCCat* ccat, GuString s, GuPool* pool)
|
||||||
{
|
{
|
||||||
PgfCncTree ret = gu_null_variant;
|
|
||||||
PgfCncTree lit = gu_null_variant;
|
PgfCncTree lit = gu_null_variant;
|
||||||
|
PgfCncTree ret = gu_null_variant;
|
||||||
|
|
||||||
PgfCncTreeLit* clit =
|
PgfCncTreeLit* clit =
|
||||||
gu_new_variant(PGF_CNC_TREE_LIT,
|
gu_new_variant(PGF_CNC_TREE_LIT,
|
||||||
@@ -234,7 +233,7 @@ pgf_cnc_resolve_def(PgfCnc* cnc,
|
|||||||
&lit, pool);
|
&lit, pool);
|
||||||
clit->n_vars = 0;
|
clit->n_vars = 0;
|
||||||
clit->context = context;
|
clit->context = context;
|
||||||
clit->fid = -1; // don't report the literal in the bracket
|
clit->fid = cnc->fid++;
|
||||||
PgfLiteralStr* lit_str =
|
PgfLiteralStr* lit_str =
|
||||||
gu_new_flex_variant(PGF_LITERAL_STR,
|
gu_new_flex_variant(PGF_LITERAL_STR,
|
||||||
PgfLiteralStr,
|
PgfLiteralStr,
|
||||||
@@ -242,7 +241,7 @@ pgf_cnc_resolve_def(PgfCnc* cnc,
|
|||||||
&clit->lit, pool);
|
&clit->lit, pool);
|
||||||
strcpy((char*) lit_str->val, (char*) s);
|
strcpy((char*) lit_str->val, (char*) s);
|
||||||
|
|
||||||
if (ccat == NULL || ccat->lindefs == NULL)
|
if (ccat->lindefs == NULL)
|
||||||
return lit;
|
return lit;
|
||||||
|
|
||||||
int index =
|
int index =
|
||||||
@@ -254,10 +253,9 @@ pgf_cnc_resolve_def(PgfCnc* cnc,
|
|||||||
gu_new_flex_variant(PGF_CNC_TREE_APP,
|
gu_new_flex_variant(PGF_CNC_TREE_APP,
|
||||||
PgfCncTreeApp,
|
PgfCncTreeApp,
|
||||||
args, 1, &ret, pool);
|
args, 1, &ret, pool);
|
||||||
capp->ccat = ccat;
|
capp->ccat = ccat;
|
||||||
capp->abs_id= abs_id;
|
capp->fun = gu_seq_get(ccat->lindefs, PgfCncFun*, index);
|
||||||
capp->fun = gu_seq_get(ccat->lindefs, PgfCncFun*, index);
|
capp->fid = cnc->fid++;
|
||||||
capp->fid = cnc->fid++;
|
|
||||||
capp->n_vars = n_vars;
|
capp->n_vars = n_vars;
|
||||||
capp->context = context;
|
capp->context = context;
|
||||||
capp->n_args = 1;
|
capp->n_args = 1;
|
||||||
@@ -297,7 +295,7 @@ pgf_lzr_wrap_linref(PgfCncTree ctree, GuPool* pool)
|
|||||||
PgfCncTreeApp* capp = cti.data;
|
PgfCncTreeApp* capp = cti.data;
|
||||||
|
|
||||||
assert(gu_seq_length(capp->ccat->linrefs) > 0);
|
assert(gu_seq_length(capp->ccat->linrefs) > 0);
|
||||||
|
|
||||||
// here we must apply the linref function
|
// here we must apply the linref function
|
||||||
PgfCncTree new_ctree;
|
PgfCncTree new_ctree;
|
||||||
PgfCncTreeApp* new_capp =
|
PgfCncTreeApp* new_capp =
|
||||||
@@ -305,7 +303,6 @@ pgf_lzr_wrap_linref(PgfCncTree ctree, GuPool* pool)
|
|||||||
PgfCncTreeApp,
|
PgfCncTreeApp,
|
||||||
args, 1, &new_ctree, pool);
|
args, 1, &new_ctree, pool);
|
||||||
new_capp->ccat = NULL;
|
new_capp->ccat = NULL;
|
||||||
new_capp->abs_id = NULL;
|
|
||||||
new_capp->fun = gu_seq_get(capp->ccat->linrefs, PgfCncFun*, 0);
|
new_capp->fun = gu_seq_get(capp->ccat->linrefs, PgfCncFun*, 0);
|
||||||
new_capp->fid = -1;
|
new_capp->fid = -1;
|
||||||
new_capp->n_vars = 0;
|
new_capp->n_vars = 0;
|
||||||
@@ -317,7 +314,7 @@ pgf_lzr_wrap_linref(PgfCncTree ctree, GuPool* pool)
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return ctree;
|
return ctree;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -399,17 +396,6 @@ pgf_cnc_resolve(PgfCnc* cnc,
|
|||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
||||||
PgfCId abs_id = "?";
|
|
||||||
if (emeta->id > 0) {
|
|
||||||
GuPool* tmp_pool = gu_local_pool();
|
|
||||||
GuExn* err = gu_new_exn(tmp_pool);
|
|
||||||
GuStringBuf* sbuf = gu_new_string_buf(tmp_pool);
|
|
||||||
GuOut* out = gu_string_buf_out(sbuf);
|
|
||||||
|
|
||||||
gu_printf(out, err, "?%d", emeta->id);
|
|
||||||
abs_id = gu_string_buf_freeze(sbuf, pool);
|
|
||||||
}
|
|
||||||
|
|
||||||
int index =
|
int index =
|
||||||
gu_choice_next(cnc->ch, gu_seq_length(ccat->lindefs));
|
gu_choice_next(cnc->ch, gu_seq_length(ccat->lindefs));
|
||||||
if (index < 0) {
|
if (index < 0) {
|
||||||
@@ -420,7 +406,6 @@ pgf_cnc_resolve(PgfCnc* cnc,
|
|||||||
PgfCncTreeApp,
|
PgfCncTreeApp,
|
||||||
args, 1, &ret, pool);
|
args, 1, &ret, pool);
|
||||||
capp->ccat = ccat;
|
capp->ccat = ccat;
|
||||||
capp->abs_id = abs_id;
|
|
||||||
capp->fun = gu_seq_get(ccat->lindefs, PgfCncFun*, index);
|
capp->fun = gu_seq_get(ccat->lindefs, PgfCncFun*, index);
|
||||||
capp->fid = cnc->fid++;
|
capp->fid = cnc->fid++;
|
||||||
capp->n_vars = 0;
|
capp->n_vars = 0;
|
||||||
@@ -450,7 +435,23 @@ pgf_cnc_resolve(PgfCnc* cnc,
|
|||||||
gu_putc(']', out, err);
|
gu_putc(']', out, err);
|
||||||
GuString s = gu_string_buf_freeze(sbuf, tmp_pool);
|
GuString s = gu_string_buf_freeze(sbuf, tmp_pool);
|
||||||
|
|
||||||
ret = pgf_cnc_resolve_def(cnc, n_vars, context, efun->fun, ccat, s, pool);
|
if (ccat != NULL) {
|
||||||
|
ret = pgf_cnc_resolve_def(cnc, n_vars, context, ccat, s, pool);
|
||||||
|
} else {
|
||||||
|
PgfCncTreeLit* clit =
|
||||||
|
gu_new_variant(PGF_CNC_TREE_LIT,
|
||||||
|
PgfCncTreeLit,
|
||||||
|
&ret, pool);
|
||||||
|
clit->n_vars = 0;
|
||||||
|
clit->context = context;
|
||||||
|
clit->fid = cnc->fid++;
|
||||||
|
PgfLiteralStr* lit =
|
||||||
|
gu_new_flex_variant(PGF_LITERAL_STR,
|
||||||
|
PgfLiteralStr,
|
||||||
|
val, strlen(s)+1,
|
||||||
|
&clit->lit, pool);
|
||||||
|
strcpy(lit->val, s);
|
||||||
|
}
|
||||||
|
|
||||||
gu_pool_free(tmp_pool);
|
gu_pool_free(tmp_pool);
|
||||||
goto done;
|
goto done;
|
||||||
@@ -498,7 +499,28 @@ redo:;
|
|||||||
index--;
|
index--;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = pgf_cnc_resolve_def(cnc, n_vars, context, ctxt->name, ccat, ctxt->name, pool);
|
if (ccat != NULL && ccat->lindefs == NULL) {
|
||||||
|
goto done;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ccat != NULL) {
|
||||||
|
ret = pgf_cnc_resolve_def(cnc, n_vars, context, ccat, ctxt->name, pool);
|
||||||
|
} else {
|
||||||
|
PgfCncTreeLit* clit =
|
||||||
|
gu_new_variant(PGF_CNC_TREE_LIT,
|
||||||
|
PgfCncTreeLit,
|
||||||
|
&ret, pool);
|
||||||
|
clit->n_vars = 0;
|
||||||
|
clit->context = context;
|
||||||
|
clit->fid = cnc->fid++;
|
||||||
|
PgfLiteralStr* lit =
|
||||||
|
gu_new_flex_variant(PGF_LITERAL_STR,
|
||||||
|
PgfLiteralStr,
|
||||||
|
val, strlen(ctxt->name)+1,
|
||||||
|
&clit->lit, pool);
|
||||||
|
strcpy(lit->val, ctxt->name);
|
||||||
|
}
|
||||||
|
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
case PGF_EXPR_TYPED: {
|
case PGF_EXPR_TYPED: {
|
||||||
@@ -917,9 +939,9 @@ pgf_lzr_linearize_tree(PgfLzr* lzr, PgfCncTree ctree, size_t lin_idx)
|
|||||||
|
|
||||||
if ((*lzr->funcs)->begin_phrase && fapp->ccat != NULL) {
|
if ((*lzr->funcs)->begin_phrase && fapp->ccat != NULL) {
|
||||||
(*lzr->funcs)->begin_phrase(lzr->funcs,
|
(*lzr->funcs)->begin_phrase(lzr->funcs,
|
||||||
fapp->ccat->cnccat->abscat->name,
|
fun->absfun->type->cid,
|
||||||
fapp->fid, lin_idx,
|
fapp->fid, lin_idx,
|
||||||
fapp->abs_id);
|
fun->absfun->name);
|
||||||
}
|
}
|
||||||
|
|
||||||
gu_require(lin_idx < fun->n_lins);
|
gu_require(lin_idx < fun->n_lins);
|
||||||
@@ -927,9 +949,9 @@ pgf_lzr_linearize_tree(PgfLzr* lzr, PgfCncTree ctree, size_t lin_idx)
|
|||||||
|
|
||||||
if ((*lzr->funcs)->end_phrase && fapp->ccat != NULL) {
|
if ((*lzr->funcs)->end_phrase && fapp->ccat != NULL) {
|
||||||
(*lzr->funcs)->end_phrase(lzr->funcs,
|
(*lzr->funcs)->end_phrase(lzr->funcs,
|
||||||
fapp->ccat->cnccat->abscat->name,
|
fun->absfun->type->cid,
|
||||||
fapp->fid, lin_idx,
|
fapp->fid, lin_idx,
|
||||||
fapp->abs_id);
|
fun->absfun->name);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -955,7 +977,7 @@ pgf_lzr_linearize_tree(PgfLzr* lzr, PgfCncTree ctree, size_t lin_idx)
|
|||||||
PgfCId cat =
|
PgfCId cat =
|
||||||
pgf_literal_cat(lzr->concr, flit->lit)->cnccat->abscat->name;
|
pgf_literal_cat(lzr->concr, flit->lit)->cnccat->abscat->name;
|
||||||
|
|
||||||
if ((*lzr->funcs)->begin_phrase && flit->fid >= 0) {
|
if ((*lzr->funcs)->begin_phrase) {
|
||||||
(*lzr->funcs)->begin_phrase(lzr->funcs,
|
(*lzr->funcs)->begin_phrase(lzr->funcs,
|
||||||
cat, flit->fid, 0,
|
cat, flit->fid, 0,
|
||||||
"");
|
"");
|
||||||
@@ -987,7 +1009,7 @@ pgf_lzr_linearize_tree(PgfLzr* lzr, PgfCncTree ctree, size_t lin_idx)
|
|||||||
(*lzr->funcs)->symbol_token(lzr->funcs, tok);
|
(*lzr->funcs)->symbol_token(lzr->funcs, tok);
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((*lzr->funcs)->end_phrase && flit->fid >= 0) {
|
if ((*lzr->funcs)->end_phrase) {
|
||||||
(*lzr->funcs)->end_phrase(lzr->funcs,
|
(*lzr->funcs)->end_phrase(lzr->funcs,
|
||||||
cat, flit->fid, 0,
|
cat, flit->fid, 0,
|
||||||
"");
|
"");
|
||||||
|
|||||||
@@ -22,7 +22,6 @@ typedef enum {
|
|||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
PgfCCat* ccat;
|
PgfCCat* ccat;
|
||||||
PgfCId abs_id;
|
|
||||||
PgfCncFun* fun;
|
PgfCncFun* fun;
|
||||||
int fid;
|
int fid;
|
||||||
|
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user