Import hs-plugins cvs
This commit is contained in:
commit
887fa59389
5
AUTHORS
Normal file
5
AUTHORS
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
|
||||||
|
Don Stewart <http://www.cse.unsw.edu.au/~dons>
|
||||||
|
Sean Seefried <http://www.cse.unsw.edu.au/~sseefried>
|
||||||
|
Andre Pang <http://www.algorithm.com.au>
|
||||||
|
|
25
BUILDING.CVS
Normal file
25
BUILDING.CVS
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
CVS BUILDING INSTRUCTIONS
|
||||||
|
=========================
|
||||||
|
|
||||||
|
These are build instructions if you've checked out hs-plugins
|
||||||
|
from CVS (instead of downloading a source distribution tarball).
|
||||||
|
|
||||||
|
1. Execute autogen.sh to generate the GNU ./configure script:
|
||||||
|
|
||||||
|
./autogen.sh
|
||||||
|
|
||||||
|
2. Build hs-plugins as usual with ./configure && make
|
||||||
|
|
||||||
|
|
||||||
|
cvsps
|
||||||
|
=====
|
||||||
|
|
||||||
|
For people who are used to more modern revision control systems
|
||||||
|
(such as Darcs, Subversion and Arch) and miss working with
|
||||||
|
'patchsets' instead of the disjoint per-file patches that CVS
|
||||||
|
uses, take a look at cvsps <http://www.cobite.com/cvsps/>,
|
||||||
|
a patchset manager for CVS. While it doesn't, by any means, give
|
||||||
|
you the many advantages that more modern source control systems
|
||||||
|
offer you, it certainly makes using CVS and managing patches far
|
||||||
|
easier!
|
||||||
|
|
504
LICENSE
Normal file
504
LICENSE
Normal file
@ -0,0 +1,504 @@
|
|||||||
|
GNU LESSER GENERAL PUBLIC LICENSE
|
||||||
|
Version 2.1, February 1999
|
||||||
|
|
||||||
|
Copyright (C) 1991, 1999 Free Software Foundation, Inc.
|
||||||
|
59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
Everyone is permitted to copy and distribute verbatim copies
|
||||||
|
of this license document, but changing it is not allowed.
|
||||||
|
|
||||||
|
[This is the first released version of the Lesser GPL. It also counts
|
||||||
|
as the successor of the GNU Library Public License, version 2, hence
|
||||||
|
the version number 2.1.]
|
||||||
|
|
||||||
|
Preamble
|
||||||
|
|
||||||
|
The licenses for most software are designed to take away your
|
||||||
|
freedom to share and change it. By contrast, the GNU General Public
|
||||||
|
Licenses are intended to guarantee your freedom to share and change
|
||||||
|
free software--to make sure the software is free for all its users.
|
||||||
|
|
||||||
|
This license, the Lesser General Public License, applies to some
|
||||||
|
specially designated software packages--typically libraries--of the
|
||||||
|
Free Software Foundation and other authors who decide to use it. You
|
||||||
|
can use it too, but we suggest you first think carefully about whether
|
||||||
|
this license or the ordinary General Public License is the better
|
||||||
|
strategy to use in any particular case, based on the explanations below.
|
||||||
|
|
||||||
|
When we speak of free software, we are referring to freedom of use,
|
||||||
|
not price. Our General Public Licenses are designed to make sure that
|
||||||
|
you have the freedom to distribute copies of free software (and charge
|
||||||
|
for this service if you wish); that you receive source code or can get
|
||||||
|
it if you want it; that you can change the software and use pieces of
|
||||||
|
it in new free programs; and that you are informed that you can do
|
||||||
|
these things.
|
||||||
|
|
||||||
|
To protect your rights, we need to make restrictions that forbid
|
||||||
|
distributors to deny you these rights or to ask you to surrender these
|
||||||
|
rights. These restrictions translate to certain responsibilities for
|
||||||
|
you if you distribute copies of the library or if you modify it.
|
||||||
|
|
||||||
|
For example, if you distribute copies of the library, whether gratis
|
||||||
|
or for a fee, you must give the recipients all the rights that we gave
|
||||||
|
you. You must make sure that they, too, receive or can get the source
|
||||||
|
code. If you link other code with the library, you must provide
|
||||||
|
complete object files to the recipients, so that they can relink them
|
||||||
|
with the library after making changes to the library and recompiling
|
||||||
|
it. And you must show them these terms so they know their rights.
|
||||||
|
|
||||||
|
We protect your rights with a two-step method: (1) we copyright the
|
||||||
|
library, and (2) we offer you this license, which gives you legal
|
||||||
|
permission to copy, distribute and/or modify the library.
|
||||||
|
|
||||||
|
To protect each distributor, we want to make it very clear that
|
||||||
|
there is no warranty for the free library. Also, if the library is
|
||||||
|
modified by someone else and passed on, the recipients should know
|
||||||
|
that what they have is not the original version, so that the original
|
||||||
|
author's reputation will not be affected by problems that might be
|
||||||
|
introduced by others.
|
||||||
|
|
||||||
|
Finally, software patents pose a constant threat to the existence of
|
||||||
|
any free program. We wish to make sure that a company cannot
|
||||||
|
effectively restrict the users of a free program by obtaining a
|
||||||
|
restrictive license from a patent holder. Therefore, we insist that
|
||||||
|
any patent license obtained for a version of the library must be
|
||||||
|
consistent with the full freedom of use specified in this license.
|
||||||
|
|
||||||
|
Most GNU software, including some libraries, is covered by the
|
||||||
|
ordinary GNU General Public License. This license, the GNU Lesser
|
||||||
|
General Public License, applies to certain designated libraries, and
|
||||||
|
is quite different from the ordinary General Public License. We use
|
||||||
|
this license for certain libraries in order to permit linking those
|
||||||
|
libraries into non-free programs.
|
||||||
|
|
||||||
|
When a program is linked with a library, whether statically or using
|
||||||
|
a shared library, the combination of the two is legally speaking a
|
||||||
|
combined work, a derivative of the original library. The ordinary
|
||||||
|
General Public License therefore permits such linking only if the
|
||||||
|
entire combination fits its criteria of freedom. The Lesser General
|
||||||
|
Public License permits more lax criteria for linking other code with
|
||||||
|
the library.
|
||||||
|
|
||||||
|
We call this license the "Lesser" General Public License because it
|
||||||
|
does Less to protect the user's freedom than the ordinary General
|
||||||
|
Public License. It also provides other free software developers Less
|
||||||
|
of an advantage over competing non-free programs. These disadvantages
|
||||||
|
are the reason we use the ordinary General Public License for many
|
||||||
|
libraries. However, the Lesser license provides advantages in certain
|
||||||
|
special circumstances.
|
||||||
|
|
||||||
|
For example, on rare occasions, there may be a special need to
|
||||||
|
encourage the widest possible use of a certain library, so that it becomes
|
||||||
|
a de-facto standard. To achieve this, non-free programs must be
|
||||||
|
allowed to use the library. A more frequent case is that a free
|
||||||
|
library does the same job as widely used non-free libraries. In this
|
||||||
|
case, there is little to gain by limiting the free library to free
|
||||||
|
software only, so we use the Lesser General Public License.
|
||||||
|
|
||||||
|
In other cases, permission to use a particular library in non-free
|
||||||
|
programs enables a greater number of people to use a large body of
|
||||||
|
free software. For example, permission to use the GNU C Library in
|
||||||
|
non-free programs enables many more people to use the whole GNU
|
||||||
|
operating system, as well as its variant, the GNU/Linux operating
|
||||||
|
system.
|
||||||
|
|
||||||
|
Although the Lesser General Public License is Less protective of the
|
||||||
|
users' freedom, it does ensure that the user of a program that is
|
||||||
|
linked with the Library has the freedom and the wherewithal to run
|
||||||
|
that program using a modified version of the Library.
|
||||||
|
|
||||||
|
The precise terms and conditions for copying, distribution and
|
||||||
|
modification follow. Pay close attention to the difference between a
|
||||||
|
"work based on the library" and a "work that uses the library". The
|
||||||
|
former contains code derived from the library, whereas the latter must
|
||||||
|
be combined with the library in order to run.
|
||||||
|
|
||||||
|
GNU LESSER GENERAL PUBLIC LICENSE
|
||||||
|
TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
|
||||||
|
|
||||||
|
0. This License Agreement applies to any software library or other
|
||||||
|
program which contains a notice placed by the copyright holder or
|
||||||
|
other authorized party saying it may be distributed under the terms of
|
||||||
|
this Lesser General Public License (also called "this License").
|
||||||
|
Each licensee is addressed as "you".
|
||||||
|
|
||||||
|
A "library" means a collection of software functions and/or data
|
||||||
|
prepared so as to be conveniently linked with application programs
|
||||||
|
(which use some of those functions and data) to form executables.
|
||||||
|
|
||||||
|
The "Library", below, refers to any such software library or work
|
||||||
|
which has been distributed under these terms. A "work based on the
|
||||||
|
Library" means either the Library or any derivative work under
|
||||||
|
copyright law: that is to say, a work containing the Library or a
|
||||||
|
portion of it, either verbatim or with modifications and/or translated
|
||||||
|
straightforwardly into another language. (Hereinafter, translation is
|
||||||
|
included without limitation in the term "modification".)
|
||||||
|
|
||||||
|
"Source code" for a work means the preferred form of the work for
|
||||||
|
making modifications to it. For a library, complete source code means
|
||||||
|
all the source code for all modules it contains, plus any associated
|
||||||
|
interface definition files, plus the scripts used to control compilation
|
||||||
|
and installation of the library.
|
||||||
|
|
||||||
|
Activities other than copying, distribution and modification are not
|
||||||
|
covered by this License; they are outside its scope. The act of
|
||||||
|
running a program using the Library is not restricted, and output from
|
||||||
|
such a program is covered only if its contents constitute a work based
|
||||||
|
on the Library (independent of the use of the Library in a tool for
|
||||||
|
writing it). Whether that is true depends on what the Library does
|
||||||
|
and what the program that uses the Library does.
|
||||||
|
|
||||||
|
1. You may copy and distribute verbatim copies of the Library's
|
||||||
|
complete source code as you receive it, in any medium, provided that
|
||||||
|
you conspicuously and appropriately publish on each copy an
|
||||||
|
appropriate copyright notice and disclaimer of warranty; keep intact
|
||||||
|
all the notices that refer to this License and to the absence of any
|
||||||
|
warranty; and distribute a copy of this License along with the
|
||||||
|
Library.
|
||||||
|
|
||||||
|
You may charge a fee for the physical act of transferring a copy,
|
||||||
|
and you may at your option offer warranty protection in exchange for a
|
||||||
|
fee.
|
||||||
|
|
||||||
|
2. You may modify your copy or copies of the Library or any portion
|
||||||
|
of it, thus forming a work based on the Library, and copy and
|
||||||
|
distribute such modifications or work under the terms of Section 1
|
||||||
|
above, provided that you also meet all of these conditions:
|
||||||
|
|
||||||
|
a) The modified work must itself be a software library.
|
||||||
|
|
||||||
|
b) You must cause the files modified to carry prominent notices
|
||||||
|
stating that you changed the files and the date of any change.
|
||||||
|
|
||||||
|
c) You must cause the whole of the work to be licensed at no
|
||||||
|
charge to all third parties under the terms of this License.
|
||||||
|
|
||||||
|
d) If a facility in the modified Library refers to a function or a
|
||||||
|
table of data to be supplied by an application program that uses
|
||||||
|
the facility, other than as an argument passed when the facility
|
||||||
|
is invoked, then you must make a good faith effort to ensure that,
|
||||||
|
in the event an application does not supply such function or
|
||||||
|
table, the facility still operates, and performs whatever part of
|
||||||
|
its purpose remains meaningful.
|
||||||
|
|
||||||
|
(For example, a function in a library to compute square roots has
|
||||||
|
a purpose that is entirely well-defined independent of the
|
||||||
|
application. Therefore, Subsection 2d requires that any
|
||||||
|
application-supplied function or table used by this function must
|
||||||
|
be optional: if the application does not supply it, the square
|
||||||
|
root function must still compute square roots.)
|
||||||
|
|
||||||
|
These requirements apply to the modified work as a whole. If
|
||||||
|
identifiable sections of that work are not derived from the Library,
|
||||||
|
and can be reasonably considered independent and separate works in
|
||||||
|
themselves, then this License, and its terms, do not apply to those
|
||||||
|
sections when you distribute them as separate works. But when you
|
||||||
|
distribute the same sections as part of a whole which is a work based
|
||||||
|
on the Library, the distribution of the whole must be on the terms of
|
||||||
|
this License, whose permissions for other licensees extend to the
|
||||||
|
entire whole, and thus to each and every part regardless of who wrote
|
||||||
|
it.
|
||||||
|
|
||||||
|
Thus, it is not the intent of this section to claim rights or contest
|
||||||
|
your rights to work written entirely by you; rather, the intent is to
|
||||||
|
exercise the right to control the distribution of derivative or
|
||||||
|
collective works based on the Library.
|
||||||
|
|
||||||
|
In addition, mere aggregation of another work not based on the Library
|
||||||
|
with the Library (or with a work based on the Library) on a volume of
|
||||||
|
a storage or distribution medium does not bring the other work under
|
||||||
|
the scope of this License.
|
||||||
|
|
||||||
|
3. You may opt to apply the terms of the ordinary GNU General Public
|
||||||
|
License instead of this License to a given copy of the Library. To do
|
||||||
|
this, you must alter all the notices that refer to this License, so
|
||||||
|
that they refer to the ordinary GNU General Public License, version 2,
|
||||||
|
instead of to this License. (If a newer version than version 2 of the
|
||||||
|
ordinary GNU General Public License has appeared, then you can specify
|
||||||
|
that version instead if you wish.) Do not make any other change in
|
||||||
|
these notices.
|
||||||
|
|
||||||
|
Once this change is made in a given copy, it is irreversible for
|
||||||
|
that copy, so the ordinary GNU General Public License applies to all
|
||||||
|
subsequent copies and derivative works made from that copy.
|
||||||
|
|
||||||
|
This option is useful when you wish to copy part of the code of
|
||||||
|
the Library into a program that is not a library.
|
||||||
|
|
||||||
|
4. You may copy and distribute the Library (or a portion or
|
||||||
|
derivative of it, under Section 2) in object code or executable form
|
||||||
|
under the terms of Sections 1 and 2 above provided that you accompany
|
||||||
|
it with the complete corresponding machine-readable source code, which
|
||||||
|
must be distributed under the terms of Sections 1 and 2 above on a
|
||||||
|
medium customarily used for software interchange.
|
||||||
|
|
||||||
|
If distribution of object code is made by offering access to copy
|
||||||
|
from a designated place, then offering equivalent access to copy the
|
||||||
|
source code from the same place satisfies the requirement to
|
||||||
|
distribute the source code, even though third parties are not
|
||||||
|
compelled to copy the source along with the object code.
|
||||||
|
|
||||||
|
5. A program that contains no derivative of any portion of the
|
||||||
|
Library, but is designed to work with the Library by being compiled or
|
||||||
|
linked with it, is called a "work that uses the Library". Such a
|
||||||
|
work, in isolation, is not a derivative work of the Library, and
|
||||||
|
therefore falls outside the scope of this License.
|
||||||
|
|
||||||
|
However, linking a "work that uses the Library" with the Library
|
||||||
|
creates an executable that is a derivative of the Library (because it
|
||||||
|
contains portions of the Library), rather than a "work that uses the
|
||||||
|
library". The executable is therefore covered by this License.
|
||||||
|
Section 6 states terms for distribution of such executables.
|
||||||
|
|
||||||
|
When a "work that uses the Library" uses material from a header file
|
||||||
|
that is part of the Library, the object code for the work may be a
|
||||||
|
derivative work of the Library even though the source code is not.
|
||||||
|
Whether this is true is especially significant if the work can be
|
||||||
|
linked without the Library, or if the work is itself a library. The
|
||||||
|
threshold for this to be true is not precisely defined by law.
|
||||||
|
|
||||||
|
If such an object file uses only numerical parameters, data
|
||||||
|
structure layouts and accessors, and small macros and small inline
|
||||||
|
functions (ten lines or less in length), then the use of the object
|
||||||
|
file is unrestricted, regardless of whether it is legally a derivative
|
||||||
|
work. (Executables containing this object code plus portions of the
|
||||||
|
Library will still fall under Section 6.)
|
||||||
|
|
||||||
|
Otherwise, if the work is a derivative of the Library, you may
|
||||||
|
distribute the object code for the work under the terms of Section 6.
|
||||||
|
Any executables containing that work also fall under Section 6,
|
||||||
|
whether or not they are linked directly with the Library itself.
|
||||||
|
|
||||||
|
6. As an exception to the Sections above, you may also combine or
|
||||||
|
link a "work that uses the Library" with the Library to produce a
|
||||||
|
work containing portions of the Library, and distribute that work
|
||||||
|
under terms of your choice, provided that the terms permit
|
||||||
|
modification of the work for the customer's own use and reverse
|
||||||
|
engineering for debugging such modifications.
|
||||||
|
|
||||||
|
You must give prominent notice with each copy of the work that the
|
||||||
|
Library is used in it and that the Library and its use are covered by
|
||||||
|
this License. You must supply a copy of this License. If the work
|
||||||
|
during execution displays copyright notices, you must include the
|
||||||
|
copyright notice for the Library among them, as well as a reference
|
||||||
|
directing the user to the copy of this License. Also, you must do one
|
||||||
|
of these things:
|
||||||
|
|
||||||
|
a) Accompany the work with the complete corresponding
|
||||||
|
machine-readable source code for the Library including whatever
|
||||||
|
changes were used in the work (which must be distributed under
|
||||||
|
Sections 1 and 2 above); and, if the work is an executable linked
|
||||||
|
with the Library, with the complete machine-readable "work that
|
||||||
|
uses the Library", as object code and/or source code, so that the
|
||||||
|
user can modify the Library and then relink to produce a modified
|
||||||
|
executable containing the modified Library. (It is understood
|
||||||
|
that the user who changes the contents of definitions files in the
|
||||||
|
Library will not necessarily be able to recompile the application
|
||||||
|
to use the modified definitions.)
|
||||||
|
|
||||||
|
b) Use a suitable shared library mechanism for linking with the
|
||||||
|
Library. A suitable mechanism is one that (1) uses at run time a
|
||||||
|
copy of the library already present on the user's computer system,
|
||||||
|
rather than copying library functions into the executable, and (2)
|
||||||
|
will operate properly with a modified version of the library, if
|
||||||
|
the user installs one, as long as the modified version is
|
||||||
|
interface-compatible with the version that the work was made with.
|
||||||
|
|
||||||
|
c) Accompany the work with a written offer, valid for at
|
||||||
|
least three years, to give the same user the materials
|
||||||
|
specified in Subsection 6a, above, for a charge no more
|
||||||
|
than the cost of performing this distribution.
|
||||||
|
|
||||||
|
d) If distribution of the work is made by offering access to copy
|
||||||
|
from a designated place, offer equivalent access to copy the above
|
||||||
|
specified materials from the same place.
|
||||||
|
|
||||||
|
e) Verify that the user has already received a copy of these
|
||||||
|
materials or that you have already sent this user a copy.
|
||||||
|
|
||||||
|
For an executable, the required form of the "work that uses the
|
||||||
|
Library" must include any data and utility programs needed for
|
||||||
|
reproducing the executable from it. However, as a special exception,
|
||||||
|
the materials to be distributed need not include anything that is
|
||||||
|
normally distributed (in either source or binary form) with the major
|
||||||
|
components (compiler, kernel, and so on) of the operating system on
|
||||||
|
which the executable runs, unless that component itself accompanies
|
||||||
|
the executable.
|
||||||
|
|
||||||
|
It may happen that this requirement contradicts the license
|
||||||
|
restrictions of other proprietary libraries that do not normally
|
||||||
|
accompany the operating system. Such a contradiction means you cannot
|
||||||
|
use both them and the Library together in an executable that you
|
||||||
|
distribute.
|
||||||
|
|
||||||
|
7. You may place library facilities that are a work based on the
|
||||||
|
Library side-by-side in a single library together with other library
|
||||||
|
facilities not covered by this License, and distribute such a combined
|
||||||
|
library, provided that the separate distribution of the work based on
|
||||||
|
the Library and of the other library facilities is otherwise
|
||||||
|
permitted, and provided that you do these two things:
|
||||||
|
|
||||||
|
a) Accompany the combined library with a copy of the same work
|
||||||
|
based on the Library, uncombined with any other library
|
||||||
|
facilities. This must be distributed under the terms of the
|
||||||
|
Sections above.
|
||||||
|
|
||||||
|
b) Give prominent notice with the combined library of the fact
|
||||||
|
that part of it is a work based on the Library, and explaining
|
||||||
|
where to find the accompanying uncombined form of the same work.
|
||||||
|
|
||||||
|
8. You may not copy, modify, sublicense, link with, or distribute
|
||||||
|
the Library except as expressly provided under this License. Any
|
||||||
|
attempt otherwise to copy, modify, sublicense, link with, or
|
||||||
|
distribute the Library is void, and will automatically terminate your
|
||||||
|
rights under this License. However, parties who have received copies,
|
||||||
|
or rights, from you under this License will not have their licenses
|
||||||
|
terminated so long as such parties remain in full compliance.
|
||||||
|
|
||||||
|
9. You are not required to accept this License, since you have not
|
||||||
|
signed it. However, nothing else grants you permission to modify or
|
||||||
|
distribute the Library or its derivative works. These actions are
|
||||||
|
prohibited by law if you do not accept this License. Therefore, by
|
||||||
|
modifying or distributing the Library (or any work based on the
|
||||||
|
Library), you indicate your acceptance of this License to do so, and
|
||||||
|
all its terms and conditions for copying, distributing or modifying
|
||||||
|
the Library or works based on it.
|
||||||
|
|
||||||
|
10. Each time you redistribute the Library (or any work based on the
|
||||||
|
Library), the recipient automatically receives a license from the
|
||||||
|
original licensor to copy, distribute, link with or modify the Library
|
||||||
|
subject to these terms and conditions. You may not impose any further
|
||||||
|
restrictions on the recipients' exercise of the rights granted herein.
|
||||||
|
You are not responsible for enforcing compliance by third parties with
|
||||||
|
this License.
|
||||||
|
|
||||||
|
11. If, as a consequence of a court judgment or allegation of patent
|
||||||
|
infringement or for any other reason (not limited to patent issues),
|
||||||
|
conditions are imposed on you (whether by court order, agreement or
|
||||||
|
otherwise) that contradict the conditions of this License, they do not
|
||||||
|
excuse you from the conditions of this License. If you cannot
|
||||||
|
distribute so as to satisfy simultaneously your obligations under this
|
||||||
|
License and any other pertinent obligations, then as a consequence you
|
||||||
|
may not distribute the Library at all. For example, if a patent
|
||||||
|
license would not permit royalty-free redistribution of the Library by
|
||||||
|
all those who receive copies directly or indirectly through you, then
|
||||||
|
the only way you could satisfy both it and this License would be to
|
||||||
|
refrain entirely from distribution of the Library.
|
||||||
|
|
||||||
|
If any portion of this section is held invalid or unenforceable under any
|
||||||
|
particular circumstance, the balance of the section is intended to apply,
|
||||||
|
and the section as a whole is intended to apply in other circumstances.
|
||||||
|
|
||||||
|
It is not the purpose of this section to induce you to infringe any
|
||||||
|
patents or other property right claims or to contest validity of any
|
||||||
|
such claims; this section has the sole purpose of protecting the
|
||||||
|
integrity of the free software distribution system which is
|
||||||
|
implemented by public license practices. Many people have made
|
||||||
|
generous contributions to the wide range of software distributed
|
||||||
|
through that system in reliance on consistent application of that
|
||||||
|
system; it is up to the author/donor to decide if he or she is willing
|
||||||
|
to distribute software through any other system and a licensee cannot
|
||||||
|
impose that choice.
|
||||||
|
|
||||||
|
This section is intended to make thoroughly clear what is believed to
|
||||||
|
be a consequence of the rest of this License.
|
||||||
|
|
||||||
|
12. If the distribution and/or use of the Library is restricted in
|
||||||
|
certain countries either by patents or by copyrighted interfaces, the
|
||||||
|
original copyright holder who places the Library under this License may add
|
||||||
|
an explicit geographical distribution limitation excluding those countries,
|
||||||
|
so that distribution is permitted only in or among countries not thus
|
||||||
|
excluded. In such case, this License incorporates the limitation as if
|
||||||
|
written in the body of this License.
|
||||||
|
|
||||||
|
13. The Free Software Foundation may publish revised and/or new
|
||||||
|
versions of the Lesser General Public License from time to time.
|
||||||
|
Such new versions will be similar in spirit to the present version,
|
||||||
|
but may differ in detail to address new problems or concerns.
|
||||||
|
|
||||||
|
Each version is given a distinguishing version number. If the Library
|
||||||
|
specifies a version number of this License which applies to it and
|
||||||
|
"any later version", you have the option of following the terms and
|
||||||
|
conditions either of that version or of any later version published by
|
||||||
|
the Free Software Foundation. If the Library does not specify a
|
||||||
|
license version number, you may choose any version ever published by
|
||||||
|
the Free Software Foundation.
|
||||||
|
|
||||||
|
14. If you wish to incorporate parts of the Library into other free
|
||||||
|
programs whose distribution conditions are incompatible with these,
|
||||||
|
write to the author to ask for permission. For software which is
|
||||||
|
copyrighted by the Free Software Foundation, write to the Free
|
||||||
|
Software Foundation; we sometimes make exceptions for this. Our
|
||||||
|
decision will be guided by the two goals of preserving the free status
|
||||||
|
of all derivatives of our free software and of promoting the sharing
|
||||||
|
and reuse of software generally.
|
||||||
|
|
||||||
|
NO WARRANTY
|
||||||
|
|
||||||
|
15. BECAUSE THE LIBRARY IS LICENSED FREE OF CHARGE, THERE IS NO
|
||||||
|
WARRANTY FOR THE LIBRARY, TO THE EXTENT PERMITTED BY APPLICABLE LAW.
|
||||||
|
EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR
|
||||||
|
OTHER PARTIES PROVIDE THE LIBRARY "AS IS" WITHOUT WARRANTY OF ANY
|
||||||
|
KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE
|
||||||
|
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
||||||
|
PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE
|
||||||
|
LIBRARY IS WITH YOU. SHOULD THE LIBRARY PROVE DEFECTIVE, YOU ASSUME
|
||||||
|
THE COST OF ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
|
||||||
|
|
||||||
|
16. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN
|
||||||
|
WRITING WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY
|
||||||
|
AND/OR REDISTRIBUTE THE LIBRARY AS PERMITTED ABOVE, BE LIABLE TO YOU
|
||||||
|
FOR DAMAGES, INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR
|
||||||
|
CONSEQUENTIAL DAMAGES ARISING OUT OF THE USE OR INABILITY TO USE THE
|
||||||
|
LIBRARY (INCLUDING BUT NOT LIMITED TO LOSS OF DATA OR DATA BEING
|
||||||
|
RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD PARTIES OR A
|
||||||
|
FAILURE OF THE LIBRARY TO OPERATE WITH ANY OTHER SOFTWARE), EVEN IF
|
||||||
|
SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH
|
||||||
|
DAMAGES.
|
||||||
|
|
||||||
|
END OF TERMS AND CONDITIONS
|
||||||
|
|
||||||
|
How to Apply These Terms to Your New Libraries
|
||||||
|
|
||||||
|
If you develop a new library, and you want it to be of the greatest
|
||||||
|
possible use to the public, we recommend making it free software that
|
||||||
|
everyone can redistribute and change. You can do so by permitting
|
||||||
|
redistribution under these terms (or, alternatively, under the terms of the
|
||||||
|
ordinary General Public License).
|
||||||
|
|
||||||
|
To apply these terms, attach the following notices to the library. It is
|
||||||
|
safest to attach them to the start of each source file to most effectively
|
||||||
|
convey the exclusion of warranty; and each file should have at least the
|
||||||
|
"copyright" line and a pointer to where the full notice is found.
|
||||||
|
|
||||||
|
<one line to give the library's name and a brief idea of what it does.>
|
||||||
|
Copyright (C) <year> <name of author>
|
||||||
|
|
||||||
|
This library is free software; you can redistribute it and/or
|
||||||
|
modify it under the terms of the GNU Lesser General Public
|
||||||
|
License as published by the Free Software Foundation; either
|
||||||
|
version 2.1 of the License, or (at your option) any later version.
|
||||||
|
|
||||||
|
This library is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
Lesser General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU Lesser General Public
|
||||||
|
License along with this library; if not, write to the Free Software
|
||||||
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
|
||||||
|
Also add information on how to contact you by electronic and paper mail.
|
||||||
|
|
||||||
|
You should also get your employer (if you work as a programmer) or your
|
||||||
|
school, if any, to sign a "copyright disclaimer" for the library, if
|
||||||
|
necessary. Here is a sample; alter the names:
|
||||||
|
|
||||||
|
Yoyodyne, Inc., hereby disclaims all copyright interest in the
|
||||||
|
library `Frob' (a library for tweaking knobs) written by James Random Hacker.
|
||||||
|
|
||||||
|
<signature of Ty Coon>, 1 April 1990
|
||||||
|
Ty Coon, President of Vice
|
||||||
|
|
||||||
|
That's all there is to it!
|
||||||
|
|
||||||
|
|
104
Makefile
Normal file
104
Makefile
Normal file
@ -0,0 +1,104 @@
|
|||||||
|
# Copyright (c) 2004 Don Stewart - http://www.cse.unsw.edu.au/~dons
|
||||||
|
# LGPL version 2.1 or later (see http://www.gnu.org/copyleft/lesser.html)
|
||||||
|
|
||||||
|
# cut down reimplementation of $fptools/mk directory
|
||||||
|
|
||||||
|
.PHONY: build all
|
||||||
|
|
||||||
|
all: headers runplugs plugs
|
||||||
|
|
||||||
|
build:
|
||||||
|
cd src && $(MAKE)
|
||||||
|
|
||||||
|
plugs: build
|
||||||
|
( cd examples/hmake/lib-plugs ; $(MAKE) build )
|
||||||
|
cp examples/hmake/lib-plugs/plugs ./
|
||||||
|
|
||||||
|
runplugs: build
|
||||||
|
( cd examples/hmake/one-shot ; $(MAKE) build )
|
||||||
|
cp examples/hmake/one-shot/runplugs ./
|
||||||
|
|
||||||
|
headers: build
|
||||||
|
cp src/eval/Eval/Haskell_stub.h EvalHaskell.h
|
||||||
|
|
||||||
|
#
|
||||||
|
# installing
|
||||||
|
#
|
||||||
|
|
||||||
|
# TODO put these in subdirs
|
||||||
|
install:
|
||||||
|
$(INSTALL_DATA_DIR) $(LIBDIR)/include
|
||||||
|
$(INSTALL_DATA) EvalHaskell.h $(LIBDIR)/include
|
||||||
|
@(cd src && $(MAKE) install)
|
||||||
|
$(INSTALL_DATA_DIR) $(PREFIX)/bin
|
||||||
|
$(INSTALL_PROGRAM) plugs $(PREFIX)/bin/
|
||||||
|
$(INSTALL_PROGRAM) runplugs $(PREFIX)/bin/
|
||||||
|
|
||||||
|
#
|
||||||
|
# and register the library with ghc package system
|
||||||
|
# Use this target if installing by hand. May need to be performed as root
|
||||||
|
#
|
||||||
|
register:
|
||||||
|
env LIBDIR=${LIBDIR} $(GHC_PKG) -u < src/altdata/altdata.conf.in
|
||||||
|
env LIBDIR=${LIBDIR} $(GHC_PKG) -u < src/hi/hi.conf.in
|
||||||
|
env LIBDIR=${LIBDIR} $(GHC_PKG) -u < src/plugins/plugins.conf.in
|
||||||
|
env LIBDIR=${LIBDIR} $(GHC_PKG) -u < src/eval/eval.conf.in
|
||||||
|
env LIBDIR=${LIBDIR} $(GHC_PKG) -u < src/printf/printf.conf.in
|
||||||
|
|
||||||
|
# and unregister the packages
|
||||||
|
unregister:
|
||||||
|
$(GHC_PKG) -r printf
|
||||||
|
$(GHC_PKG) -r eval
|
||||||
|
$(GHC_PKG) -r plugins
|
||||||
|
$(GHC_PKG) -r hi
|
||||||
|
$(GHC_PKG) -r altdata
|
||||||
|
|
||||||
|
#
|
||||||
|
# regress check. TODO check expected output
|
||||||
|
#
|
||||||
|
check:
|
||||||
|
@if [ ! -f EvalHaskell.h ] ; then \
|
||||||
|
echo "run 'make' first" ; \
|
||||||
|
exit 1 ;\
|
||||||
|
fi
|
||||||
|
@( d=/tmp/plugins.tmp.$$$$ ; mkdir $$d ; export TMPDIR=$$d ;\
|
||||||
|
for i in `find examples ! -name CVS -type d -maxdepth 2 -mindepth 2` ; do \
|
||||||
|
printf "=== testing %-50s ... " "$$i" ; \
|
||||||
|
( cd $$i ; if [ -f dont_test ] ; then \
|
||||||
|
echo "ignored." ;\
|
||||||
|
else ${MAKE} -sk && ${MAKE} -ksi check |\
|
||||||
|
sed '/^Compil/d;/^Load/d;/Read/d;/Expan/d;/Savi/d;/Writ/d' ;\
|
||||||
|
${MAKE} -sk clean ;\
|
||||||
|
fi ) 2> /dev/null ;\
|
||||||
|
done ; rm -rf $$d )
|
||||||
|
|
||||||
|
|
||||||
|
#
|
||||||
|
# making clean
|
||||||
|
#
|
||||||
|
|
||||||
|
CLEAN_FILES += *.conf.*.old *~
|
||||||
|
EXTRA_CLEANS+=*.conf.inplace* *.conf.in *.h autom4te.cache \
|
||||||
|
config.h config.mk config.log config.status configure
|
||||||
|
|
||||||
|
clean:
|
||||||
|
cd docs && $(MAKE) clean
|
||||||
|
cd src && $(MAKE) clean
|
||||||
|
rm -rf $(CLEAN_FILES)
|
||||||
|
find examples -name '*.a' -exec rm {} \;
|
||||||
|
find examples -name '*~' -exec rm {} \;
|
||||||
|
find examples -name 'a.out' -exec rm {} \;
|
||||||
|
find examples -name '*.hi' -exec rm {} \;
|
||||||
|
find examples -name '*.o' -exec rm {} \;
|
||||||
|
find examples -name '*.core' -exec rm {} \;
|
||||||
|
find examples -name 'package.conf' -exec rm {} \;
|
||||||
|
rm -rf plugs
|
||||||
|
rm -rf runplugs
|
||||||
|
rm -rf examples/hmake/lib-plugs/plugs
|
||||||
|
rm -rf examples/hmake/one-shot/runplugs
|
||||||
|
rm -f EvalHaskell.h
|
||||||
|
|
||||||
|
distclean: clean
|
||||||
|
rm -rf $(EXTRA_CLEANS)
|
||||||
|
|
||||||
|
include config.mk
|
97
README
Normal file
97
README
Normal file
@ -0,0 +1,97 @@
|
|||||||
|
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
hs-plugins
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
|
||||||
|
Compiler and tool support for compiling and loading, and evaluating
|
||||||
|
Haskell at runtime.
|
||||||
|
|
||||||
|
The library provides a convenient interface to GHC's runtime loader
|
||||||
|
and linker, letting you load compiled Haskell code.
|
||||||
|
|
||||||
|
It also provides a `make' system for compiling plugin source
|
||||||
|
automagically and for combining the user's .hs file with a stub of
|
||||||
|
standard declarations and syntax, saving the user from having to write
|
||||||
|
standard code themselves.
|
||||||
|
|
||||||
|
It provides an eval() function, for generating new, well-typed,
|
||||||
|
compiled code from a Haskell source string.
|
||||||
|
|
||||||
|
It also provides a new variation of printf for Haskell-- a runtime
|
||||||
|
generated, dynamically-typed printf.
|
||||||
|
|
||||||
|
Read the documentation in doc/ for more.
|
||||||
|
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
DEPENDENCIES:
|
||||||
|
|
||||||
|
* Requires GNU make or BSD make to build
|
||||||
|
* Requires GHC > 6.2 (for Typeable.h)
|
||||||
|
* 'plugs' requires a working readline library.
|
||||||
|
|
||||||
|
* If you wish to use TH in plugins, or to run load()-programs in GHCi,
|
||||||
|
you require a patch to GHC's linker, that was committed into ghc
|
||||||
|
6.3, and ghc 6.2 -stable branch, and is available from 6.2.2 onwards.
|
||||||
|
|
||||||
|
* If you need to regenerate ./configure you need >= autoconf-2.53
|
||||||
|
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
BUILDING:
|
||||||
|
$ ./configure --prefix=/usr/local
|
||||||
|
$ make
|
||||||
|
$ make install
|
||||||
|
|
||||||
|
If you wish to register the libraries as official ghc pkg (probably as
|
||||||
|
root):
|
||||||
|
$ make register
|
||||||
|
|
||||||
|
And to unregister (maybe as root). Note that the unistall order
|
||||||
|
matters:
|
||||||
|
|
||||||
|
$ ghc-pkg -r printf
|
||||||
|
$ ghc-pkg -r eval
|
||||||
|
$ ghc-pkg -r plugins
|
||||||
|
$ ghc-pkg -r hi
|
||||||
|
$ ghc-pkg -r altdata
|
||||||
|
|
||||||
|
Once it is registered, you can link against the library by just adding
|
||||||
|
-package plugins or, e.g. -package eval, to your command line.
|
||||||
|
|
||||||
|
The documentation relies on latex, dvips, tex2page:
|
||||||
|
|
||||||
|
$ cd doc && make
|
||||||
|
|
||||||
|
EXAMPLES:
|
||||||
|
|
||||||
|
Have a look in the examples/ directory for many examples of how to
|
||||||
|
arrange your code.
|
||||||
|
|
||||||
|
LICENSE:
|
||||||
|
|
||||||
|
This library is distributed under the terms of the LGPL. The runtime
|
||||||
|
loader code is based on code written by André Pang, and others, and is
|
||||||
|
distributed under the BSD-style Glasgow University license.
|
||||||
|
|
||||||
|
PORTABILITY:
|
||||||
|
|
||||||
|
Requires GHC 6.2 or greater, though most testing has be done on 6.3.
|
||||||
|
They dynamic loader requires a functional GHCi implementation.
|
||||||
|
|
||||||
|
---------------------+--------------------------------------------------
|
||||||
|
Platform | Works Should work* Unknown Won't work
|
||||||
|
---------------------+--------------------------------------------------
|
||||||
|
i386-*-linux | X
|
||||||
|
i386-*-freebsd | X
|
||||||
|
i386-*-openbsd | X
|
||||||
|
powerpc-apple-darwin | X
|
||||||
|
sparc-*-solaris2 | X
|
||||||
|
ia64-*-linux | #
|
||||||
|
i386-*-solaris2 | X
|
||||||
|
sparc-*-linux | X
|
||||||
|
sparc-*-openbsd | X
|
||||||
|
i386-*-netbsd | X
|
||||||
|
amd64-*-openbsd | X
|
||||||
|
mips64-sgi-irix | X
|
||||||
|
---------------------+--------------------------------------------------
|
||||||
|
|
||||||
|
# .hi file parsing is currently broken
|
30
TODO
Normal file
30
TODO
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
For 0.1
|
||||||
|
----------
|
||||||
|
|
||||||
|
+ have eval, printf return errors as arguments, not to stdout
|
||||||
|
|
||||||
|
+ nice functions for cleaning up /tmp files, given a module name
|
||||||
|
|
||||||
|
+ PORTABILITY -- pretty much all of this is in main/SysTools.lhs in GHC
|
||||||
|
-- where to /tmp files go? Use SysTools code from GHC
|
||||||
|
-- need to dosify file names on in and out
|
||||||
|
-- try to confirm the implementation of forkProcess
|
||||||
|
|
||||||
|
+ write a script to strip down the release code.
|
||||||
|
|
||||||
|
+ .hi file parser is broken on Itanium, again.
|
||||||
|
|
||||||
|
+ Implement hs_eval by marshalling Dynamics across to the C side for
|
||||||
|
checking.
|
||||||
|
|
||||||
|
+ Make data structures used by the library Storable, for C programs
|
||||||
|
|
||||||
|
+ insert iface info into the state, building up a dependency graph like
|
||||||
|
hram's. use this to allow cascading unloading. Does anyone want this?
|
||||||
|
|
||||||
|
+ enable more .hi interface code to provide full GHC-like :t options
|
||||||
|
to plugs.
|
||||||
|
|
||||||
|
+ replace the String interface to eval with an ExpQ interface.
|
||||||
|
|
||||||
|
+ build way=p and way=''
|
6
autogen.sh
Normal file
6
autogen.sh
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
#!/bin/sh -x
|
||||||
|
|
||||||
|
# this is the world's most complicated autogen.sh script :)
|
||||||
|
|
||||||
|
exec autoconf
|
||||||
|
|
1354
config.guess
vendored
Normal file
1354
config.guess
vendored
Normal file
File diff suppressed because it is too large
Load Diff
21
config.h.in
Normal file
21
config.h.in
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2004 Don Stewart - http://www.cse.unsw.edu.au/~dons
|
||||||
|
* LGPL version 2.1 or later (see http://www.gnu.org/copyleft/lesser.html)
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* symbols that must be assigned to variables in Haskell code */
|
||||||
|
|
||||||
|
/* NOTE: this is not the same as symbols needed for cpp of .hs code */
|
||||||
|
|
||||||
|
/* path to ghc */
|
||||||
|
#define GHC "@GHC@"
|
||||||
|
|
||||||
|
/* path to GHC libraries */
|
||||||
|
#define GHC_LIB_PATH "@GHC_LIB_PATH@"
|
||||||
|
|
||||||
|
#define TOP "@TOP@"
|
||||||
|
|
||||||
|
#define LEADING_UNDERSCORE @LEADING_UNDERSCORE@
|
||||||
|
|
||||||
|
#define CABAL @CABAL@
|
||||||
|
|
57
config.mk.in
Normal file
57
config.mk.in
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2004 Don Stewart - http://www.cse.unsw.edu.au/~dons
|
||||||
|
# LGPL version 2.1 or later (see http://www.gnu.org/copyleft/lesser.html)
|
||||||
|
#
|
||||||
|
|
||||||
|
#
|
||||||
|
# config.mk.in :
|
||||||
|
# variables that need to be visible in Makefiles
|
||||||
|
#
|
||||||
|
|
||||||
|
# all is the default rule for everyone
|
||||||
|
default: all
|
||||||
|
|
||||||
|
PACKAGE = plugins
|
||||||
|
UPACKAGE = Plugins
|
||||||
|
|
||||||
|
TOP = @TOP@
|
||||||
|
|
||||||
|
PREFIX = @PREFIX@
|
||||||
|
LIBDIR = $(PREFIX)/lib/hs-$(PACKAGE)
|
||||||
|
BINDIR = $(PREFIX)/bin
|
||||||
|
|
||||||
|
WHOLE_ARCHIVE_FLAG = @WHOLE_ARCHIVE_FLAG@
|
||||||
|
|
||||||
|
# Are we using the new Cabal packages?
|
||||||
|
CABAL = @CABAL@
|
||||||
|
|
||||||
|
|
||||||
|
GHC = @GHC@
|
||||||
|
GHC_LIB_PATH = @GHC_LIB_PATH@
|
||||||
|
GHC_VERSION = @GHC_VERSION@
|
||||||
|
GLASGOW_HASKELL = @GLASGOW_HASKELL@
|
||||||
|
GHC_EXTRA_OPTS = @SYMS@ @DEBUG_OPTS@
|
||||||
|
GHC_LD_OPTS =
|
||||||
|
|
||||||
|
GHC_PKG = @GHCPKG@-@GHC_VERSION@
|
||||||
|
|
||||||
|
LD = @LD@
|
||||||
|
LD_X = -x
|
||||||
|
|
||||||
|
HAPPY = @HAPPY@
|
||||||
|
HAPPY_OPTS = -a -g -c
|
||||||
|
ALEX = @ALEX@
|
||||||
|
ALEX_OPTS = --ghc
|
||||||
|
HADDOCK = @HADDOCK@
|
||||||
|
|
||||||
|
AR = @AR@
|
||||||
|
RANLIB = @RANLIB@
|
||||||
|
|
||||||
|
RM = @RM@
|
||||||
|
|
||||||
|
INSTALL = @INSTALL@
|
||||||
|
|
||||||
|
# A few aliases
|
||||||
|
INSTALL_PROGRAM = ${INSTALL} -s -m 755
|
||||||
|
INSTALL_DATA = ${INSTALL} -m 644
|
||||||
|
INSTALL_DATA_DIR= ${INSTALL} -d -m 755
|
1460
config.sub
vendored
Normal file
1460
config.sub
vendored
Normal file
File diff suppressed because it is too large
Load Diff
198
configure.ac
Normal file
198
configure.ac
Normal file
@ -0,0 +1,198 @@
|
|||||||
|
#
|
||||||
|
# Copyright (c) 2004 Don Stewart - http://www.cse.unsw.edu.au/~dons
|
||||||
|
# LGPL version 2.1 or later (see http://www.gnu.org/copyleft/lesser.html)
|
||||||
|
#
|
||||||
|
|
||||||
|
# sanity test
|
||||||
|
AC_INIT(src/plugins/Plugins.hs)
|
||||||
|
|
||||||
|
# untested on earlier than 2.52, but it won't work anyway
|
||||||
|
AC_PREREQ(2.53)
|
||||||
|
|
||||||
|
# Find out what type of system we're running on
|
||||||
|
AC_CANONICAL_BUILD
|
||||||
|
|
||||||
|
PREFIX="$prefix"
|
||||||
|
if test "$prefix" = "NONE"
|
||||||
|
then
|
||||||
|
PREFIX="$ac_default_prefix"
|
||||||
|
fi
|
||||||
|
AC_SUBST(PREFIX)
|
||||||
|
|
||||||
|
Platform="$build_cpu-$build_vendor-$build_os"
|
||||||
|
|
||||||
|
case $Platform in
|
||||||
|
powerpc-apple-darwin*)
|
||||||
|
MACOSX=yes
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
MACOSX=no
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
if test "$MACOSX" = "yes"
|
||||||
|
then
|
||||||
|
WHOLE_ARCHIVE_FLAG=-all_load
|
||||||
|
LEADING_UNDERSCORE=1
|
||||||
|
else
|
||||||
|
WHOLE_ARCHIVE_FLAG=--whole-archive
|
||||||
|
LEADING_UNDERSCORE=0
|
||||||
|
fi
|
||||||
|
|
||||||
|
AC_SUBST(WHOLE_ARCHIVE_FLAG)
|
||||||
|
AC_SUBST(LEADING_UNDERSCORE)
|
||||||
|
|
||||||
|
|
||||||
|
TOP=`pwd`
|
||||||
|
AC_SUBST(TOP)
|
||||||
|
|
||||||
|
# necessary tools
|
||||||
|
|
||||||
|
# allow user supplied haskell compiler
|
||||||
|
AC_ARG_WITH(ghc,
|
||||||
|
AC_HELP_STRING([--with-ghc=<ghc>],[use a specific Haskell compiler]),
|
||||||
|
[ GHC="$withval"
|
||||||
|
if test ! -f "$GHC" ; then
|
||||||
|
AC_MSG_ERROR([$GHC not found. You need GHC to build this project])
|
||||||
|
fi
|
||||||
|
],
|
||||||
|
[ AC_CHECK_PROG(GHC,ghc,ghc)
|
||||||
|
if test -z "$GHC" ; then
|
||||||
|
AC_MSG_ERROR([You need GHC to build this project])
|
||||||
|
fi
|
||||||
|
]
|
||||||
|
)
|
||||||
|
AC_SUBST(GHC)
|
||||||
|
|
||||||
|
# find path to GHC libs, for runtime_loader
|
||||||
|
if test -n "$GHC" ; then
|
||||||
|
AC_MSG_CHECKING([for ghc library directory])
|
||||||
|
GHC_LIB_PATH=`$GHC --print-libdir`
|
||||||
|
AC_MSG_RESULT([$GHC_LIB_PATH])
|
||||||
|
fi
|
||||||
|
AC_SUBST(GHC_LIB_PATH)
|
||||||
|
|
||||||
|
# check ghc version here
|
||||||
|
if test -n "$GHC" ; then
|
||||||
|
AC_MSG_CHECKING([for ghc version])
|
||||||
|
GHC_VERSION=`$GHC --numeric-version`
|
||||||
|
AC_MSG_RESULT([$GHC_VERSION])
|
||||||
|
fi
|
||||||
|
AC_SUBST(GHC_VERSION)
|
||||||
|
|
||||||
|
# Work out value of __GLASGOW_HASKELL__
|
||||||
|
if test -n "$GHC" ; then
|
||||||
|
AC_MSG_CHECKING([for value of __GLASGOW_HASKELL__])
|
||||||
|
echo "main = print __GLASGOW_HASKELL__" > t.hs
|
||||||
|
GLASGOW_HASKELL=`echo 'main' | "$GHC" --interactive -v0 -cpp t.hs`
|
||||||
|
rm t.hs
|
||||||
|
AC_MSG_RESULT([$GLASGOW_HASKELL])
|
||||||
|
fi
|
||||||
|
AC_SUBST(GLASGOW_HASKELL)
|
||||||
|
|
||||||
|
dnl ** quote char breaks sed
|
||||||
|
changequote(, )dnl
|
||||||
|
MAJOR=`echo "$GHC_VERSION" | sed 's/^\([^\.]*\)\.\([^\.]*\).*/\1/'`
|
||||||
|
MINOR=`echo "$GHC_VERSION" | sed 's/^\([^\.]*\)\.\([^\.]*\).*/\2/'`
|
||||||
|
changequote([, ])dnl
|
||||||
|
|
||||||
|
if test "$MAJOR" -lt "6"; then
|
||||||
|
AC_MSG_ERROR(Found major $MAJOR. You need a ghc version >= 6.2) ;
|
||||||
|
fi
|
||||||
|
if test "$MINOR" -lt "2"; then
|
||||||
|
AC_MSG_ERROR(You need a ghc version >= 6.2) ;
|
||||||
|
fi
|
||||||
|
|
||||||
|
#Allow plugins to be built with Cabal libraries
|
||||||
|
AC_ARG_ENABLE(cabal,
|
||||||
|
[ --enable-cabal Enable use of Cabal packages in pluggable-1-branch
|
||||||
|
of GHC],
|
||||||
|
[ CABAL=1 ],
|
||||||
|
[ CABAL=0 ])
|
||||||
|
|
||||||
|
# used by the Makefile`s to alter dependencies.
|
||||||
|
if test "$MAJOR" -ge "6" -a "$MINOR" -ge "4"; then
|
||||||
|
CABAL=1
|
||||||
|
fi
|
||||||
|
|
||||||
|
AC_SUBST(CABAL)
|
||||||
|
|
||||||
|
# Allow a debugging version of hs-plugins to be built
|
||||||
|
AC_ARG_ENABLE(debug,
|
||||||
|
[ --enable-debug Enable a debug version of hs-plugins to be built],
|
||||||
|
[ DEBUG_OPTS=-DDEBUG ],
|
||||||
|
[ DEBUG_OPTS= ])
|
||||||
|
|
||||||
|
AC_SUBST(DEBUG_OPTS)
|
||||||
|
|
||||||
|
# allow user supplied haskell package manager
|
||||||
|
AC_ARG_WITH(ghc-pkg,
|
||||||
|
AC_HELP_STRING([--with-ghc-pkg=<ghc-pkg>],[use a specific ghc-pkg]),
|
||||||
|
[ GHCPKG="$withval"
|
||||||
|
if test ! -f "$GHCPKG" ; then
|
||||||
|
AC_MSG_ERROR([$GHCPKG not found. You need ghc-pkg])
|
||||||
|
fi
|
||||||
|
],
|
||||||
|
[ AC_CHECK_PROG(GHCPKG,ghc-pkg,ghc-pkg)
|
||||||
|
if test -z "$GHCPKG" ; then
|
||||||
|
AC_MSG_ERROR([You need ghc-pkg])
|
||||||
|
fi
|
||||||
|
]
|
||||||
|
)
|
||||||
|
|
||||||
|
AC_SUBST(GHCPKG)
|
||||||
|
|
||||||
|
AC_CHECK_PROG(HADDOCK,haddock,haddock)
|
||||||
|
if test -z "$HADDOCK" ; then
|
||||||
|
AC_MSG_WARN(You need Haddock if you want developer documentation)
|
||||||
|
fi
|
||||||
|
|
||||||
|
AC_CHECK_PROG(HAPPY,happy,happy)
|
||||||
|
if test -z "$HAPPY" ; then
|
||||||
|
AC_MSG_WARN(If you change or remove the parser you'll need Happy)
|
||||||
|
fi
|
||||||
|
|
||||||
|
AC_CHECK_PROG(ALEX,alex,alex)
|
||||||
|
if test -z "$ALEX" ; then
|
||||||
|
AC_MSG_WARN(If you change or remove the lexer files you'll need alex)
|
||||||
|
fi
|
||||||
|
|
||||||
|
AC_CHECK_PROG(LD,ld,ld)
|
||||||
|
if test -z "$LD" ; then
|
||||||
|
AC_MSG_WARN(You need ld -export-dynamic)
|
||||||
|
fi
|
||||||
|
|
||||||
|
AC_CHECK_PROG(AR,ar,ar)
|
||||||
|
if test -z "$AR" ; then
|
||||||
|
AC_MSG_WARN(You need ar to build the library)
|
||||||
|
fi
|
||||||
|
|
||||||
|
AC_CHECK_PROG(RANLIB,ranlib,ranlib)
|
||||||
|
if test -z "$RANLIB" ; then
|
||||||
|
AC_MSG_WARN(You need randlib to build the library)
|
||||||
|
fi
|
||||||
|
|
||||||
|
AC_CHECK_PROG(RM,rm,rm)
|
||||||
|
if test -z "$RM" ; then
|
||||||
|
AC_MSG_WARN(You need rm!)
|
||||||
|
fi
|
||||||
|
|
||||||
|
AC_CHECK_PROG(TEX,tex,tex)
|
||||||
|
if test -z "$TEX" ; then
|
||||||
|
AC_MSG_WARN(You'll need tex if you wish to build the documentation)
|
||||||
|
fi
|
||||||
|
AC_CHECK_PROG(TEX2PAGE,tex2page,tex2page)
|
||||||
|
if test -z "$TEX2PAGE" ; then
|
||||||
|
AC_MSG_WARN(You'll need tex2page if you wish to build the
|
||||||
|
documentation: http://www.ccs.neu.edu/home/dorai/tex2page/tex2page-doc.html)
|
||||||
|
fi
|
||||||
|
|
||||||
|
AC_CHECK_FUNC(arc4random, [SYMS="$SYMS -DHAVE_ARC4RANDOM"])
|
||||||
|
|
||||||
|
AC_SUBST(SYMS)
|
||||||
|
|
||||||
|
AC_PROG_INSTALL
|
||||||
|
|
||||||
|
AC_CONFIG_FILES(config.mk config.h)
|
||||||
|
|
||||||
|
AC_OUTPUT
|
||||||
|
|
31
docs/Makefile
Normal file
31
docs/Makefile
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
# Copyright (c) 2004 Don Stewart - http://www.cse.unsw.edu.au/~dons
|
||||||
|
# LGPL version 2.1 or later (see http://www.gnu.org/copyleft/lesser.html)
|
||||||
|
|
||||||
|
.PHONY: build clean html
|
||||||
|
|
||||||
|
SRC = hs-plugins
|
||||||
|
|
||||||
|
build: $(SRC).ps html
|
||||||
|
|
||||||
|
$(SRC).ps: $(SRC).dvi
|
||||||
|
dvips -f $(SRC).dvi > $@
|
||||||
|
|
||||||
|
html: $(SRC).tex
|
||||||
|
tex2page $(SRC)
|
||||||
|
tex2page $(SRC)
|
||||||
|
./munge.sed < $(SRC)/$(SRC).html > tmp.out
|
||||||
|
mv tmp.out $(SRC)/$(SRC).html
|
||||||
|
cp $(SRC)/$(SRC).html $(SRC)/index.html
|
||||||
|
tar czf $(SRC).html.tar.gz $(SRC)
|
||||||
|
mv $(SRC).html.tar.gz $(SRC)/
|
||||||
|
|
||||||
|
$(SRC).dvi: $(SRC).tex
|
||||||
|
latex $(SRC).tex && latex $(SRC).tex
|
||||||
|
|
||||||
|
CLEANS= *.{ps,dvi,aux,log} *~ hs-plugins *-Z-* *.toc
|
||||||
|
|
||||||
|
clean:
|
||||||
|
rm -rf $(CLEANS)
|
||||||
|
|
||||||
|
all: doc
|
||||||
|
|
452
docs/haskell.sty
Normal file
452
docs/haskell.sty
Normal file
@ -0,0 +1,452 @@
|
|||||||
|
%%% This is a LaTeX2e style file.
|
||||||
|
%%%
|
||||||
|
%%% It supports setting functional languages like Haskell.
|
||||||
|
%%%
|
||||||
|
%%% Manuel M. T. Chakravarty <chak@cse.unsw.edu.au> [1998..2000]
|
||||||
|
%%%
|
||||||
|
%%% $Id: haskell.sty,v 1.2 2004/05/16 08:20:09 dons Exp $
|
||||||
|
%%%
|
||||||
|
%%% This file is free software; you can redistribute it and/or modify
|
||||||
|
%%% it under the terms of the GNU General Public License as published by
|
||||||
|
%%% the Free Software Foundation; either version 2 of the License, or
|
||||||
|
%%% (at your option) any later version.
|
||||||
|
%%%
|
||||||
|
%%% This file is distributed in the hope that it will be useful,
|
||||||
|
%%% but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
%%% MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
%%% GNU General Public License for more details.
|
||||||
|
%%%
|
||||||
|
%%% Acknowledegments ==========================================================
|
||||||
|
%%%
|
||||||
|
%%% Thanks to Gabriele Keller <keller@cs.tu-berlin.de> for beta testing and
|
||||||
|
%%% code contributions. Thanks to the LaTeX3 project for improving the LaTeX
|
||||||
|
%%% sources (which helped me writing this code). Furthermore, I am grateful
|
||||||
|
%%% to Martin Erwig <Martin.Erwig@FernUni-Hagen.de> for feedback and
|
||||||
|
%%% suggestions, and to Conal Elliott <conal@MICROSOFT.com> for pointing out
|
||||||
|
%%% a tricky bug.
|
||||||
|
%%%
|
||||||
|
%%% TODO ======================================================================
|
||||||
|
%%%
|
||||||
|
%%% B ~ bug; F ~ feature
|
||||||
|
%%%
|
||||||
|
%%% * F: Along the lines of the discussion with Martin Erwig add support for
|
||||||
|
%%% keywords etc (see the emails)
|
||||||
|
%%%
|
||||||
|
%%% * B: If we have as input
|
||||||
|
%%%
|
||||||
|
%%% \<map
|
||||||
|
%%% g\>
|
||||||
|
%%%
|
||||||
|
%%% there won't be a `\hsap' inserted!! (Can this be solved by using
|
||||||
|
%%% \obeylines in \<...\>?)
|
||||||
|
%%%
|
||||||
|
%%% * B: A \relax is needed after a & if it immediately followed by a \hsbody{}
|
||||||
|
%%% (See TeXbook, S.240)
|
||||||
|
%%%
|
||||||
|
%%% * F: Implement a \hstext{...} as \(\text{...}\).
|
||||||
|
%%%
|
||||||
|
%%% * We would like hswhere* etc that are like haskell* (\hsalign already
|
||||||
|
%%% supports this, ie, there is a \hsalign*).
|
||||||
|
%%%
|
||||||
|
%%% * Star-Versions of if, let etc that use a single line layout (maybe not
|
||||||
|
%%% with star, because of the above).
|
||||||
|
%%%
|
||||||
|
%%% * Support for enforcing and prohibiting breaks in `haskell' displays.
|
||||||
|
%%%
|
||||||
|
%%% * Comments in a let-in should be aligned in the same way for the bindings
|
||||||
|
%%% and the body.
|
||||||
|
%%%
|
||||||
|
%%% * It would be nice to have different styles (indentation after in of
|
||||||
|
%%% let-in or not) etc; either to be set with a package option or in the
|
||||||
|
%%% preamble (the latter probably makes more sense).
|
||||||
|
%%%
|
||||||
|
%%% * Literate programming facility: Variant of the `haskell' env (maybe
|
||||||
|
%%% `hschunk', which is named and can be used in other chunks). But maybe
|
||||||
|
%%% it is not necessary to provide a chunk-based reordering mechanism,
|
||||||
|
%%% because most of the Haskell stuff can be in any order anyway...
|
||||||
|
%%% Important is to provide a way to define visually pleasing layout
|
||||||
|
%%% together with the raw Haskell form for program output. (Maybe `haskell*'
|
||||||
|
%%% as Haskell env that outputs its contents?)
|
||||||
|
%%%
|
||||||
|
|
||||||
|
%% Initialization
|
||||||
|
%% ==============
|
||||||
|
|
||||||
|
\NeedsTeXFormat{LaTeX2e}
|
||||||
|
\ProvidesPackage{haskell}[2000/10/05 v1.0e Chilli's Haskell Style]
|
||||||
|
|
||||||
|
|
||||||
|
%% Parameters
|
||||||
|
%% ==========
|
||||||
|
|
||||||
|
\newskip\hsmargin
|
||||||
|
\hsmargin\leftmargini
|
||||||
|
|
||||||
|
|
||||||
|
%% Main macros and environments
|
||||||
|
%% ============================
|
||||||
|
|
||||||
|
% applications
|
||||||
|
%
|
||||||
|
\newcommand{\hsap}{% % application by juxtaposition
|
||||||
|
\unskip\mskip 4mu plus 1mu} % only the last \hsap counts
|
||||||
|
|
||||||
|
% commands to start and stop setting spaces as \hsap
|
||||||
|
%
|
||||||
|
{\obeyspaces\gdef\@hsSpaceToApp{\obeyspaces\let =\hsap}} % spaces matter!!!
|
||||||
|
{\obeyspaces\gdef\@hsNormalSpace{\let =\space}}
|
||||||
|
|
||||||
|
% commands to start and stop treating numbers specially, ie, we don't want
|
||||||
|
% them to be affected by font changing commands in Haskell contexts as this
|
||||||
|
% would give italic numerals; the trick is to redefine their math code such
|
||||||
|
% that they go into math class 0 and thus don't change families (cf. `The
|
||||||
|
% TeXbook', Chapter 17, pp152)
|
||||||
|
%
|
||||||
|
\newcommand{\@hsRmNumbers}{%
|
||||||
|
\mathcode`0="0030
|
||||||
|
\mathcode`1="0031
|
||||||
|
\mathcode`2="0032
|
||||||
|
\mathcode`3="0033
|
||||||
|
\mathcode`4="0034
|
||||||
|
\mathcode`5="0035
|
||||||
|
\mathcode`6="0036
|
||||||
|
\mathcode`7="0037
|
||||||
|
\mathcode`8="0038
|
||||||
|
\mathcode`9="0039
|
||||||
|
}
|
||||||
|
\newcommand{\@hsNormalNumbers}{%
|
||||||
|
\mathcode`0="7030
|
||||||
|
\mathcode`1="7031
|
||||||
|
\mathcode`2="7032
|
||||||
|
\mathcode`3="7033
|
||||||
|
\mathcode`4="7034
|
||||||
|
\mathcode`5="7035
|
||||||
|
\mathcode`6="7036
|
||||||
|
\mathcode`7="7037
|
||||||
|
\mathcode`8="7038
|
||||||
|
\mathcode`9="7039
|
||||||
|
}
|
||||||
|
|
||||||
|
% Save the bindings of the standard math commands
|
||||||
|
%
|
||||||
|
% This is somewhat subtle as we want to able to enter the original math mode
|
||||||
|
% within Haskell mode and we have to ensure that the different opening
|
||||||
|
% commands are matched by the correct versions of the closing commands.
|
||||||
|
%
|
||||||
|
\let\@hsmathorg=\(
|
||||||
|
\let\@hsmathendorg=\)
|
||||||
|
\let\hs@crorg=\\
|
||||||
|
\newcommand{\@hsmath}{%
|
||||||
|
\relax\hbox\bgroup
|
||||||
|
\@hsNormalSpace
|
||||||
|
\@hsNormalNumbers
|
||||||
|
\let\(=\@hsmathorgx
|
||||||
|
\let\)=\@hsmathend
|
||||||
|
\def\\{\hs@crorg}%
|
||||||
|
\@hsmathorg
|
||||||
|
}
|
||||||
|
\newcommand{\@hsmathend}{%
|
||||||
|
\@hsmathendorg
|
||||||
|
\egroup
|
||||||
|
}
|
||||||
|
\newcommand{\@hsmathorgx}{%
|
||||||
|
\relax\@hsmathorg
|
||||||
|
\let\)=\@hsmathendorg
|
||||||
|
}
|
||||||
|
|
||||||
|
%% Typesetting of Haskell
|
||||||
|
%% ======================
|
||||||
|
|
||||||
|
% Inline Haskell phrases are delimited by `\<' and `\>'.
|
||||||
|
%
|
||||||
|
% Note: `\>' is only locally redefined.
|
||||||
|
%
|
||||||
|
\newcommand{\<}{%
|
||||||
|
\@hsmathorg
|
||||||
|
\mathit\bgroup
|
||||||
|
\@hsSpaceToApp
|
||||||
|
\@hsRmNumbers
|
||||||
|
\let\>=\@endhs
|
||||||
|
\let\(=\@hsmath
|
||||||
|
\def\\{\cr} % for Haskell alignments
|
||||||
|
}
|
||||||
|
\newcommand{\@endhs}{%
|
||||||
|
\egroup
|
||||||
|
\@hsmathendorg
|
||||||
|
}
|
||||||
|
|
||||||
|
% Displayed Haskell (environment `haskell' and `haskell*')
|
||||||
|
%
|
||||||
|
% There are two kind of preambles for \halign: \hs@preambleNorm is for
|
||||||
|
% `amsmath' style alignments and \hs@preambleStar for `equation' style
|
||||||
|
% alignments (but with an unbound number of columns to its right)
|
||||||
|
%
|
||||||
|
% We need #### to get a ## in the \edef building the \halign command.
|
||||||
|
%
|
||||||
|
% first the preambles (also used in \hs@align below):
|
||||||
|
%
|
||||||
|
\def\hs@preambleNorm{%
|
||||||
|
\noexpand\<####\unskip\noexpand\>\hfil&&\noexpand%
|
||||||
|
\<{}####\unskip\noexpand\>\hfil}
|
||||||
|
\def\hs@preambleStar{%
|
||||||
|
\noexpand\<####\unskip\noexpand\>\hfil&\hfil\noexpand%
|
||||||
|
\<{}####\unskip{}\noexpand\>\hfil&&\noexpand\<{}####\noexpand\>\hfil}
|
||||||
|
%
|
||||||
|
% the environments:
|
||||||
|
%
|
||||||
|
\newenvironment{haskell}{%
|
||||||
|
\@haskell\hs@preambleNorm}{%
|
||||||
|
\@endhaskell
|
||||||
|
}
|
||||||
|
\newenvironment{haskell*}{%
|
||||||
|
\@haskell\hs@preambleStar}{%
|
||||||
|
\@endhaskell
|
||||||
|
}
|
||||||
|
%
|
||||||
|
% auxiliary definition getting the preamble as its first argument and starting
|
||||||
|
% the environment:
|
||||||
|
%
|
||||||
|
\def\@haskell#1{%
|
||||||
|
\bgroup
|
||||||
|
\vspace\abovedisplayskip
|
||||||
|
\let\(=\@hsmath % Important when `\(' occurs after `&'!
|
||||||
|
\edef\@preamble{%
|
||||||
|
\halign\bgroup\hskip\hsmargin#1\cr}
|
||||||
|
\@preamble
|
||||||
|
}
|
||||||
|
%
|
||||||
|
% Auxiliary definition ending environment:
|
||||||
|
%
|
||||||
|
\def\@endhaskell{%
|
||||||
|
\crcr\egroup
|
||||||
|
\vspace\belowdisplayskip
|
||||||
|
\egroup\noindent\ignorespaces\global\@ignoretrue%
|
||||||
|
}
|
||||||
|
|
||||||
|
% single line comment and keyword style
|
||||||
|
%
|
||||||
|
\newcommand{\hscom}[1]{%
|
||||||
|
\relax\(\quad\textnormal{--- #1}\)}
|
||||||
|
\newcommand{\hskwd}[1]{%
|
||||||
|
\mathbf{#1}}
|
||||||
|
|
||||||
|
% informal description
|
||||||
|
%
|
||||||
|
\newcommand{\hsinf}[1]{%
|
||||||
|
\(\langle\textnormal{#1}\rangle\)}
|
||||||
|
|
||||||
|
% literals and some special symbols
|
||||||
|
%
|
||||||
|
\newcommand{\hschar}[1]{\textrm'\mathrm{#1}\textrm'} % character literals
|
||||||
|
\newcommand{\hsstr}[1]{"\mathrm{#1}"} % strings literals
|
||||||
|
\newcommand{\hsfrom}{\leftarrow} % <-
|
||||||
|
|
||||||
|
% aligned subphrases
|
||||||
|
%
|
||||||
|
% check for an optional star and combine prefix (in #1) with one of the two
|
||||||
|
% preambles (with star means to center the material between the first and
|
||||||
|
% second &)
|
||||||
|
%
|
||||||
|
\def\hs@align#1{%
|
||||||
|
\@ifstar
|
||||||
|
{\hs@align@pre{#1\hs@preambleStar}}%
|
||||||
|
{\hs@align@pre{#1\hs@preambleNorm}}%
|
||||||
|
}
|
||||||
|
%
|
||||||
|
% test for optional argument; #1: preamble
|
||||||
|
%
|
||||||
|
\def\hs@align@pre#1{%
|
||||||
|
\@testopt{\hs@align@prealign#1}t}
|
||||||
|
%
|
||||||
|
% got all arguments, now for the real code; #1: preamble; #2: alignment;
|
||||||
|
% #3: body (the material set by the \halign)
|
||||||
|
%
|
||||||
|
\def\hs@align@prealign#1[#2]#3{%
|
||||||
|
\relax\(
|
||||||
|
\edef\@preamble{%
|
||||||
|
\halign\bgroup#1\cr}
|
||||||
|
\if #2t\vtop \else \if#2b\vbox \else \vcenter \fi\fi
|
||||||
|
\bgroup%
|
||||||
|
\@preamble
|
||||||
|
#3%
|
||||||
|
\crcr\egroup%
|
||||||
|
\egroup\)
|
||||||
|
}
|
||||||
|
%
|
||||||
|
% user-level command: alignment without a prefix
|
||||||
|
%
|
||||||
|
\newcommand{\hsalign}{%
|
||||||
|
\relax
|
||||||
|
\hs@align\relax%
|
||||||
|
}
|
||||||
|
|
||||||
|
% subphrase breaking the surrounding alignment being flushed left
|
||||||
|
%
|
||||||
|
\newcommand{\hsnoalign}[1]{%
|
||||||
|
\noalign{%
|
||||||
|
\hs@align{\hskip\hsmargin}{#1}%
|
||||||
|
}%
|
||||||
|
}
|
||||||
|
|
||||||
|
% body expression breaking the surrounding alignment
|
||||||
|
%
|
||||||
|
% * setting \hsmargin to 0pt within the preamble (and _after_ it is used in
|
||||||
|
% the preamble) is crucial, as we want \hsmargin only to be applied in
|
||||||
|
% _outermost_ alignments
|
||||||
|
%
|
||||||
|
\newcommand{\hsbody}[1]{%
|
||||||
|
{}\\
|
||||||
|
\noalign{%
|
||||||
|
\hs@align{\hskip\hsmargin\quad\hsmargin0pt}{#1}%
|
||||||
|
}%
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
%% Defining commands for use in the Haskell mode
|
||||||
|
%% =============================================
|
||||||
|
%%
|
||||||
|
%% We use some of the low-level machinery defined in LaTeX's source file
|
||||||
|
%% `ltdefns.dtx'.
|
||||||
|
%%
|
||||||
|
%% \hscommand is similar to \newcommand, but there is no *-version.
|
||||||
|
%%
|
||||||
|
%% We use our own definitions here to insert a strategic `\relax' (see below)
|
||||||
|
%% and to obey spaces within the bodies of Haskell definitions.
|
||||||
|
|
||||||
|
\newcommand{\hscommand}[1]{\@testopt{\hs@newcommand#1}0}
|
||||||
|
\def\hs@newcommand#1[#2]{%
|
||||||
|
\obeyspaces % spaces count in Haskell macros
|
||||||
|
\@ifnextchar [{\hs@xargdef#1[#2]}%
|
||||||
|
{\hs@argdef#1[#2]}}
|
||||||
|
|
||||||
|
% All this trouble only to be able to add the `\relax' into the expansion
|
||||||
|
% process. If we don't that, commands without optional arguments when
|
||||||
|
% invoked after an alignment character & don't work properly (actually, the
|
||||||
|
% \obeyspaces doesn't work). I am sure that has to do with the scanning for
|
||||||
|
% \omit etc in \halign (TeXbook, p240), but I don't understand yet why it
|
||||||
|
% is problematic in this case.
|
||||||
|
%
|
||||||
|
% Furthermore, we switch off \obeyspaces in the end.
|
||||||
|
%
|
||||||
|
\long\def\hs@argdef#1[#2]#3{%
|
||||||
|
\@ifdefinable#1{%
|
||||||
|
\expandafter\def\expandafter#1\expandafter{%
|
||||||
|
\relax % in order to stop token expansion after &
|
||||||
|
\csname\string#1\expandafter\endcsname}%
|
||||||
|
\expandafter\@yargdef
|
||||||
|
\csname\string#1\endcsname
|
||||||
|
\@ne
|
||||||
|
{#2}%
|
||||||
|
{#3}}%
|
||||||
|
\catcode`\ =10% % stop obeying spaces now
|
||||||
|
}
|
||||||
|
|
||||||
|
% Switch off \obeyspaces in the end.
|
||||||
|
%
|
||||||
|
\long\def\hs@xargdef#1[#2][#3]#4{%
|
||||||
|
\@ifdefinable#1{%
|
||||||
|
\expandafter\def\expandafter#1\expandafter{%
|
||||||
|
\expandafter
|
||||||
|
\@protected@testopt
|
||||||
|
\expandafter
|
||||||
|
#1%
|
||||||
|
\csname\string#1\expandafter\endcsname
|
||||||
|
{#3}}%
|
||||||
|
\expandafter\@yargdef
|
||||||
|
\csname\string#1\endcsname
|
||||||
|
\tw@
|
||||||
|
{#2}%
|
||||||
|
{#4}}%
|
||||||
|
\catcode`\ =10% % stop obeying spaces now
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
%% Abbreviations
|
||||||
|
%% =============
|
||||||
|
|
||||||
|
% infix operators
|
||||||
|
%
|
||||||
|
\newcommand{\hsapp}{\mathbin{+\mkern-7mu+}}
|
||||||
|
\newcommand{\hsifix}[1]{\mathbin{\string`#1\string`}}
|
||||||
|
|
||||||
|
% let expression
|
||||||
|
%
|
||||||
|
\hscommand{\hslet}[3][t]{%
|
||||||
|
\hsalign[#1]{%
|
||||||
|
\hskwd{let}\\
|
||||||
|
\quad\hsalign{#2}\\
|
||||||
|
\hskwd{in}\\
|
||||||
|
#3
|
||||||
|
}%
|
||||||
|
}
|
||||||
|
|
||||||
|
% if expression
|
||||||
|
%
|
||||||
|
\hscommand{\hsif}[4][t]{%
|
||||||
|
\hsalign[#1]{%
|
||||||
|
\hskwd{if} #2 \hskwd{then}\\
|
||||||
|
\quad\hsalign{#3}\\
|
||||||
|
\hskwd{else}\\
|
||||||
|
\quad\hsalign{#4}%
|
||||||
|
}%
|
||||||
|
}
|
||||||
|
|
||||||
|
% case expression
|
||||||
|
%
|
||||||
|
\hscommand{\hscase}[3][t]{%
|
||||||
|
\hsalign[#1]{%
|
||||||
|
\hskwd{case} #2 \hskwd{of}\\
|
||||||
|
\quad\hsalign{#3}%
|
||||||
|
}%
|
||||||
|
}
|
||||||
|
|
||||||
|
% where clause
|
||||||
|
%
|
||||||
|
% * it is important to take the \quad into the preamble, so that nested
|
||||||
|
% \noaligns can break it
|
||||||
|
%
|
||||||
|
\hscommand{\hswhere}[1]{%
|
||||||
|
\hsbody{%
|
||||||
|
\hskwd{where}\\
|
||||||
|
\hs@align{\quad}{#1}%
|
||||||
|
}%
|
||||||
|
}
|
||||||
|
|
||||||
|
% do expression
|
||||||
|
%
|
||||||
|
\hscommand{\hsdo}[2][t]{%
|
||||||
|
\hsalign[#1]{%
|
||||||
|
\hskwd{do}\\
|
||||||
|
\quad\hsalign{#2}\\
|
||||||
|
}%
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
%% Extensions for Distributed Haskell (Goffin)
|
||||||
|
%% ===========================================
|
||||||
|
%%
|
||||||
|
%% These definitions may change in the future.
|
||||||
|
|
||||||
|
\hscommand{\hsunif}{\mathbin{:=:}}
|
||||||
|
\hscommand{\hsalias}{\mathrel{\sim}}
|
||||||
|
\hscommand{\hsoutof}{\twoheadleftarrow}
|
||||||
|
\hscommand{\hsinto}{\twoheadrightarrow}
|
||||||
|
\hscommand{\hsparc}{\binampersand}
|
||||||
|
\hscommand{\hsseq}{\Longrightarrow}
|
||||||
|
\hscommand{\hsex}[2]{{\hskwd{ex} #1 \hskwd{in} #2}}
|
||||||
|
|
||||||
|
\hscommand{\hsexin}[3][t]{%
|
||||||
|
\hsalign[#1]{%
|
||||||
|
\hskwd{ex} #2 \hskwd{in}\\
|
||||||
|
\quad\hsalign{#3}\\
|
||||||
|
}%
|
||||||
|
}
|
||||||
|
|
||||||
|
\hscommand{\hschoice}[2][t]{%
|
||||||
|
\hsalign[#1]{%
|
||||||
|
\hskwd{choice}\\
|
||||||
|
\quad\hsalign{#2}\\
|
||||||
|
}%
|
||||||
|
}
|
||||||
|
|
||||||
|
|
36
docs/hs-plugins.1
Normal file
36
docs/hs-plugins.1
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
.TH HS-PLUGINS 1 2005-03-26 "hs-plugins version 0.9.8" "User Manual"
|
||||||
|
|
||||||
|
.SH NAME
|
||||||
|
hs-plugins \- dynamic linker library for Haskell
|
||||||
|
|
||||||
|
.SH DESCRIPTION
|
||||||
|
.ds c \fIhs-plugins\fP
|
||||||
|
\*c is a library for loading plugins written in Haskell into an
|
||||||
|
application at runtime. It also provides a mechanism for (re)compiling
|
||||||
|
Haskell source at runtime. Thirdly, a combination of runtime compilation
|
||||||
|
and dynamic loading provides a suite of eval functions. Values exported
|
||||||
|
by plugins are transparently available to Haskell host applications, and
|
||||||
|
bindings exist to use Haskell plugins from at least C and Objective C
|
||||||
|
programs. hs-plugins requires ghc-6.2.2 or greater.
|
||||||
|
|
||||||
|
.SH DOCUMENTATION
|
||||||
|
The hs-plugins user manual is distributed in html format, and may be
|
||||||
|
found at <http://www.cse.unsw.edu.au/~dons/hs-plugins/>
|
||||||
|
|
||||||
|
.SH BUGS
|
||||||
|
Bug reports, and any other feedback, should be sent to
|
||||||
|
Don Stewart <dons@cse.unsw.edu.au>
|
||||||
|
.SH COPYRIGHT
|
||||||
|
Copyright \(co 2004-2005 Don Stewart
|
||||||
|
.PP
|
||||||
|
The hs-plugins library modules are distributed under the terms of the
|
||||||
|
LGPL.
|
||||||
|
.SH "SEE ALSO"
|
||||||
|
.BR dlopen (3)
|
||||||
|
|
||||||
|
.SH AUTHOR
|
||||||
|
|
||||||
|
This manual page was written by Don Stewart, based on the man page for
|
||||||
|
cpphs (written by Ian Lynagh).
|
||||||
|
|
||||||
|
|
1
docs/hs-plugins.hdir
Normal file
1
docs/hs-plugins.hdir
Normal file
@ -0,0 +1 @@
|
|||||||
|
hs-plugins
|
1808
docs/hs-plugins.tex
Normal file
1808
docs/hs-plugins.tex
Normal file
File diff suppressed because it is too large
Load Diff
17
docs/munge.sed
Normal file
17
docs/munge.sed
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
#!/usr/bin/sed -f
|
||||||
|
|
||||||
|
# de-boldify and <p>-ify the Contents.
|
||||||
|
|
||||||
|
/Contents/ {
|
||||||
|
:loop
|
||||||
|
/Go to/ {
|
||||||
|
b end
|
||||||
|
}
|
||||||
|
s,<p>,,
|
||||||
|
s,<b>,,
|
||||||
|
s,</b>,,
|
||||||
|
s,</p>,,
|
||||||
|
n
|
||||||
|
b loop
|
||||||
|
}
|
||||||
|
:end
|
9
docs/tex2page.sty
Normal file
9
docs/tex2page.sty
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
% tex2page.sty
|
||||||
|
% Dorai Sitaram
|
||||||
|
|
||||||
|
% Loading this file in a LaTeX document
|
||||||
|
% gives it all the macros of tex2page.tex,
|
||||||
|
% but via a more LaTeX-convenient filename.
|
||||||
|
|
||||||
|
\input{tex2page}
|
||||||
|
|
1381
docs/tex2page.tex
Normal file
1381
docs/tex2page.tex
Normal file
File diff suppressed because it is too large
Load Diff
17
examples/README
Normal file
17
examples/README
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
These examples illustrate the various uses of hs-plugins.
|
||||||
|
|
||||||
|
conf a configuration file edsl using plugins
|
||||||
|
dynload dynamically typed load
|
||||||
|
eval runtime evaluation of haskell strings, from Haskell and C
|
||||||
|
hmake the 'plugs' haskell interpreter
|
||||||
|
iface test the interface file parser
|
||||||
|
load load a plugin
|
||||||
|
make build a Haskell file
|
||||||
|
makewith merge and build a Haskell file
|
||||||
|
multi load multiple plugins at once
|
||||||
|
objc load Haskell plugins into object C programs
|
||||||
|
pkgconf test package.conf parsing
|
||||||
|
popen test popen
|
||||||
|
reload reload a plugin when it changes
|
||||||
|
shell a simple string filter
|
||||||
|
unload test unloading of plugins
|
45
examples/TIMINGS
Normal file
45
examples/TIMINGS
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
Method:
|
||||||
|
* "pdynload"
|
||||||
|
comes from pdynload/small
|
||||||
|
* "load + ghc"
|
||||||
|
comes from pdynload/null, with lines 13-14
|
||||||
|
uncommented from prog/Main.hs
|
||||||
|
* "dynload"
|
||||||
|
from dynload/simple
|
||||||
|
* "load, no check"
|
||||||
|
from pdynload/null, with lines 13-14 of prog/Main.hs
|
||||||
|
commented out
|
||||||
|
|
||||||
|
For example, to run the "pdynload" test:
|
||||||
|
$ cd pdynload/small
|
||||||
|
$ make
|
||||||
|
$ make check # to prime caches, etc.
|
||||||
|
$ time make check
|
||||||
|
$ time make check
|
||||||
|
$ time make check # run 'time make check' until value converges
|
||||||
|
|
||||||
|
The converged value is entered into the "Raw" timings, and then the
|
||||||
|
scaled timing is calculated for each machine. These scaled values were
|
||||||
|
then averaged over the number of machines, yielding the final
|
||||||
|
"Average" scores -- the average over a number of machines and os.
|
||||||
|
|
||||||
|
Raw timing:
|
||||||
|
pdynload load+ghc dynload load, no check
|
||||||
|
|
||||||
|
0.33 0.25 0.22 0.21 -- P4 2.6 , OpenBSD
|
||||||
|
0.38 0.31 0.29 0.27 -- P4 2.66, Linux
|
||||||
|
0.84 0.77 0.64 0.55 -- Quad P4 2.4, Linux
|
||||||
|
0.76 0.60 0.52 0.50 -- AMD 1.1G, Linux
|
||||||
|
0.95 0.83 0.75 0.72 -- G5 2.0G, Mac OS X
|
||||||
|
-- Quad Itanium 1,Linux
|
||||||
|
|
||||||
|
Scaled:
|
||||||
|
1.57 1.19 1.05 1
|
||||||
|
1.40 1.15 1.07
|
||||||
|
1.52 1.4 1.16
|
||||||
|
1.52 1.2 1.04
|
||||||
|
1.32 1.15 1.04
|
||||||
|
|
||||||
|
Average:
|
||||||
|
=1.46 = 1.218 = 1.07
|
||||||
|
|
41
examples/build.mk
Normal file
41
examples/build.mk
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
# how to build the default projects
|
||||||
|
|
||||||
|
include $(TOP)/config.mk
|
||||||
|
include $(TOP)/examples/check.mk
|
||||||
|
|
||||||
|
BIN= prog/a.out
|
||||||
|
OBJ= prog/Main.o
|
||||||
|
SRC= prog/Main.hs
|
||||||
|
|
||||||
|
BINDIR= prog
|
||||||
|
REALBIN= ./a.out
|
||||||
|
|
||||||
|
API_OBJ= api/API.o
|
||||||
|
|
||||||
|
INCLUDES= -i$(TOP)/examples/$(TEST)/api
|
||||||
|
PKGFLAGS= -package-conf $(TOP)/plugins.conf.inplace -package plugins
|
||||||
|
GHCFLAGS= -Onot -cpp -fglasgow-exts
|
||||||
|
|
||||||
|
.SUFFIXES : .o .hs .hi .lhs .hc .s
|
||||||
|
|
||||||
|
all: $(BIN)
|
||||||
|
|
||||||
|
$(BIN) : $(PRIOR_OBJS) $(API_OBJ) $(SRC) $(EXTRA_OBJS)
|
||||||
|
@rm -f $@
|
||||||
|
@$(GHC) --make -o $@ $(INCLUDES) $(PKGFLAGS) $(GHCFLAGS) $(EXTRAFLAGS) $(API) $(SRC)
|
||||||
|
|
||||||
|
# Standard suffix rules
|
||||||
|
.o.hi:
|
||||||
|
@:
|
||||||
|
.hs.o:
|
||||||
|
@$(GHC) $(INCLUDES) $(PKGFLAGS) $(GHCFLAGS) $(EXTRAFLAGS) -c $<
|
||||||
|
|
||||||
|
clean:
|
||||||
|
find . -name '*~' -exec rm {} \;
|
||||||
|
rm -rf *.{o,hi,dep}
|
||||||
|
rm -rf */*.{hi,o,old} */a.out
|
||||||
|
rm -rf */*core
|
||||||
|
rm -rf */*.a
|
||||||
|
rm -rf */package.conf
|
||||||
|
rm -rf *.a
|
||||||
|
|
24
examples/check.mk
Normal file
24
examples/check.mk
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
include $(TOP)/config.mk
|
||||||
|
|
||||||
|
check: $(BIN)
|
||||||
|
@(cd $(BINDIR) ;\
|
||||||
|
expected="expected" ;\
|
||||||
|
if [ -f "expected" -o -f "expected.$(GLASGOW_HASKELL)" ] ;\
|
||||||
|
then \
|
||||||
|
actual_out="/tmp/hs-plugins-actual.out.$$$$" ;\
|
||||||
|
diff_out="/tmp/hs-plugins.diff.$$$$" ;\
|
||||||
|
$(REALBIN) > $$actual_out 2>&1 || true ;\
|
||||||
|
if [ -f "expected.$(GLASGOW_HASKELL)" ] ; then \
|
||||||
|
expected="expected.$(GLASGOW_HASKELL)" ;\
|
||||||
|
fi ;\
|
||||||
|
diff -u $$expected $$actual_out > $$diff_out || true ;\
|
||||||
|
if [ -s "$$diff_out" ] ; then \
|
||||||
|
echo "failed with:" ;\
|
||||||
|
cat "$$diff_out" | sed '1,3d' ;\
|
||||||
|
else \
|
||||||
|
echo "ok." ;\
|
||||||
|
fi ;\
|
||||||
|
rm $$actual_out $$diff_out ;\
|
||||||
|
else \
|
||||||
|
$(REALBIN) 2>&1 || true ;\
|
||||||
|
fi)
|
11
examples/conf/simple/Mailrc.conf
Normal file
11
examples/conf/simple/Mailrc.conf
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
import System.Directory
|
||||||
|
|
||||||
|
resource = mail {
|
||||||
|
-- editor = do b <- doesFileExist "/usr/bin/emacs"
|
||||||
|
-- return $ if b then "emacs" else "vi" ,
|
||||||
|
editor = do b <- doesFileExist "/bin/sh"
|
||||||
|
return "sh",
|
||||||
|
|
||||||
|
attribution = \name -> "Today, "++name++" wrote :"
|
||||||
|
}
|
||||||
|
|
28
examples/conf/simple/Mailrc.stub
Normal file
28
examples/conf/simple/Mailrc.stub
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
module Mailrc ( resource ) where
|
||||||
|
|
||||||
|
import API
|
||||||
|
|
||||||
|
resource :: Interface
|
||||||
|
resource = mail
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
4
examples/conf/simple/Makefile
Normal file
4
examples/conf/simple/Makefile
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
TEST= conf/simple
|
||||||
|
|
||||||
|
TOP=../../..
|
||||||
|
include ../../build.mk
|
27
examples/conf/simple/api/API.hs
Normal file
27
examples/conf/simple/api/API.hs
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
--
|
||||||
|
-- the configuration file interface.
|
||||||
|
--
|
||||||
|
|
||||||
|
module API where
|
||||||
|
|
||||||
|
data Color = Black | Grey | Green | Cyan | Yellow | Magenta | Red
|
||||||
|
|
||||||
|
data Interface = Interface {
|
||||||
|
editor :: IO String,
|
||||||
|
attribution :: String -> String,
|
||||||
|
header_color :: Color,
|
||||||
|
colorize :: [String],
|
||||||
|
include :: Bool
|
||||||
|
}
|
||||||
|
|
||||||
|
-- Default settings
|
||||||
|
mail :: Interface
|
||||||
|
mail = Interface {
|
||||||
|
editor = return "vi",
|
||||||
|
|
||||||
|
attribution = (\user -> user ++ " wrote:"),
|
||||||
|
header_color = Grey,
|
||||||
|
colorize = [],
|
||||||
|
include = True
|
||||||
|
}
|
||||||
|
|
22
examples/conf/simple/prog/Main.hs
Normal file
22
examples/conf/simple/prog/Main.hs
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
|
||||||
|
import Plugins
|
||||||
|
import API
|
||||||
|
|
||||||
|
conf = "../Mailrc.conf"
|
||||||
|
stub = "../Mailrc.stub"
|
||||||
|
apipath = "../api"
|
||||||
|
|
||||||
|
main = do
|
||||||
|
status <- makeWith conf stub ["-i"++apipath]
|
||||||
|
o <- case status of
|
||||||
|
MakeFailure e -> mapM_ putStrLn e >> error "failed"
|
||||||
|
MakeSuccess _ o -> return o
|
||||||
|
status <- load o [apipath] [] "resource"
|
||||||
|
v <- case status of
|
||||||
|
LoadFailure err -> mapM_ putStrLn err >> error "no"
|
||||||
|
LoadSuccess _ v -> return v
|
||||||
|
|
||||||
|
user_editor <- editor v
|
||||||
|
putStrLn user_editor
|
||||||
|
makeCleaner o
|
||||||
|
|
1
examples/conf/simple/prog/expected
Normal file
1
examples/conf/simple/prog/expected
Normal file
@ -0,0 +1 @@
|
|||||||
|
sh
|
6
examples/dynload/io/Makefile
Normal file
6
examples/dynload/io/Makefile
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
TEST=dynload/io
|
||||||
|
|
||||||
|
EXTRA_OBJS=TestIO.o
|
||||||
|
|
||||||
|
TOP=../../..
|
||||||
|
include ../../build.mk
|
86
examples/dynload/io/TestIO.hs
Normal file
86
examples/dynload/io/TestIO.hs
Normal file
@ -0,0 +1,86 @@
|
|||||||
|
{-# OPTIONS -fglasgow-exts -cpp #-}
|
||||||
|
--
|
||||||
|
-- Copyright (c) 2004 Don Stewart - http://www.cse.unsw.edu.au/~dons
|
||||||
|
-- LGPL version 2.1 or later (see http://www.gnu.org/copyleft/lesser.html)
|
||||||
|
--
|
||||||
|
|
||||||
|
module TestIO ( resource_dyn ) where
|
||||||
|
|
||||||
|
import API
|
||||||
|
import AltData
|
||||||
|
|
||||||
|
import System.IO
|
||||||
|
import System.Posix.Types ( ProcessID, Fd )
|
||||||
|
import System.Posix.Process ( forkProcess, executeFile, getProcessID )
|
||||||
|
import System.Posix.IO ( createPipe, stdInput,
|
||||||
|
stdOutput, fdToHandle, closeFd, dupTo )
|
||||||
|
|
||||||
|
resource_dyn :: Dynamic
|
||||||
|
resource_dyn = toDyn resource
|
||||||
|
|
||||||
|
resource :: TestIO
|
||||||
|
resource = testio { field = date }
|
||||||
|
|
||||||
|
|
||||||
|
--
|
||||||
|
-- call a shell command , returning it's output
|
||||||
|
--
|
||||||
|
date :: IO String
|
||||||
|
date = do (hdl,_,_) <- catch (popen "/bin/date") (\_->error "popen failed")
|
||||||
|
hGetLine hdl
|
||||||
|
|
||||||
|
------------------------------------------------------------------------
|
||||||
|
--
|
||||||
|
-- my implementation of $val = `cmd`; (if this was perl)
|
||||||
|
--
|
||||||
|
-- provide similar functionality to popen(3),
|
||||||
|
-- along with bidirectional ipc via pipes
|
||||||
|
-- return's the pid of the child process
|
||||||
|
--
|
||||||
|
-- there are two different forkProcess functions. the pre-620 was a
|
||||||
|
-- unix-fork style function, and the modern function has semantics more
|
||||||
|
-- like the Awkward-Squad paper. We provide implementations of popen
|
||||||
|
-- using both versions, depending on which GHC the user wants to try.
|
||||||
|
--
|
||||||
|
|
||||||
|
popen :: FilePath -> IO (Handle, Handle, ProcessID)
|
||||||
|
popen cmd = do
|
||||||
|
(pr, pw) <- createPipe
|
||||||
|
(cr, cw) <- createPipe
|
||||||
|
|
||||||
|
-- parent --
|
||||||
|
let parent = do closeFd cw
|
||||||
|
closeFd pr
|
||||||
|
-- child --
|
||||||
|
let child = do closeFd pw
|
||||||
|
closeFd cr
|
||||||
|
exec cmd (pr,cw)
|
||||||
|
error "exec cmd failed!" -- typing only
|
||||||
|
|
||||||
|
-- if the parser front end understood cpp, this would work
|
||||||
|
-- #if __GLASGOW_HASKELL__ >= 601
|
||||||
|
pid <- forkProcess child -- fork child
|
||||||
|
parent -- and run parent code
|
||||||
|
-- #else
|
||||||
|
-- p <- forkProcess
|
||||||
|
-- pid <- case p of
|
||||||
|
-- Just pid -> parent >> return pid
|
||||||
|
-- Nothing -> child
|
||||||
|
-- #endif
|
||||||
|
|
||||||
|
hcr <- fdToHandle cr
|
||||||
|
hpw <- fdToHandle pw
|
||||||
|
|
||||||
|
return (hcr,hpw,pid)
|
||||||
|
|
||||||
|
--
|
||||||
|
-- execve cmd in the child process, dup'ing the file descriptors passed
|
||||||
|
-- as arguments to become the child's stdin and stdout.
|
||||||
|
--
|
||||||
|
exec :: FilePath -> (Fd,Fd) -> IO ()
|
||||||
|
exec cmd (pr,cw) = do
|
||||||
|
dupTo pr stdInput
|
||||||
|
dupTo cw stdOutput
|
||||||
|
executeFile cmd False [] Nothing
|
||||||
|
|
||||||
|
------------------------------------------------------------------------
|
19
examples/dynload/io/api/API.hs
Normal file
19
examples/dynload/io/api/API.hs
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
{-# OPTIONS -fglasgow-exts #-}
|
||||||
|
|
||||||
|
module API where
|
||||||
|
|
||||||
|
import AltData
|
||||||
|
|
||||||
|
data TestIO = TestIO {
|
||||||
|
field :: IO String
|
||||||
|
}
|
||||||
|
|
||||||
|
instance Typeable TestIO where
|
||||||
|
#if __GLASGOW_HASKELL__ >= 603
|
||||||
|
typeOf i = mkTyConApp (mkTyCon "API.TestIO") []
|
||||||
|
#else
|
||||||
|
typeOf i = mkAppTy (mkTyCon "API.TestIO") []
|
||||||
|
#endif
|
||||||
|
|
||||||
|
testio :: TestIO
|
||||||
|
testio = TestIO { field = return "default value" }
|
12
examples/dynload/io/prog/Main.hs
Normal file
12
examples/dynload/io/prog/Main.hs
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
|
||||||
|
import Plugins
|
||||||
|
import API
|
||||||
|
|
||||||
|
main = do
|
||||||
|
m_v <- dynload "../TestIO.o" ["../api"]
|
||||||
|
["../../../../plugins.conf.inplace"] "resource_dyn"
|
||||||
|
case m_v of
|
||||||
|
LoadFailure _ -> error "couldn't compile"
|
||||||
|
LoadSuccess _ v -> do
|
||||||
|
s <- field v
|
||||||
|
if s /= [] then print True else print False
|
1
examples/dynload/io/prog/expected
Normal file
1
examples/dynload/io/prog/expected
Normal file
@ -0,0 +1 @@
|
|||||||
|
True
|
4
examples/dynload/poly/Makefile
Normal file
4
examples/dynload/poly/Makefile
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
TEST=dynload/poly
|
||||||
|
EXTRA_OBJS=Plugin.o
|
||||||
|
TOP=../../..
|
||||||
|
include ../../build.mk
|
12
examples/dynload/poly/Plugin.hs
Normal file
12
examples/dynload/poly/Plugin.hs
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
module Plugin where
|
||||||
|
|
||||||
|
import API
|
||||||
|
import AltData
|
||||||
|
|
||||||
|
my_fun = plugin {
|
||||||
|
equals = \x y -> (x /= y) -- a strange equals function :)
|
||||||
|
}
|
||||||
|
|
||||||
|
resource_dyn :: Dynamic
|
||||||
|
resource_dyn = toDyn my_fun
|
||||||
|
|
24
examples/dynload/poly/api/API.hs
Normal file
24
examples/dynload/poly/api/API.hs
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
{-# OPTIONS -cpp #-}
|
||||||
|
|
||||||
|
module API where
|
||||||
|
|
||||||
|
import AltData
|
||||||
|
|
||||||
|
data Interface = Interface {
|
||||||
|
equals :: forall t. Eq t => t -> t -> Bool
|
||||||
|
}
|
||||||
|
|
||||||
|
--
|
||||||
|
-- see how it hides the internal type.. but to compile GHC still checks
|
||||||
|
-- the type.
|
||||||
|
--
|
||||||
|
instance Typeable Interface where
|
||||||
|
#if __GLASGOW_HASKELL__ >= 603
|
||||||
|
typeOf i = mkTyConApp (mkTyCon "API.Interface") []
|
||||||
|
#else
|
||||||
|
typeOf i = mkAppTy (mkTyCon "API.Interface") []
|
||||||
|
#endif
|
||||||
|
|
||||||
|
plugin :: Interface
|
||||||
|
plugin = Interface { equals = (==) }
|
||||||
|
|
17
examples/dynload/poly/prog/Main.hs
Normal file
17
examples/dynload/poly/prog/Main.hs
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
{-# OPTIONS -cpp #-}
|
||||||
|
|
||||||
|
#include "../../../../config.h"
|
||||||
|
|
||||||
|
import Plugins
|
||||||
|
import API
|
||||||
|
|
||||||
|
main = do
|
||||||
|
m_v <- dynload "../Plugin.o" ["../api"]
|
||||||
|
["../../../../plugins.conf.inplace"]
|
||||||
|
"resource_dyn"
|
||||||
|
case m_v of
|
||||||
|
LoadFailure _ -> error "didn't compile"
|
||||||
|
LoadSuccess _ (Interface eq) -> do
|
||||||
|
putStrLn $ show $ 1 `eq` 2
|
||||||
|
putStrLn $ show $ 'a' `eq` 'b'
|
||||||
|
|
2
examples/dynload/poly/prog/expected
Normal file
2
examples/dynload/poly/prog/expected
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
True
|
||||||
|
True
|
4
examples/dynload/should_fail/Makefile
Normal file
4
examples/dynload/should_fail/Makefile
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
TEST= dynload/should_fail
|
||||||
|
EXTRA_OBJS=Plugin.o
|
||||||
|
TOP=../../..
|
||||||
|
include ../../build.mk
|
12
examples/dynload/should_fail/Plugin.hs
Normal file
12
examples/dynload/should_fail/Plugin.hs
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
{-# OPTIONS -fglasgow-exts #-}
|
||||||
|
module Plugin where
|
||||||
|
|
||||||
|
import API
|
||||||
|
import AltData
|
||||||
|
|
||||||
|
v :: Int
|
||||||
|
v = 0xdeadbeef
|
||||||
|
|
||||||
|
resource_dyn :: Dynamic
|
||||||
|
resource_dyn = toDyn v
|
||||||
|
|
20
examples/dynload/should_fail/api/API.hs
Normal file
20
examples/dynload/should_fail/api/API.hs
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
{-# OPTIONS -fglasgow-exts #-}
|
||||||
|
|
||||||
|
module API where
|
||||||
|
|
||||||
|
import AltData
|
||||||
|
|
||||||
|
data Interface = Interface {
|
||||||
|
function :: String
|
||||||
|
}
|
||||||
|
|
||||||
|
instance Typeable Interface where
|
||||||
|
#if __GLASGOW_HASKELL__ >= 603
|
||||||
|
typeOf i = mkTyConApp (mkTyCon "API.Interface") []
|
||||||
|
#else
|
||||||
|
typeOf i = mkAppTy (mkTyCon "API.Interface") []
|
||||||
|
#endif
|
||||||
|
|
||||||
|
plugin :: Interface
|
||||||
|
plugin = Interface { function = "goodbye" }
|
||||||
|
|
14
examples/dynload/should_fail/prog/Main.hs
Normal file
14
examples/dynload/should_fail/prog/Main.hs
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
|
||||||
|
import Plugins
|
||||||
|
import API
|
||||||
|
|
||||||
|
main = do
|
||||||
|
m_v <- dynload "../Plugin.o"
|
||||||
|
["../api"]
|
||||||
|
["../../../../plugins.conf.inplace"]
|
||||||
|
"resource_dyn"
|
||||||
|
|
||||||
|
case m_v of
|
||||||
|
LoadFailure _ -> putStrLn "didn't compile"
|
||||||
|
LoadSuccess _ v -> putStrLn $ function v
|
||||||
|
|
4
examples/dynload/should_fail/prog/expected
Normal file
4
examples/dynload/should_fail/prog/expected
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
Couldn't match `API.Interface' against `Int'
|
||||||
|
Expected type: API.Interface
|
||||||
|
Inferred type: Int
|
||||||
|
didn't compile
|
4
examples/dynload/should_fail_1/Makefile
Normal file
4
examples/dynload/should_fail_1/Makefile
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
TEST= dynload/should_fail_1
|
||||||
|
EXTRA_OBJS=Plugin.o
|
||||||
|
TOP=../../..
|
||||||
|
include ../../build.mk
|
15
examples/dynload/should_fail_1/Plugin.hs
Normal file
15
examples/dynload/should_fail_1/Plugin.hs
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
--
|
||||||
|
-- trying to be really mean.
|
||||||
|
--
|
||||||
|
|
||||||
|
module Plugin where
|
||||||
|
|
||||||
|
import API
|
||||||
|
import AltData
|
||||||
|
|
||||||
|
v :: Int -> Int
|
||||||
|
v = \x -> 0xdeadbeef
|
||||||
|
|
||||||
|
resource_dyn :: Dynamic
|
||||||
|
resource_dyn = toDyn v
|
||||||
|
|
20
examples/dynload/should_fail_1/api/API.hs
Normal file
20
examples/dynload/should_fail_1/api/API.hs
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
{-# OPTIONS -fglasgow-exts #-}
|
||||||
|
|
||||||
|
module API where
|
||||||
|
|
||||||
|
import AltData
|
||||||
|
|
||||||
|
data Interface = Interface {
|
||||||
|
function :: String
|
||||||
|
}
|
||||||
|
|
||||||
|
instance Typeable Interface where
|
||||||
|
#if __GLASGOW_HASKELL__ >= 603
|
||||||
|
typeOf i = mkTyConApp (mkTyCon "API.Interface") []
|
||||||
|
#else
|
||||||
|
typeOf i = mkAppTy (mkTyCon "API.Interface") []
|
||||||
|
#endif
|
||||||
|
|
||||||
|
plugin :: Interface
|
||||||
|
plugin = Interface { function = "goodbye" }
|
||||||
|
|
11
examples/dynload/should_fail_1/prog/Main.hs
Normal file
11
examples/dynload/should_fail_1/prog/Main.hs
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
|
||||||
|
import Plugins
|
||||||
|
import API
|
||||||
|
|
||||||
|
main = do
|
||||||
|
m_v <- dynload "../Plugin.o" ["../api"]
|
||||||
|
["../../../../plugins.conf.inplace"] "resource_dyn"
|
||||||
|
case m_v of
|
||||||
|
LoadFailure _ -> putStrLn "didn't compile"
|
||||||
|
LoadSuccess _ v -> putStrLn $ (function v)
|
||||||
|
|
4
examples/dynload/should_fail_1/prog/expected
Normal file
4
examples/dynload/should_fail_1/prog/expected
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
Couldn't match `API.Interface' against `Int -> Int'
|
||||||
|
Expected type: API.Interface
|
||||||
|
Inferred type: Int -> Int
|
||||||
|
didn't compile
|
4
examples/dynload/should_fail_2/Makefile
Normal file
4
examples/dynload/should_fail_2/Makefile
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
TEST= dynload/should_fail_2
|
||||||
|
|
||||||
|
TOP=../../..
|
||||||
|
include ../../build.mk
|
19
examples/dynload/should_fail_2/Plugin.in
Normal file
19
examples/dynload/should_fail_2/Plugin.in
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
--
|
||||||
|
-- the plugin doesn't even make the resource_dyn a Dynamic.
|
||||||
|
--
|
||||||
|
-- let's hope that makeWith strips out the invalid declarations
|
||||||
|
--
|
||||||
|
|
||||||
|
{-# OPTIONS -fglasgow-exts #-}
|
||||||
|
|
||||||
|
module Plugin where
|
||||||
|
|
||||||
|
import API
|
||||||
|
import AltData
|
||||||
|
import GHC.Base
|
||||||
|
|
||||||
|
v :: Int
|
||||||
|
v = 0xdeadbeef
|
||||||
|
|
||||||
|
resource_dyn = (typeOf v, unsafeCoerce v)
|
||||||
|
|
12
examples/dynload/should_fail_2/Plugin.stub
Normal file
12
examples/dynload/should_fail_2/Plugin.stub
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
{-# OPTIONS -fglasgow-exts #-}
|
||||||
|
|
||||||
|
module Plugin ( resource_dyn ) where
|
||||||
|
|
||||||
|
import API
|
||||||
|
import AltData
|
||||||
|
|
||||||
|
resource = plugin
|
||||||
|
|
||||||
|
resource_dyn :: Dynamic
|
||||||
|
resource_dyn = toDyn resource
|
||||||
|
|
22
examples/dynload/should_fail_2/api/API.hs
Normal file
22
examples/dynload/should_fail_2/api/API.hs
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
{-# OPTIONS -fglasgow-exts #-}
|
||||||
|
|
||||||
|
module API where
|
||||||
|
|
||||||
|
import AltData
|
||||||
|
import GHC.Base
|
||||||
|
|
||||||
|
data Interface = Interface {
|
||||||
|
function :: String
|
||||||
|
}
|
||||||
|
|
||||||
|
instance Typeable Interface where
|
||||||
|
#if __GLASGOW_HASKELL__ >= 603
|
||||||
|
typeOf i = mkTyConApp (mkTyCon "API.Interface") []
|
||||||
|
#else
|
||||||
|
typeOf i = mkAppTy (mkTyCon "API.Interface") []
|
||||||
|
#endif
|
||||||
|
|
||||||
|
plugin :: Interface
|
||||||
|
plugin = Interface { function = "goodbye" }
|
||||||
|
|
||||||
|
unsafeCoerce = unsafeCoerce#
|
19
examples/dynload/should_fail_2/prog/Main.hs
Normal file
19
examples/dynload/should_fail_2/prog/Main.hs
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
|
||||||
|
import Plugins
|
||||||
|
import API
|
||||||
|
|
||||||
|
conf = "../Plugin.in"
|
||||||
|
stub = "../Plugin.stub"
|
||||||
|
|
||||||
|
main = do
|
||||||
|
status <- makeWith conf stub ["-i../api", "-i../../../../src/altdata/"]
|
||||||
|
case status of
|
||||||
|
MakeFailure e -> mapM_ putStrLn e >> putStrLn "failed"
|
||||||
|
MakeSuccess _ o -> do {
|
||||||
|
; m_v <- dynload o ["../api"] [] "resource_dyn"
|
||||||
|
; makeCleaner o
|
||||||
|
; case m_v of
|
||||||
|
LoadFailure _ -> putStrLn "didn't load"
|
||||||
|
LoadSuccess _ v -> putStrLn $ (function v)
|
||||||
|
}
|
||||||
|
|
8
examples/dynload/should_fail_2/prog/expected
Normal file
8
examples/dynload/should_fail_2/prog/expected
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
|
||||||
|
../Plugin.in:18:
|
||||||
|
Couldn't match `Dynamic' against `(t, t1)'
|
||||||
|
Expected type: Dynamic
|
||||||
|
Inferred type: (t, t1)
|
||||||
|
In the definition of `resource_dyn':
|
||||||
|
resource_dyn = (typeOf v, unsafeCoerce v)
|
||||||
|
failed
|
7
examples/dynload/should_fail_2/prog/expected.604
Normal file
7
examples/dynload/should_fail_2/prog/expected.604
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
|
||||||
|
../Plugin.in:18:15:
|
||||||
|
Couldn't match `Dynamic' against `(a, b)'
|
||||||
|
Expected type: Dynamic
|
||||||
|
Inferred type: (a, b)
|
||||||
|
In the definition of `resource_dyn': resource_dyn = (typeOf v, unsafeCoerce v)
|
||||||
|
failed
|
4
examples/dynload/should_fail_3/Makefile
Normal file
4
examples/dynload/should_fail_3/Makefile
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
TEST= dynload/should_fail_3
|
||||||
|
|
||||||
|
TOP=../../..
|
||||||
|
include ../../build.mk
|
19
examples/dynload/should_fail_3/Plugin.in
Normal file
19
examples/dynload/should_fail_3/Plugin.in
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
--
|
||||||
|
-- the plugin doesn't even make the resource_dyn a Dynamic.
|
||||||
|
-- let's hope that makeWith strips out the invalid declarations
|
||||||
|
--
|
||||||
|
|
||||||
|
{-# OPTIONS -fglasgow-exts #-}
|
||||||
|
|
||||||
|
module Plugin where
|
||||||
|
|
||||||
|
import API
|
||||||
|
|
||||||
|
import AltData
|
||||||
|
import GHC.Base
|
||||||
|
|
||||||
|
v :: Int
|
||||||
|
v = 0xdeadbeef
|
||||||
|
|
||||||
|
resource_dyn = (typeOf plugin, unsafeCoerce v)
|
||||||
|
|
12
examples/dynload/should_fail_3/Plugin.stub
Normal file
12
examples/dynload/should_fail_3/Plugin.stub
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
{-# OPTIONS -fglasgow-exts #-}
|
||||||
|
|
||||||
|
module Plugin ( resource_dyn ) where
|
||||||
|
|
||||||
|
import API
|
||||||
|
import AltData.Dynamic
|
||||||
|
|
||||||
|
resource = plugin
|
||||||
|
|
||||||
|
resource_dyn :: Dynamic
|
||||||
|
resource_dyn = toDyn resource
|
||||||
|
|
22
examples/dynload/should_fail_3/api/API.hs
Normal file
22
examples/dynload/should_fail_3/api/API.hs
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
{-# OPTIONS -cpp -fglasgow-exts #-}
|
||||||
|
|
||||||
|
module API where
|
||||||
|
|
||||||
|
import AltData
|
||||||
|
import GHC.Base
|
||||||
|
|
||||||
|
data Interface = Interface {
|
||||||
|
function :: String
|
||||||
|
}
|
||||||
|
|
||||||
|
instance Typeable Interface where
|
||||||
|
#if __GLASGOW_HASKELL__ >= 603
|
||||||
|
typeOf _ = mkTyConApp (mkTyCon "API.Interface") []
|
||||||
|
#else
|
||||||
|
typeOf _ = mkAppTy (mkTyCon "API.Interface") []
|
||||||
|
#endif
|
||||||
|
|
||||||
|
plugin :: Interface
|
||||||
|
plugin = Interface { function = "goodbye" }
|
||||||
|
|
||||||
|
unsafeCoerce = unsafeCoerce#
|
18
examples/dynload/should_fail_3/prog/Main.hs
Normal file
18
examples/dynload/should_fail_3/prog/Main.hs
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
|
||||||
|
import Plugins
|
||||||
|
import API
|
||||||
|
|
||||||
|
conf = "../Plugin.in"
|
||||||
|
stub = "../Plugin.stub"
|
||||||
|
|
||||||
|
main = do
|
||||||
|
status <- makeWith conf stub ["-i../api", "-i../../../../src/altdata"]
|
||||||
|
o <- case status of
|
||||||
|
MakeFailure e -> mapM_ putStrLn e >> error "failed"
|
||||||
|
MakeSuccess _ o -> return o
|
||||||
|
m_v <- dynload o ["../api"] [] "resource_dyn"
|
||||||
|
case m_v of
|
||||||
|
LoadFailure _ -> error "didn't compile"
|
||||||
|
LoadSuccess _ v -> do putStrLn $ (function v)
|
||||||
|
makeCleaner o
|
||||||
|
|
9
examples/dynload/should_fail_3/prog/expected
Normal file
9
examples/dynload/should_fail_3/prog/expected
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
|
||||||
|
../Plugin.in:18:
|
||||||
|
Couldn't match `Dynamic' against `(t, t1)'
|
||||||
|
Expected type: Dynamic
|
||||||
|
Inferred type: (t, t1)
|
||||||
|
In the definition of `resource_dyn':
|
||||||
|
resource_dyn = (typeOf plugin, unsafeCoerce v)
|
||||||
|
|
||||||
|
Fail: failed
|
8
examples/dynload/should_fail_3/prog/expected.604
Normal file
8
examples/dynload/should_fail_3/prog/expected.604
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
|
||||||
|
../Plugin.in:18:15:
|
||||||
|
Couldn't match `Dynamic' against `(a, b)'
|
||||||
|
Expected type: Dynamic
|
||||||
|
Inferred type: (a, b)
|
||||||
|
In the definition of `resource_dyn':
|
||||||
|
resource_dyn = (typeOf plugin, unsafeCoerce v)
|
||||||
|
a.out: failed
|
4
examples/dynload/simple/Makefile
Normal file
4
examples/dynload/simple/Makefile
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
TEST=dynload/simple
|
||||||
|
EXTRA_OBJS=Plugin.o
|
||||||
|
TOP=../../..
|
||||||
|
include ../../build.mk
|
11
examples/dynload/simple/Plugin.hs
Normal file
11
examples/dynload/simple/Plugin.hs
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
{-# OPTIONS -fglasgow-exts #-}
|
||||||
|
module Plugin where
|
||||||
|
|
||||||
|
import API
|
||||||
|
import AltData
|
||||||
|
|
||||||
|
my_fun = plugin { function = "plugin says \"hello\"" }
|
||||||
|
|
||||||
|
resource_dyn :: Dynamic
|
||||||
|
resource_dyn = toDyn my_fun
|
||||||
|
|
20
examples/dynload/simple/api/API.hs
Normal file
20
examples/dynload/simple/api/API.hs
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
{-# OPTIONS -cpp #-}
|
||||||
|
|
||||||
|
module API where
|
||||||
|
|
||||||
|
import AltData
|
||||||
|
|
||||||
|
data Interface = Interface {
|
||||||
|
function :: String
|
||||||
|
}
|
||||||
|
|
||||||
|
instance Typeable Interface where
|
||||||
|
#if __GLASGOW_HASKELL__ >= 603
|
||||||
|
typeOf i = mkTyConApp (mkTyCon "API.Interface") []
|
||||||
|
#else
|
||||||
|
typeOf i = mkAppTy (mkTyCon "API.Interface") []
|
||||||
|
#endif
|
||||||
|
|
||||||
|
plugin :: Interface
|
||||||
|
plugin = Interface { function = "goodbye" }
|
||||||
|
|
15
examples/dynload/simple/prog/Main.hs
Normal file
15
examples/dynload/simple/prog/Main.hs
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
{-# OPTIONS -cpp #-}
|
||||||
|
|
||||||
|
#include "../../../../config.h"
|
||||||
|
|
||||||
|
import Plugins
|
||||||
|
import API
|
||||||
|
|
||||||
|
main = do
|
||||||
|
m_v <- dynload "../Plugin.o" ["../api"]
|
||||||
|
["../../../../plugins.conf.inplace"]
|
||||||
|
"resource_dyn"
|
||||||
|
case m_v of
|
||||||
|
LoadFailure _ -> error "didn't compile"
|
||||||
|
LoadSuccess _ v -> putStrLn $ (function v)
|
||||||
|
|
1
examples/dynload/simple/prog/expected
Normal file
1
examples/dynload/simple/prog/expected
Normal file
@ -0,0 +1 @@
|
|||||||
|
plugin says "hello"
|
27
examples/eval.mk
Normal file
27
examples/eval.mk
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
include $(TOP)/config.mk
|
||||||
|
include $(TOP)/examples/check.mk
|
||||||
|
|
||||||
|
PKGFLAGS= -package-conf $(TOP)/plugins.conf.inplace -package eval -package plugins -package printf
|
||||||
|
|
||||||
|
BIN=a.out
|
||||||
|
SRC=Main.hs
|
||||||
|
|
||||||
|
BINDIR= "."
|
||||||
|
REALBIN= ./$(BIN)
|
||||||
|
|
||||||
|
.SUFFIXES : .o .hs .hi .lhs .hc .s
|
||||||
|
|
||||||
|
all: $(BIN)
|
||||||
|
|
||||||
|
$(BIN): $(SRC) $(OBJS)
|
||||||
|
@rm -f $@
|
||||||
|
@$(GHC) --make -fglasgow-exts $(GHCFLAGS) $(PKGFLAGS) $(EXTRAFLAGS) $(SRC)
|
||||||
|
|
||||||
|
# Standard suffix rules
|
||||||
|
.o.hi:
|
||||||
|
@:
|
||||||
|
.hs.o:
|
||||||
|
@$(GHC) $(INCLUDES) $(PKGFLAGS) $(GHCFLAGS) $(EXTRAFLAGS) -c $<
|
||||||
|
|
||||||
|
clean:
|
||||||
|
rm -rf *.hi *.o *~ $(BIN)
|
5
examples/eval/eval1/Main.hs
Normal file
5
examples/eval/eval1/Main.hs
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
|
||||||
|
import Eval.Haskell
|
||||||
|
|
||||||
|
main = do i <- eval "1 + 6 :: Int" [] :: IO (Maybe Int)
|
||||||
|
if isJust i then putStrLn $ show (fromJust i) else return ()
|
2
examples/eval/eval1/Makefile
Normal file
2
examples/eval/eval1/Makefile
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
TOP=../../..
|
||||||
|
include ../../eval.mk
|
1
examples/eval/eval1/expected
Normal file
1
examples/eval/eval1/expected
Normal file
@ -0,0 +1 @@
|
|||||||
|
7
|
6
examples/eval/eval2/Main.hs
Normal file
6
examples/eval/eval2/Main.hs
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
import Eval.Haskell
|
||||||
|
|
||||||
|
main = do m_s <- eval "map toUpper \"haskell\"" ["Data.Char"]
|
||||||
|
case m_s of
|
||||||
|
Nothing -> putStrLn "typechecking failed"
|
||||||
|
Just s -> putStrLn s
|
2
examples/eval/eval2/Makefile
Normal file
2
examples/eval/eval2/Makefile
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
TOP=../../..
|
||||||
|
include ../../eval.mk
|
1
examples/eval/eval2/expected
Normal file
1
examples/eval/eval2/expected
Normal file
@ -0,0 +1 @@
|
|||||||
|
HASKELL
|
42
examples/eval/eval3/Main.hs
Normal file
42
examples/eval/eval3/Main.hs
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
{-# OPTIONS -cpp #-}
|
||||||
|
--
|
||||||
|
-- Should evaluate to '3', unless something goes wrong.
|
||||||
|
--
|
||||||
|
-- Not so bad to use AltData, as it is already derived for all the basic
|
||||||
|
-- types. Then, just replace deriving Typeable, with hand-derived
|
||||||
|
-- instance of Typeable (see hs-plugins/examples/eval/eval_fn1/Poly.hs
|
||||||
|
--
|
||||||
|
--
|
||||||
|
|
||||||
|
#include "../../../config.h"
|
||||||
|
|
||||||
|
import Eval.Haskell
|
||||||
|
import AltData.Dynamic
|
||||||
|
|
||||||
|
-- import Data.Dynamic
|
||||||
|
|
||||||
|
pkgconf = TOP ++ "/plugins.conf.inplace"
|
||||||
|
|
||||||
|
main = do
|
||||||
|
a <- return $ toDyn (3::Int)
|
||||||
|
|
||||||
|
m_b <- unsafeEval_ "\\dyn -> fromMaybe (7 :: Int) (fromDyn dyn)"
|
||||||
|
["AltData.Dynamic","Data.Maybe"] -- imports
|
||||||
|
|
||||||
|
[ "-package-conf "++pkgconf , "-package altdata" ]
|
||||||
|
|
||||||
|
[ pkgconf ]
|
||||||
|
[]
|
||||||
|
|
||||||
|
|
||||||
|
{-
|
||||||
|
-- should work, but doesn't. type check fails
|
||||||
|
-- (due to static vs dynamic typing issue)
|
||||||
|
|
||||||
|
m_b <- unsafeEval_ "\\dyn -> fromMaybe (7 :: Int) (fromDynamic dyn)"
|
||||||
|
["Data.Dynamic","Data.Maybe"] [] []
|
||||||
|
-}
|
||||||
|
|
||||||
|
case m_b of
|
||||||
|
Left s -> mapM_ putStrLn s
|
||||||
|
Right b -> putStrLn $ show (b a :: Int)
|
2
examples/eval/eval3/Makefile
Normal file
2
examples/eval/eval3/Makefile
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
TOP=../../..
|
||||||
|
include ../../eval.mk
|
1
examples/eval/eval3/expected
Normal file
1
examples/eval/eval3/expected
Normal file
@ -0,0 +1 @@
|
|||||||
|
3
|
9
examples/eval/eval_/Main.hs
Normal file
9
examples/eval/eval_/Main.hs
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
|
||||||
|
import Eval.Haskell
|
||||||
|
|
||||||
|
main = do i <- eval_ "Just (7 :: Int)"
|
||||||
|
["Maybe"]
|
||||||
|
["-fglasgow-exts"]
|
||||||
|
[]
|
||||||
|
[] :: IO (Either [String] (Maybe (Maybe Int)))
|
||||||
|
print i
|
2
examples/eval/eval_/Makefile
Normal file
2
examples/eval/eval_/Makefile
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
TOP=../../..
|
||||||
|
include ../../eval.mk
|
1
examples/eval/eval_/expected
Normal file
1
examples/eval/eval_/expected
Normal file
@ -0,0 +1 @@
|
|||||||
|
Right (Just (Just 7))
|
10
examples/eval/eval_fn/Main.hs
Normal file
10
examples/eval/eval_fn/Main.hs
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
--
|
||||||
|
-- lambda abstraction!
|
||||||
|
--
|
||||||
|
--
|
||||||
|
-- needs unsafeEval because eval has a broken Dynamic check
|
||||||
|
--
|
||||||
|
import Eval.Haskell
|
||||||
|
|
||||||
|
main = do fn <- unsafeEval "(\\(x::Int) -> (x,x))" [] :: IO (Maybe (Int -> (Int,Int)))
|
||||||
|
when (isJust fn) $ putStrLn $ show $ (fromJust fn) 7
|
2
examples/eval/eval_fn/Makefile
Normal file
2
examples/eval/eval_fn/Makefile
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
TOP=../../..
|
||||||
|
include ../../eval.mk
|
1
examples/eval/eval_fn/expected
Normal file
1
examples/eval/eval_fn/expected
Normal file
@ -0,0 +1 @@
|
|||||||
|
(7,7)
|
15
examples/eval/eval_fn1/Main.hs
Normal file
15
examples/eval/eval_fn1/Main.hs
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
{-# OPTIONS -fglasgow-exts #-}
|
||||||
|
--
|
||||||
|
-- polymorphic eval!
|
||||||
|
--
|
||||||
|
|
||||||
|
module Main where
|
||||||
|
|
||||||
|
import Poly
|
||||||
|
import Eval.Haskell
|
||||||
|
|
||||||
|
main = do m_f <- eval "Fn (\\x y -> x == y)" ["Poly"]
|
||||||
|
when (isJust m_f) $ do
|
||||||
|
let (Fn f) = fromJust m_f
|
||||||
|
putStrLn $ show (f True True)
|
||||||
|
putStrLn $ show (f 1 2)
|
2
examples/eval/eval_fn1/Makefile
Normal file
2
examples/eval/eval_fn1/Makefile
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
TOP=../../..
|
||||||
|
include ../../eval.mk
|
16
examples/eval/eval_fn1/Poly.hs
Normal file
16
examples/eval/eval_fn1/Poly.hs
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
{-# OPTIONS -cpp -fglasgow-exts #-}
|
||||||
|
module Poly where
|
||||||
|
|
||||||
|
import AltData.Typeable
|
||||||
|
|
||||||
|
data Fn = Fn {fn :: forall t. Eq t => t -> t -> Bool}
|
||||||
|
|
||||||
|
--
|
||||||
|
-- ignore type inside the Fn... is this correct?
|
||||||
|
--
|
||||||
|
instance Typeable Fn where
|
||||||
|
#if __GLASGOW_HASKELL__ >= 603
|
||||||
|
typeOf _ = mkTyConApp (mkTyCon "Poly.Fn") []
|
||||||
|
#else
|
||||||
|
typeOf _ = mkAppTy (mkTyCon "Poly.Fn") []
|
||||||
|
#endif
|
2
examples/eval/eval_fn1/expected
Normal file
2
examples/eval/eval_fn1/expected
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
True
|
||||||
|
False
|
2
examples/eval/foreign_eval/Makefile
Normal file
2
examples/eval/foreign_eval/Makefile
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
TOP=../../..
|
||||||
|
include ../../foreign.mk
|
1
examples/eval/foreign_eval/README
Normal file
1
examples/eval/foreign_eval/README
Normal file
@ -0,0 +1 @@
|
|||||||
|
run a string of Haskell code from a C program.
|
1
examples/eval/foreign_eval/expected
Normal file
1
examples/eval/foreign_eval/expected
Normal file
@ -0,0 +1 @@
|
|||||||
|
10946
|
16
examples/eval/foreign_eval/main.c
Normal file
16
examples/eval/foreign_eval/main.c
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
#include <stdio.h>
|
||||||
|
|
||||||
|
#include "EvalHaskell.h"
|
||||||
|
|
||||||
|
int main(int argc, char *argv[])
|
||||||
|
{
|
||||||
|
int *p;
|
||||||
|
hs_init(&argc, &argv);
|
||||||
|
p = hs_eval_i("let fibs = 1:1:zipWith (+) fibs (tail fibs) in fibs !! 20 :: Int");
|
||||||
|
if (p == NULL)
|
||||||
|
printf("failed!\n");
|
||||||
|
else
|
||||||
|
printf("%d\n",*p);
|
||||||
|
hs_exit();
|
||||||
|
return 0;
|
||||||
|
}
|
2
examples/eval/foreign_eval1/Makefile
Normal file
2
examples/eval/foreign_eval1/Makefile
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
TOP=../../..
|
||||||
|
include ../../foreign.mk
|
1
examples/eval/foreign_eval1/expected
Normal file
1
examples/eval/foreign_eval1/expected
Normal file
@ -0,0 +1 @@
|
|||||||
|
10946
|
16
examples/eval/foreign_eval1/main.c
Normal file
16
examples/eval/foreign_eval1/main.c
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
#include <stdio.h>
|
||||||
|
|
||||||
|
#include "EvalHaskell.h"
|
||||||
|
|
||||||
|
int main(int argc, char *argv[])
|
||||||
|
{
|
||||||
|
char *p;
|
||||||
|
hs_init(&argc, &argv);
|
||||||
|
p = hs_eval_s("show $ let fibs = 1:1:zipWith (+) fibs (tail fibs) in fibs !! 20");
|
||||||
|
if (p == NULL)
|
||||||
|
printf("failed!\n");
|
||||||
|
else
|
||||||
|
printf("%s\n",p);
|
||||||
|
hs_exit();
|
||||||
|
return 0;
|
||||||
|
}
|
2
examples/eval/foreign_should_fail/Makefile
Normal file
2
examples/eval/foreign_should_fail/Makefile
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
TOP=../../..
|
||||||
|
include ../../foreign.mk
|
2
examples/eval/foreign_should_fail/expected
Normal file
2
examples/eval/foreign_should_fail/expected
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
<eval>:1: parse error on input `in'
|
||||||
|
failed!
|
1
examples/eval/foreign_should_fail/expected.604
Normal file
1
examples/eval/foreign_should_fail/expected.604
Normal file
@ -0,0 +1 @@
|
|||||||
|
failed!
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
x
Reference in New Issue
Block a user