NAME
Emacs::Lisp - Support for Perl embedded in GNU Emacs
SYNOPSIS
In Emacs
M-x load-library RET perl RET
M-x perl-eval-expression RET 2+2 RET
In Perl
use Emacs::Lisp;
&switch_to_buffer('*scratch*');
&insert("Hello, world!\n");
setq { $cperl_font_lock = t };
&add_hook(\*find_file_hooks,
sub { &message("found a file!") });
use Emacs::Lisp qw($emacs_version $perlmacs_version);
save_excursion {
&set_buffer(&get_buffer_create("whatever"));
&insert("This is Emacs version $emacs_version,\n");
&insert("Perlmacs version $perlmacs_version.\n");
&insert("Emacs::Lisp version is $Emacs::Lisp::VERSION.\n");
};
DESCRIPTION
Until now, you could customize your Emacs environment using Lisp. Now you can use Perl, too. This module allows Perl code to call functions and access variables of Lisp. It also maps some Perl syntax into Lisp semantics.
You still need to learn some Lisp in order to understand The Elisp Manual, which you will need if you wish to learn about the details of Emacs programming. Hopefully, this situation will be cured by the appearance of well-documented Perl modules that give everything in Emacs a nice, object-oriented wrapping.
EMACS SUPPORT FOR PERL
The pmacs program can compile and evaluate Perl code via Emacs Lisp or the command line, independently of the Emacs::Lisp module.
Functions
Some Perl-related functions are built into Lisp. Use C-h f <function-name> RET
within Emacs for documentation on these.
perl-eval STRING &optional CONTEXT
perl-call SUB &optional CONTEXT &rest ARGS
make-perl-interpreter &rest ARGV
get-perl-interpreter
set-perl-interpreter INTERPRETER
perl-run &optional INTERPRETER
perl-destruct &optional INTERPRETER
You need to do M-x load-library RET perl RET
(or hack your ~/.emacs) to get the following functions.
perl-eval-expression EXPRESSION
perl-eval-region START END
perl-eval-buffer
perl-load-file NAME
Caution: the exact meaning of perl-load-file
is a topic of research and may change.
Data Conversions
When data is passed between Lisp and Perl within Emacs, some basic principles apply. These are goals that may never be entirely met, due to the differences between Perl and Lisp.
Whatever Perl gives to Lisp should get dereferenced once.
A Perl reference gets converted into a Lisp reference to the thing referenced, not a Lisp reference to the Perl reference. (I find this to be intuitively correct behavior.)
As an exception to the previous rule, arrayrefs become lists.
Lists are a central data structure in Lisp. To make it as easy as possible to pass lists to Lisp functions that require them, I think it is necessary to perform a double dereference when converting Perl array references. (By "double dereference" I mean that
\@a
is dereferenced once to produce@a
, and again to get$a[0]
,$a[1]
, and so on.) Therefore, a Perl expression such as["x", ["y", 1]]
is converted to
'("x" ("y" 1))
in Lisp.
This kind of conversion entails quite a bit of overhead and precludes "passing by reference" between the two languages, since it is a "deep" copying operation. Changes made by Lisp to the list will not affect the Perl array of which it is a copy. See "TO DO" for a planned solution to this problem.
Converting a non-dereferenceable value requires some kind of primitive conversion.
For example, Lisp integers, floats, and strings all become Perl scalars. A scalar (other than a reference) converted to Lisp will become either an integer, a float, or a string. Glob references in package
main
become symbols in Lisp (subject to the usual underscore-to-hyphen translation).Lisp's `nil' is equivalent to Perl's `undef' or `()'.
In Lisp,
nil
is really a symbol. However, it is typically used as the boolean value false. Perl's equivalent of symbols (glob references) evaluate to true in boolean contexts. Convertingnil
to anything other thanundef
would be disastrous for programmers' mental health!Generally, converted values can be converted back to Perl more or less unchanged.
But not always. For example,
\*nil
would becomeundef
.Lisp objects which are not Perl stuff and have no natural counterpart become `Emacs::Lisp::Object' blessed references.
These hold the actual Lisp data and a means of protecting them from garbage collection. An
Emacs::Lisp::Object
reference converted back to Lisp is the Lisp object, naturally.
Scripts
Perlmacs can run Perl programs. By default, Perlmacs is installed under two names, pmacs and perlmacs. Which name is used to invoke the program can determine how it parses its command line.
If perlmacs is used (or, more generally, any name containing "perl"), it behaves like Perl (expecting a script, etc.). Otherwise, it behaves like Emacs (opening a window, creating a buffer, etc.) In either case, the first command line arg can take precedence. If it is --emacs, Emacs takes control. If it is --perl, we play by Perl's rules.
When you use Emacs::Lisp
in a perlmacs script, a Perl sub named Emacs::main
may be used to invoke the Emacs editor. This makes it possible to put customization code, which would normally appear as Lisp in ~/.emacs, into a Perl script. For example, this startup code
(setq
user-mail-address "gnaeus@perl.moc"
mail-self-blind t
mail-yank-prefix "> "
)
(put 'eval-expression 'disabled nil)
(global-font-lock-mode 1 t)
(set-face-background 'highlight "maroon")
(set-face-background 'region "Sienna")
could be placed in a file with the following contents:
#! /usr/local/bin/perlmacs
use Emacs::Lisp;
setq {
$user_mail_address = 'gnaeus@perl.moc';
$mail_self_blind = t;
$mail_yank_prefix = '> ';
$eval_expression{\*disabled} = undef;
};
&global_font_lock_mode(1, t);
&set_face_background(\*highlight, "maroon");
&set_face_background(\*region, "Sienna");
exit Emacs::main($0, "-q", @ARGV);
When you wanted to run Emacs, you would invoke this program.
The arguments to Emacs::main
correspond to the argv
of the main
function in a C program. The first argument should be the program's invocation name, as in this example. -q inhibits running ~/.emacs (which is the point, after all).
See "BUGS" for problems with Emacs::main
.
PERL SUPPORT FOR LISP
The Emacs::Lisp module allows Perl programs to invoke Lisp functions and handle Lisp variables using Perl's syntax.
You don't have to do any work, other than use Emacs::Lisp
, to make subs call their Lisp counterpart. However, tying Lisp variables to Perl variables is not quite so automatic. In all cases, hyphens appearing in Lisp names are translated to underscores in Perl, and vice versa.
Functions
This code calls the hypothetical Lisp function foo-bar
with arguments 4
and t
.
&foo_bar(4, t);
The Lisp syntax for the same call would be
(foo-bar 4 t)
The ampersand (&
) is really only needed for calling Lisp functions, such as read
, eval
, and print
, which are Perl keywords. But using it is a good habit.
Symbols
Many Lisp functions take arguments that may be, or are required to be, symbols. In Lisp, a symbol is a kind of name, but it not the same type as a string.
Lisp programs typically use the quote
operator to specify a symbol. For example, this Lisp code refers to the beep
symbol:
(run-at-time nil 1 'beep)
The above is actually an abbreviated syntax for this:
(run-at-time nil 1 (quote beep))
Perlmacs uses glob references of package main
to specify symbols. A literal globref begins with a backslash followed by an asterisk, so the last example would be written as
&run_at_time(undef, 1, \*beep);
in Perl. (You may want to do &cancel_function_timers(\*beep)
soon after trying this example.)
Note that only globs from package main
may be used as Lisp symbols, so code that is compiled in another package must use the form \*::sym
rather than \*sym
.
Variables
In Lisp, variables play a role akin to that of Perl scalars. A variable may hold a number, a string, or a reference to any type of complex Lisp data structure. (They are not called references in Lisp, but rather "objects".)
You can create a Perl alias for any reasonably named Lisp variable by saying use Emacs::Lisp qw($varname)
. Thereafter, assignment to $varname
will update the Lisp value. Changes made to the variable in Lisp will be reflected in Perl when $varname
is used in expressions.
This example saves and replaces the value of the Lisp variable inhibit-eol-conversion
:
use Emacs::Lisp qw($inhibit_eol_conversion);
$old_val = $inhibit_eol_conversion;
$inhibit_eol_conversion = 1;
This sort of thing could be accomplished in Lisp as follows:
(setq old-val inhibit-eol-conversion)
(setq inhibit-eol-conversion 1)
(but you would probably rather use let
instead, for which there is still no convenient Emacs::Lisp alternative). See also the setq
function below.
Property Lists
Lisp symbols all have an associated object called a plist (for "property list"). Although it is actually an object just like any other, the plist is typically used in a way vaguely resembling Perl's hashes.
Note that a plist is different from a Perl hash. Lookups are not based on string equality as with Perl, but rather on Lisp object equality (of the eq
variety). For this reason, it is best to stick to the Lisp convention of using only symbols as keys. (See "Symbols".)
Emacs::Lisp provides a shorthand notation for getting and setting plist elements. If you say use Emacs::Lisp qw(%any_name)
, then subsequent access to the hash elements of %any_name
will really get or set the corresponding plist entries (i.e., properties of the Lisp symbol any-name
).
For example, the following Perl and Lisp fragments are more or less equivalent:
# Perl fragment
use Emacs::Lisp qw(%booboo %upcase_region);
$booboo{\*error_conditions} = [\*booboo, \*error];
$can_upcase = ! $upcase_region{\*disabled};
; Lisp fragment
(put 'booboo 'error-conditions '(booboo error))
(setq can-upcase (not (get 'upcase-region 'disabled)))
See also the setq
function below.
Special Forms
So-called "special forms" in Lisp, such as setq
and defun
, do not work the same way functions do, although they are invoked using the function syntax. (Here you see the vast philosophical chasm separating Perl from Lisp. While Perl might have five syntaxes for doing the same thing, Lisp uses one syntax for two different purposes!)
Some special forms are equivalent to Perl operators, such as if
and while
. Others have meanings peculiar to Lisp. A few special forms are implemented in Emacs::Lisp. They are listed below. If you try to call a special form that has not been implemented, you will get an error message which may propose an alternative.
- defun SYMBOL,DOCSTRING,SPEC,CODE
- defun SYMBOL,DOCSTRING,CODE
- defun SYMBOL,SPEC,CODE
- defun SYMBOL,CODE
-
Make CODE callable as the Lisp function SYMBOL. This is Lisp's version of Perl's
sub
keyword. A function defined in this way becomes visible to Lisp code.This is useful for defining Emacs commands. Commands are functions that the user can invoke by typing
M-x <function-name>
. A command may be bound to a key or sequence of keystrokes. See the Emacs documentation for specifics.As in Emacs Lisp, when defining a command, you must specify the interactive nature of the command. There are various codes to indicate that the command acts on the current region, a file name to be read from the minibuffer, etc. Please see The Elisp Manual for details. Emacs::Lisp'
defun
uses a value returned byinteractive
as the SPEC for this purpose. See "interactive".This example creates a command,
reverse-region-words
, that replaces a region of text with the same text after reversing the order of words. To be user-friendly, we'll provide a documentation string, which will be accessible through the Emacs help system (C-h f
).use Emacs::Lisp; defun (\*reverse_region_words, "Reverse the order of the words in the region.", interactive("r"), sub { my ($start, $end) = @_; my $text = &buffer_substring($start, $end); $text = join('', reverse split (/(\s+)/, $text)); &delete_region($start, $end); &insert($text); });
- interactive SPEC
- interactive
-
Used to generate the third (or, in the absence of a doc string, second) argument to
defun
, which see. This determines how a command's arguments are obtained.SPEC may be a string, as described in The Elisp Manual, or a reference to code which returns the argument list.
- save_excursion BLOCK
-
Execute BLOCK within a Lisp
save-excursion
construct. This restores the current buffer and other settings to their original values after the code has completed.Please read The Elisp Manual for details.
- setq BLOCK
-
BLOCK is searched for assignments (currently only at top level, but this may change) of the form
$var = EXPR;
where
$var
is a scalar variable or hash element. Every such variable is imported from the Emacs::Lisp module, as if you had saiduse Emacs::Lisp qw($var)
.Afterwards, BLOCK is executed. Thus, this code
use Emacs::Lisp; setq { $A = 2*$foo[5]; $B{\*foo} = "more than $A"; };
would have exactly the same effect as
use Emacs::Lisp qw(:DEFAULT $A %B); $A = 2*$foo[5]; $B{\*foo} = "more than $A";
The following, which does not tie or import any variables, has the same effect on Lisp as the above:
use Emacs::Lisp (); &Emacs::Lisp::set( \*A, 2*$foo[5] ); &Emacs::Lisp::put( \*B, \*foo, "more than " . &Emacs::Lisp::symbol_value( \*A ));
BUGS
There are quite a few bugs in Perlmacs and Emacs::Lisp. These are some of the ones I know about. See also the file src/BUGS in the Perlmacs distribution. If you find other bugs, please check that you have the latest version, and email me.
Crashes during goto
This type of thing crashes or destabilizes Perl:
perlmacs -MEmacs::Lisp -le 'exit Emacs::main("test","-batch","-eval",q((perl-eval "goto foo;"))); foo:' perlmacs -MEmacs::Lisp -le '&perl_eval(q(goto foo;)); foo:'
Possible memory leaks.
I have not tested for memory leaks.
Problems with `Emacs::main'.
The
Emacs::main
sub may open an X display and not close it. That is the most obvious of many problems withEmacs::main
. The thing is, Emacs was not written with the expectation of being embedded in another program, least of all a language interpreter such as Perl. Therefore, when Emacs is told to exit, it assumes the process is really about to exit, and it neglects to tidy up after itself.For best results, the value returned by
Emacs::main
should be passed to Perl'sexit
soon, as in this code:exit (Emacs::main($0, @args));
Perl's `local()' doesn't have the effect of Lisp's `let'.
It should. At least, there should be an easy way to make a local binding of a Lisp variable in Perl.
Input/Output.
If Perl code tries to read from
STDIN
, Emacs becomes unresponsive. This is very distressing. Also, Perl loves to print error and warning messages toSTDERR
, which can be disconcerting. The messages should probably be redirected to a buffer or the minibuffer.I shudder to think what would happen if you used Perl compiled with sfio.
%ENV, %SIG, `setenv'.
Perl and Emacs use environment variables and signal handlers in different, incompatible ways. This needs to be coordinated. I've made a start with %ENV, but more needs to be done.
Probably can't use a coderef as an error handler or protected form.
I think this will be easy to fix, I just haven't had the need yet.
A crash is likely if Perl code modifies the scalar value in an `Emacs::Lisp::Object' blessed reference.
Don't do that.
See also the (many) FIXMEs in src/perlmacs.c and elsewhere in the Perlmacs source.
CAVEATS
Circular data structures are bad.
See "Two-Phased Garbage Collection" in perlobj. Lisp data structures may be recursive (contain references to themselves) without the danger of a memory leak, because Lisp uses a periodic-mark-and-sweep garbage collector. However, if a recursive structure involves any Perl references, it may never be destroyable.
For best results, Perl code should deal mainly with Perl data, and Lisp code should deal mainly with Lisp data.
Cross-language references incur a slight overhead.
For the benefit of Lisp's garbage collection, all Perl data that is referenced by Lisp participates in the mark phase. For the benefit of Perl's garbage collection, all Lisp objects that are referenced by Perl maintain a (kind of) reference count.
A chain of Perl -> Lisp -> ... -> Perl references may take several garbage collection cycles to be freed. (At least, that is my theory. I haven't verified it experimentally.) It is therefore probably best to keep the number and complexity of such references to a minimum.
To the extent that the Perl-to-Emacs interface is independent of the Lispish implementation of Emacs, these performance issues are fixable in principle by reimplementing Emacs' internals.
TO DO
Special forms: let, catch, unwind-protect, condition-case, etc.
Better support for Lisp macros.
Find a way to convert filehandles to the Emacs equivalent.
Make a way to get a tied filehandle that reads a buffer.
Provide a way to bypass the default data conversion routines when passing arguments and function return values between the languages.
Possible syntax:
$x = Emacs::Lisp::Object->funcall("func", [1, "2"], 3.0);
The value returned by the Lisp function
func
would be stored in$x
as a blessed reference to a Lisp object, even if the object has a natural Perl equivalent. The first argument([1, "2"])
would be given tofunc
as a real Perl arrayref. (Itstype-of
would beperl-array
.) However,3.0
could be converted to a Lisp float, since there is already a way to pass it as a Perl scalar (by using\3.0
).If
$x
received, say, a Lisp list fromfunc
,$x
would belong to packageEmacs::Lisp::Cons
, where@Emacs::Lisp::Cons::ISA
is qw(Emacs::Lisp::Object)>. This package would provide methodscar
,cdr
,setcar
, andsetcdr
. As a special case, when Lisp returned objects that are really Perl data, they would be "unwrapped" rather than wrapped. (I don't see much point in wrapping a Perl object inside a Lisp object inside a Perl object.)Conversely, Lisp could be given functions to act on Perl data, such as
perl-aref
for indexing into aperl-array
. Lisp could also have a way to call Perl subs with no argument or return-value conversion.Improve perl-eval-buffer, perl-load-file, et al. (in C?)
ACKNOWLEDGMENTS
These are among the giants on whose shoulders we stand:
- Larry Wall, inventor of Perl.
-
'Nuff said.
- The developers of GNU, and Richard Stallman in particular.
-
Many thanks for the most beautiful code base that it has ever been, or will ever likely be, my pleasure to hack.
- John McCarthy, inventor of Lisp.
- Dennis Ritchie, inventor of C.
-
Both Perl and Emacs are written in C. The existence of Perlmacs kind of rests on that fact.
- �Évariste Galois (1811-1832) and Wolfgang Amadeus Mozart (1756-1791).
-
Each of them demonstrated just how much a young man can accomplish with a good mind.
- Ludwig van Beethoven (1770-1827).
-
Showed that even neurotic, deaf, aging crabapples can, from time to time, engender lasting beauty.
-
I was lucky enough to come onto the Perl scene soon after dynamic loading had reached its present form. How you all got by before then completely baffles me.
-
ExtUtils::Embed is a cornerstone of Perlmacs.
This list is incomplete.
Personal thanks to Nate Patwardhan, who sparked my early interest in Perl--and shared his .emacs with me--during our NFIC days. Nate also introduced me to gdb under gud-mode, O well was I served! :)
Thanks also to Ilya Zakharevich for (1) encouraging me in my first contribution to the Perl development effort (an xsubpp patch), and (2) a comment in cperl-mode.el about changing Emacs C source. If not for that comment, I may never have realized that it is even possible for mortals to change Emacs C source. ;-) Although I didn't implement the change he requested, I hope Ilya approves.
COPYRIGHT
Copyright (C) 1998 by John Tobey, jtobey@channel1.com. All rights reserved.
This program 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 program 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.
You should have received a copy of the GNU General Public License
along with this program; see the file COPYING. If not, write to the
Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston,
MA 02111-1307 USA
Please note: The GNU Emacs license (which is the GNU General Public License or "GPL") requires that all extensions and code designed specifically for use with Emacs be distributable under the same license. At least, that is the intention of its author. This includes dynamically linked code such as the Emacs::Lisp module and (probably) any other Perl modules that use Emacs::Lisp or the GPL-covered functions of Emacs. Refer to the file COPYING and the Emacs documentaion for full details.
SEE ALSO
perl, emacs, and The Elisp Manual (available where you got the Emacs source, or from ftp://prep.ai.mit.edu/pub/gnu/).
1 POD Error
The following errors were encountered while parsing the POD:
- Around line 920:
Non-ASCII character seen before =encoding in '�Évariste'. Assuming CP1252