Hacker News new | past | comments | ask | show | jobs | submit login
Dale – A Lisp-flavoured C (github.com/tomhrr)
192 points by macco on April 10, 2017 | hide | past | favorite | 65 comments



I would love to see announcements of new languages include a rationale. Why was this created? What problem was the author trying to solve that they thought the solution was a new language? What short comings did existing languages have that this new language overcomes?

This is not a critique; I'm just curious.


A lot of the responses to hermitdev point out that a language can just be for fun, and need not "solve" anything.

This is true, but hermitdev's point still stands: why was this particular idea pleasurable? Why c and lisp? Is it because they want a fast compiling lisp? Did they like the syntax of c but find it easier to implement in lisp's tree-like structure?

What's the character motivation in this scene?


Further clouding the picture is that Dale is actually implemented in C++. The inner syntax under the S-exps is grotty C++ data structures and some brutally ugly code to deal with them that doesn't seem to even try to leverage the ability of C++ to simulate a dynamic language (like with suitable.

Like look at this fragment from https://github.com/tomhrr/dale/blob/master/src/dale/Function...

  bool
  isUnoverloadedMacro(Units *units, const char *name,
                      std::vector<Node*> *lst,
                      Function **macro_to_call)
  {
      std::map<std::string, std::vector<Function *> *>::iterator
          iter;
      Function *fn = NULL;
      for (std::vector<NSNode *>::reverse_iterator
              rb = units->top()->ctx->used_ns_nodes.rbegin(),
              re = units->top()->ctx->used_ns_nodes.rend();
              rb != re;
              ++rb) {
          iter = (*rb)->ns->functions.find(name);
          if (iter != (*rb)->ns->functions.end()) {
              fn = (*iter->second)[0];
              break;
          }
      }
This is not even up to good C++ coding practice, what with the raw exception-unsafe pointers and whatnot.

Name represented as char * ? I stopped using C strings in C++ code around 1998, other than in low level code interfacing with things that require them. Kids that were born then are now in college. If you're doing Lisp manipulation, you want anything that is a "name" of some kind to be an interned symbol, which quickly compares to another symbol variable as a pointer.

Why would you do all this to yourself, if (or so it seems) you know enough about Lisp to want a C-like language in S-exp syntax?!


This is pretty standard "C-with-classes" style of coding. You see it a lot in embedded systems too. Ok in this case it's more like "C with the STL", but it's close enough. Those would exactly be the two things I'd want to add if I were a C programmer longing for a little more power.

All your critique would also apply if this were written in idiomatic C. C pointers are exception unsafe by definition because there are no exceptions. STL containers don't throw exceptions other than out-of-range exceptions, which isn't much better or worse than the access violation you'd get for the same bug in plain C.

One good reason why the code might be like this is the author doesn't know all 2568287 C++ features that are required to write modern idiomatic C++ - but the author does know C and std::map.


I've just been browsing the txr code and I must say, it's pretty poorly written.

Just take this random example http://www.kylheku.com/cgit/txr/tree/rand.c#n138 . The abuse of the preprocessor for checking pointer size is definitely poor practice.

Because of this I won't even bother thinking critically about the language itself. (Though it does make my eyes bleed to read.)

In summary, your comment is low effort, and I hope more people call you out for it.


> The abuse of the preprocessor for checking pointer size is definitely poor practice.

[Non-moving-target reference with commit hash: http://www.kylheku.com/cgit/txr/tree/rand.c?id=6ca6be767f8ac... ]

The problem with your statement is that it isn't true.

Selecting code alternatives using #ifdef is one of the benign uses of the C preprocessing feature, exemplifying "use it like this" in contrast with "bad" uses. In many a coding guide you can it recommended to use the preprocessor to do a little #ifdef here and there, while avoiding crazy macros. (Of course, code can turn into a hairball of nested #ifdefs, which everyone rightfully hates, but this sort of use is not characterized with the word abuse. It is just use.)

We could split the function into two copies for those alternatives, put them into separate files and then have something slick in the Makefile to pick the correct file (not the GNU Make ifeq syntax, of course; that would be ironic).

However, those two functions would then contain lots of repeated code also. If something has to change, two identical parts in similar functions have to be updated.

Okay so then we could refactor the function so that all the common things are in a generic part, and then the switched pieces are in a helper inline function (included from one of two different files, etc).

It's not clear that it would improve things in that particular case.

I really have no idea what, if anything, would be worth doing, and that could be due to my mental limitations. If someone sends me a plan about how to refactor that code in a good way (just an outline with bullet points in plain language, no code) I will seriously consider it, and possibly do the work.


Your comment is amusing but otherwise doesn't contradict the claims of parent. Probably better off on a thread discussing parent's tech. This thread is about the Dale language and its implementation. Surely you of all people would appreciate that both should be done with some quality, esp in implementation language. You didn't pick shoddy C++. You went with a language that you thought was better among a number of them. I haven't heard gripes about your code so far so maybe the style was OK, too.

So, your parody is interesting but maybe counterproductive. It doesn't change the fact that the OP is already using language that isn't great for this in a way that doesn't inspire confidence. That's worth noting always as shitty implementations can lead to bugs for early adopters. Best to call out problems in parent's work in threads on that, a mailing list or repo if there is one, etc.


My code is certainly flawed, the question is if it behaves more or less as advertised.

I would be more open to such criticism from users of the software. In my opinion that type of comment is worse than backseat driving, at least in that case the passenger is a stakeholder.


"In my opinion that type of comment is worse than backseat driving"

By that logic, I'd have to lease a mainframe for millions of dollars before critiquing aspects of their offering. Likewise, I'd have to pay Oracle $70,000 per processor. If flawed FOSS, I'd have to go through the pain of setting up and using it instead of submitting the flaw. Your analogy would apply better if you said "person in back seat shouting about an obstacle they're about to hit to driver that fell asleep."

Realistically, though, better to critique than use flawed products unless it gets the job done enough to be worth using anyway. No need to become a stakeholder.


The thing is, I really think that the sort of thing which Dale is is a very, very good idea. Any time I see a link to "C-like language done up in S-exps with macros", I'm very keenly interested, and have all these expectations. I should perhaps do a better job of hiding my disappointment.


Yikes, yeah, this is bad. I only went through the readme, I didn't look at the source, and just...no. NULL? no. nullptr, please. And, as you've said, const char* as strings? again, no. I know GLS isn't terribly old, but something like string_view is in order. Arguments as undecorated pointers...no. Are they in or out? optional? C++ has semantics to indicate these things (even without GLS).


This thread of conversation is unproductive. Could you link me to both of your respective github accounts so I can nitpick without any stake or contribution in your projects?


You may criticize anything I've ever written that you can find. As a software developer, that's my daily assumption.

The increasing adoption of code review tools like Gerrit is one of the best things that has been happening in recent years.

No piece of code I've written in the past few years on the job has gone into the stream without the approval of several people.

Frequently, it had to be revised. More than once.


No, because I'm bound by contract, cannot disclose my source, and I'm under NDA. That said, my comments are not unreasonabe or overly critical for C++11 or newer code. Quite honestly, if I desired to, there's more I could probably find issue with in even that small of a code sample.


It's not unproductive. The project was linked, and code quality is relevant.


I think that this developer, just like I, have drunk the homoiconic kool-aid. Homoiconic syntax is _fucking serious_ as far as productivity and enjoyment goes, it's like a whole separate world.


Or they just had fun? Why does a language have to be useful for anything?


This is a completely legitimate answer to my questions.

I'm just curious - I'm not passing judgement at all. Hell, I've a C++ compiler I've been working on for a while (it doesn't get much attention). Why? It forces you to confront the spec (all +1200 pages of it) and to understand the nuances of the language. I don't ever expect my compiler to be released publicly or used in production, but it's for my own personal growth and to increase my understanding of the language, which makes me a better C++ developer. I'm more of a data systems programmer (I work in finance), so working on a compiler exposed me to other areas I wasn't familiar with (parsers, grammars, ASTs, etc).

But, more substantive answers to the questions I've asked of a language(s) designer(s) can provide insight into what the language is well (or poorly) suited for or for it's potential longevity. If it was created for fun by the author, it's probably significantly less likely to have large adoption.

Unfortunately, I now commute 2 hours a day in the car, so my spare reading time is greatly diminished, so I don't have 40 minutes twice a day to read something novel like this in detail on the train like I used to have.


I'm holding out for Google Narration.

"Google, read me Hacker News!"

"Are you sure? Looks like nothing interesting this morning. Oh wait, Aphry just posted something. Would you like to hear it?"

"Sure!.. Wait, Google, why don't you drive, and I'll read it!"

"Sounds great. Should I start the espresso machine as well?"


Very few languages reach any significant adoption, regardless of why/how they were designed.

And a lot of the popular languages are pretty terrible.

Im also sceptical of that designing the language with practicallity in mind would make it signifficantly more useful.

You'd have to judge each of them by their own merit.


Writing a language for fun is totally fine, but it is useful to know that fact when the language is being shared. Am I looking at this language to see if it will be useful to solve a problem I have that existing languages don't solve, or is it just an exercise, or is it an improvement?


All of those points are moot without adoption and maturity, something a new language by definition won't have.


No, but there is still a big difference between a new language with a still-small community, but with ambition and some early attempts to apply it at "real" problems, vs a brainfuck-like, just for kicks language.


Everything is done for a reason. Sometimes that reason is just "to have fun / learn something new / expand ones horizons" and that's okay. But sometimes the reason is "to get the same amount of work done faster" and that's also okay. A lot of the mainstream languages were made for the second reason, but probably most hobbyist languages were made for the first.


I created one a long time ago for reasons that seem to show up in others. Those included: Lisp-style macros are raw productivity/power; incremental, per-function compilation plus REPL = blazing, iteration speed; if using LCD of features, I can synthesize to more than one language/VM target; I can automate safety checks for C pitfalls without looking at cluttered code; way cleaner way to handle errors with similar benefit.

Sadly, I lost that tool in a triple, HD crash along with most work. Loved it, though. I never even fully learned LISP or C but the subsets let me crank out tons functionality really fast then run it through optimizing C compilers.


You should have put it up on GitHub.


It didn't exist at the time. I was gonna maybe turn it into a product, too. That be a great default today, though. Now I also know one can have paid, shared source. I might have opened it with free, perpetual licenses for any contributors.


A regular, non-convoluted[1] syntax for C? Hygienic macros instead of defines? These two are sufficient reasons to consider it.

But they also seem to offer namespaces, type inference (even through macros), sum types, anonymous functions, overloaded functions, a form of runtime introspection, and a bunch of goodies like containers in the stdlib.

Quite an offering, I'd say.

[1]: http://c-faq.com/decl/spiral.anderson.html


Namespaces & function overloading would make C a far saner place to live...


I would think that the rationale for any language with the syntax of Lisp but semantics similar to X is obvious: you want a language like X (that can be used for the things that X is used for) but with the meta-programming capabilities provided by Lisp-style macros.

I can definitely see a Lisper thinking this rationale is so obvious that it does not require mentioning.


Yep, macros are the only motivation needed for wrapping a language like C, and this project does a lot more than that. I've tried to come up with a way to do this myself, but I always hit a wall when it comes to integrating with the C preprocessor.


From forum discussion:

" Its development was prompted by the Common Lisp and Scheme tutorials that contrast syntactic macros with C preprocessor macros, and wanting to see whether syntactic macros could work in a lower-level language. " -Tom Harrison [1]

[1] https://groups.google.com/forum/#!msg/dale-lang/h73oNq5U6MQ/...


Thanks for posting this. I had not perused the forums. This definitely sheds more light on what I was originally asking than just reading the readme.


You're welcome; I had to dig to find it. I, like you, would have preferred to see the motivation in the top-level documentation. It helps so much to see the core concerns that an abstraction is built upon, especially something as tangled and complex as a high-programming language.


> What problem was the author trying to solve that they thought the solution was a new language?

It could well be a cultural thing: for some people, "creating a new language" is a serious undertaking, but for others (especially those exposed to Lisp and Scheme), creating a new language is something you might do if you're curious about something. At the end of the day, creating a language just isn't all that hard to give it so much thought.


In Lisp / Scheme, people may create a tiny language to express the problem.

Steps: 1. Create a new language in which the problem is trivially expressed. 2. Solve the problem in the new language.

Lisp / Scheme are ideal for this.

Example: solve some sort of puzzle game. Create a data structure that represents the board, a game piece and a move. Operators upon these. Given a board and a legal move, return a new board with the move applied. Given a board, give all of the possible legal moves (for a particular player if this a multi-player game instead of a puzzle).

Once you have that language it becomes easy to use your favorite off the shelf search algorithms. Depth first. Breadth first. A*. Etc.

But, this is probably not the same reason to create a "language" as the article. And the notion of "language" is quite different.


What makes that a 'language' and not just a program? When I create a 'GameBoard' class in an OO language, I don't tell someone I created a new language.

Doesn't a language have to be turing complete to be a language?


I haven't used any of these, but my understanding is that Agda and Coq are not Turing-complete, nor is Idris with the totality checker on. Never seen anyone dispute these being referred to as "languages."


Are you sure your GameBoard class doesn't support the necessary primitives to be Turing complete? All you need is store and jump.


Why don't you just read the readme? The additional features are at the top, and examples in the body.


I did. I saw lots of examples, but no answers to the questions I posed. The one source of rationale I saw was Lisp-like with no GC.

Additionally, my comment was not directed solely at Dale, but new language announcements in general.


Oh hey, I posted this on /r/lisp just earlier today. I discovered it because of a Google rabbit hole triggered by a rediscovery of this article[0] on Naughty Dog's GOAL (Game-Oriented Assembly Lisp). It looks really interesting, although I can't imagine doing serious programming in it (as much as I'd love to).

[0]: http://www.codersnotes.com/notes/disassembling-jak/


Very neat! Looks like this is an even older language than the one I'm familiar with: bone-lisp. Bone-lisp is interpreted but ops for "explicit regions instead of garbage collection". There is also an awesome list of similar projects on the bone-lisp README (on which I now see Dale included) which I'll quote from:

> Somewhat related Free Software projects:

> * Pre-Scheme is a GC-free (LIFO) subset of Scheme

> * Carp is "a statically typed lisp, without a GC"

> * newLISP uses "One Reference Only" memory management

> * MLKit uses region inference (and a GC)

> * Linear Lisp produces no garbage

> * Dale is basically C in S-Exprs (but with macros)

> * ThinLisp is a subset of Common Lisp that can be used without GC

[1] https://github.com/wolfgangj/bone-lisp


At a glance, the compiler looks solid, with full LLVM integration instead of compiling to C. That'll pay dividends down the road. Really great job :)


This looks really nice. A low level Lisp as a system programming language - a C replacement - is an interesting idea. First thought after going through the readme: does it try to provide stable ABI? This is an important feature for a C replacement; for example, C++ has not managed to get it right, although the idea was to provide a better, object oriented C... What worries me in particular is that Dale has overloaded functions, that makes things harder as far as stable API is concerned, but maybe I'm wrong. Nice stuff, anyway. I can see a little PLT trend of new high level languages that want to be good in doing low level stuff and don't use GC. I like that!


A systems programming language need not eschew GC. Zetalisp and Common Lisp are perfectly fine for Lisp Machine operating system implementation, without falling back to a language other than assembly for isolated bits of the very lowest-level code. Mezzano reaffirms this on x86-64 too.



It saddens me that one of the main comment threads there is complaining about parentheses. :(

Some of the shouts out to thinlisp and other systems that did similar to this before are interesting. Very interesting.


Another Lisp implemented in C++, but with a GC:

https://github.com/rongarret/Ciel/


Programmed by Mr Worldwide himself?!


isn't this a C-flavored lisp?

I mean, it's not C. And it is a lisp.

So doesn't it make sense to call it "lisp with a flavor of C" rather than the other way around?

Cool project!


It might've been named in the same vein as Lisp-Flavoured Erlang, which is indeed (last I checked) a Lisp and very much not Erlang (at least language-wise; it still uses the Erlang VM and OTP).

On another note, the lack of garbage collection at the very least would make Dale very abnormal in terms of being a Lisp v. some other language that happens to embrace s-expressions for its syntax.


Can we have a C-flavored C with the extra features of Dale?


I used one that almost did what you ask. I refused to work with C++ codebases. Better to start with a language actually designed for metaprogramming, etc that thrn synthesizes C. If it has IDE & ecosystem, you get its language-level benefits then C's on tooling side. Win, win.


Any high-order functions, TCO? Is there an implementation of Everything Is An Expression and Everything Is A First-class Value principles? Type-tagging, which implements Values Has A Type, Not Variables principle, which gives The Numerical Tower and other nice things for free? Proper lexical scoping based on Environments which makes every lambda a closure? I am not asking about homogeneity and the possibility of the First-class Macros it provides.

No? So what then makes it Lisp-flavored? CONS CAR CDR and parentheses?


C's syntax is horrible, but we can live with it. Where are the attempts to fix C's semantics, dammit? This project is as if a mechanic were given a broken car, and the first thing he did were repainting it.


How does this compare with Chicken Scheme?


How do you pronounce it? Like Pitbull in "dale!"?


Wonder how hard it is to write a quine in Dale...


I want this for Go!


Why isn't it written in Lisp? It would be so much easier...

See for example C-MERA:

https://github.com/kiselgra/c-mera


C-MERA's default code generator is textual C. This project uses LLVM's API, which is sadly difficult to utilize outside of C++.


https://github.com/drmeister/clasp

From the CLASP page:

> Clasp is a new Common Lisp implementation that seamlessly interoperates with C++ libraries and programs using LLVM for compilation to native code. This allows Clasp to take advantage of a vast array of preexisting libraries and programs, such as out of the scientific computing ecosystem. Embedding them in a Common Lisp environment allows you to make use of rapid prototyping, incremental development, and other capabilities that make it a powerful language.

you wrote:

> This project uses LLVM's API, which is sadly difficult to utilize outside of C++.

I would guess that you could use LLVM's API via CLASP. The author initially wrote ( https://drmeister.wordpress.com/2014/09/18/announcing-clasp/) :

> Clasp exposes the LLVM C++ library within Common Lisp providing a rich, dynamic, programming environment for exploring the LLVM library and for writing new compilers that generate LLVM-IR.

Then you could write the parser/compiler in Lisp, which supports manipulating s-expressions on a simpler level than C++ does.


1) That's an interesting project. Worthy of discussion on its own.

2) "Why isn't it written in Lisp?" It's 42% C++ according to GitHub.

3) Based on a hunch and past experiences, I'm going to ignore any follow ups on this thread.

EDIT: And now I look like a jerk because lispm replaced his terse comment with a fuller comment. My apologies for even engaging in the first place. Should have known better.


I'm just wondering if there isn't already some large infrastructure in the Lisp world, which could have been reused. Something like CLASP was especially written to seamlessly interoperate with LLVM. Sometimes working with an existing system is difficult, but sometimes it can help both projects.

Anyway, Dale looks like a cool project.




Guidelines | FAQ | Lists | API | Security | Legal | Apply to YC | Contact

Search: