I stopped everything and started writing C again
The author transitioned from Ruby on Rails to C after working with Common Lisp, developing the KC3 language and libc3 library, focusing on performance, portability, and defensive programming principles.
Read original articleThe author reflects on their journey from being a freelance developer primarily using Ruby on Rails to rediscovering the C programming language after extensive work with Common Lisp. Initially, the author aimed to learn Common Lisp quickly for a project but became deeply involved in creating various packages, including a graph database and other utilities. Despite the success and speed of Common Lisp, the author faced challenges with garbage collection and the limited applicability of their projects. Observing frustrations within the programming community regarding virtual machines and containers, the author concluded that C would be a more practical choice for performance and portability. This led to the development of a new utility library, libc3, which evolved into a language called KC3, featuring an interpreter and a web server. The author successfully ported their graph database to C89 and created various data structures and parsers, emphasizing defensive programming to minimize bugs. The project has grown to include a REPL for testing and a documentation website, showcasing the author's commitment to building a robust programming environment.
- The author transitioned from Ruby on Rails to C after extensive work with Common Lisp.
- They developed a new programming language, KC3, and a utility library, libc3.
- The focus on performance and portability led to the decision to use C over Common Lisp.
- The project includes a graph database, various data structures, and a web server.
- Defensive programming principles were emphasized to reduce bugs and security issues.
Related
The Liberating Experience of Common Lisp
The author critiques modern programming languages for their complexity, praising Common Lisp for its stability, unique developer experience, and creative freedom, making it preferable for software development.
C3 lang – A modern C alternative – 0.6.3 released
C3 is a programming language derived from C, emphasizing ergonomics and safety, with features like optionals, SIMD support, and a developing standard library for graphics and game frameworks.
On: Lisp, on Lisp, and MySelf
The author shares mixed feelings about Common Lisp, noting frustrations with its syntax but appreciating its flexibility. They recommend "On Lisp" and encourage experimentation and openness to new programming experiences.
A Closed and Common Lisp (2021)
The author enjoyed learning Common Lisp during the Advent of Code challenge, appreciating its unique blend of programming styles, powerful tooling, and multiple return values, despite some library limitations.
Why I Chose Common Lisp
The author switched from Clojure to Common Lisp after seven years, citing performance issues. Common Lisp met their needs for standalone executables, multithreading, and offered a supportive community and resources.
- Many commenters express nostalgia for C's simplicity but also highlight its challenges, such as manual memory management and lack of modern features.
- Several users mention their preference for newer languages like Rust, Zig, and Go, citing better safety, tooling, and productivity.
- There is a common theme of frustration with C's verbose syntax and the complexity of managing dependencies and build systems.
- Some commenters appreciate the learning experience and control that C provides, while others find it tedious and error-prone.
- Discussions about the potential of the KC3 language and its focus on performance and defensive programming principles are noted, with mixed reactions.
Then I moved into backend development, where I was doing all Java, Scala, and Python. It was... dare I say... easy! Sure, these kinds of languages bring with them other problems, but I loved batteries-included standard libraries, build systems that could automatically fetch dependencies -- and oh my, such huge communities with open-source libraries for nearly anything I could imagine needing. Even if most of the build systems (maven, sbt, gradle, pip, etc.) have lots of rough edges, at least they exist.
Fast forward 12 years, and I find myself getting back in to Xfce. Ugh. C is such a pain in the ass. I keep reinventing wheels, because even if there's a third-party library, most of the time it's not packaged on many of the distros/OSes our users use. Memory leaks, NULL pointer dereferences, use-after-free, data races, terrible concurrency primitives, no tuples, no generics, primitive type system... I hate it.
I've been using Rust for other projects, and despite it being an objectively more difficult language to learn and use, I'm still much more productive in Rust than in C.
I also like that C forces me to do stuff myself. It doesn't hide the magic and complexity. Also, my typical experience is that if you have to write your standard data structures on your own, you not only learn much more, but you also quickly see possibly performance improvements for your specific use case, that would have otherwise been hidden below several layers of library abstractions.
This has put me in a strange situation: everyone around me is always trying to use the latest feature of the newest C++ version, while I increasingly try to get rid of C++ features. A typical example I have encountered several times now is people using elaborate setups with std::string_view to avoid string copying, while exactly the same functionality could've been achieved by fewer code, using just a simple raw const char* pointer.
Then I try actually going through the motions of writing a production-grade application in C and I realise why I left it behind all those years ago. There's just so much stuff one has to do on one's own, with no support from the computer. So many things that one has to get just right for it to work across edge cases and in the face of adversarial users.
If I had to pick up a low-level language today, it'd likely be Ada. Similar to C, but with much more help from the compiler with all sorts of things.
def route = fn (request) {
if (request.method == GET ||
request.method == HEAD) do
locale = "en"
slash = if Str.ends_with?(request.url, "/") do "" else "/" end
path_html = "./pages#{request.url}#{slash}index.#{locale}.html"
if File.exists?(path_html) do
show_html(path_html, request.url)
else
path_md = "./pages#{request.url}#{slash}index.#{locale}.md"
if File.exists?(path_md) do
show_md(path_md, request.url)
else
path_md = "./pages#{request.url}.#{locale}.md"
if File.exists?(path_md) do
show_md(path_md, request.url)
end
end
end
end
}
[1] https://git.kmx.io/kc3-lang/kc3/_tree/master/httpd/page/app/...Otherwise I use Go if a GC is acceptable and I want a simple language or Rust if I really need performance and safety.
It's difficult because I do believe there's an aesthetic appeal in doing certain one-off projects in C: compiled size, speed of compilation, the sense of accomplishment, etc. but a lot of it is just tedious grunt work.
Yes it is unsafe and you can do absurd things. But it also doesn't get in the way of just doing what you want to do.
After I finished I was puzzled, "what is the author trying to communicate to the reader here?"
As near as I can determine, enough people weren't using the author's program/utility because it was written in a language that hasn't been blessed by the crowd? It is hinted at that there might be issues involving memory consumption.
The author does not write lessons learned or share statistics of user uptake after the re-write.
No new functionality was gained, presumably this exercise was done as practice reps because the author could do it and had time.
No argument was made that the author has seen the light and now only C from this point on.
What is your killer app? What CL has to do with no one running it? What problem you had with garbage collectors? Why is C is the only option? Are you sure all those RCEs because of VMs and containers and not because it all written in C? "There are no security implications of running KC3 code" - are you sure?
Could he have jumped right into C and had amazing results, if not for the Journey learning Lisp and changing how he thought of programming.
Maybe learning Lisp is how to learn to program. Then other languages become better by virtue of how someone structures the logic.
A proper virtual machine is extremely difficult to break out of (but it can still happen [1]). Containers are a lot easier to break out of. I virtual machines were more efficient in either CPU or RAM, I would want to use them more, but it's the worst of both.
[1] https://www.zerodayinitiative.com/advisories/ZDI-23-982/
Your brain works a certain way, but you're forced to evolve into the nightmare half-done complex stacks we run these days, and it's just not the same job any more.
Just pick the right projects and the language shines.
The way he writes about his work in this article, I think he's a true master. Very impressive to see people with such passion and skill.
we also had to build a CPU from discrete bit-slice components and then program it. One of the most time intensive courses I took at CMU. Do computer engineers still have to do that?
I would certainly encourage all computer engineers, and perhaps even software engineers, to learn the "full stack".
But as to programming and C, I haven't done that in almost 30 years. It would be an interesting experiment to see how much of that skill if any I still possess.
I think it's a pretty normal pattern I've seen (and been though) of learning-oriented development rather than thoughtful engineering.
But personally, AI coding has pushed me full circle back to ruby. Who wants to mentally interpret generated C code which could have optimisations and could also have fancy looking bugs. Why would anyone want to try disambiguating those when they could just read ruby like English?
Even in ecosystems where you don't have a way to opt-out of GC, there are strategies for dramatically minimizing the impact. Just because you have some aspects of the product rely on it should not be fatal unless we are being a bit hyperbolic about the purity of tech. If your allocation rate is <1 megabyte per minute and you are expressly forcing GC to run along the grain of the allocations (i.e., after each frame or sim tick), then it is exceedingly unlikely you will experience the canonical multiple-second, stop-the-world GC boogeyman.
Additionally, I don't think this is a fair take on how an RTS game would be engineered in practice. Each unit should be part of a big contiguous array in memory somewhere. The collection of garbage can be made largely irrelevant in the scenario of managing game state if you are willing to employ approximately the same approach you'd use in many other languages. Nothing stops you from newing up a 2 gig byte array and handing out spans to consumers just because you have GC turned on.
I looked at C++, but it seems that despite being more feature-rich, it also cannot auto-generate functions/methods for working with structures?
Also returning errors with dynamically allocated strings (and freeing them) makes functions bloated.
Also Gnome infrastructure (GObject, GTK and friends) requires writing so much code that I feel sorry for people writing Gnome.
Also, how do you install dependencies in C? How do you lock a specific version (or range of versions) of a dependency with specific build options, for example?
I do not want to be rude, but C has some error-prone syntax: if you forget a *, you will be in trouble. If you do 1 byte offset error on an array in the stack, you get erratic behavior, core dump if you are lucky.
Buffer overlflows also poses security risks.
try...catch was not present on C, and it is one of the most powerful addition of C++ for code structuring.
Thread management/async programming without support from the language (which is fine, but if you see Erlang or Java, they have far more support for thread monitors).
Said that, there are very high quality library in C (pthreads, memory management and protection, lib-eventio etc) which can overcome most of its limit but... it is still error-prone
I refuse to touch anything else, but i keep an eye on the new languages that are being worked on, Zig for example
If only a very tiny fraction of the resources effort, research, time, money etc of all ML/AI funds are directed for the best design of high performance GC, it will make a software world a much better place. The fact that we have a very few books dedicated on GC design and thousands of books now dedicated on AI/ML, it is quite telling.
For real-world example and analogy, automotive industry dedicated their resources on the best design of high performance automatic transmission and now it has a faster auto transmission than manual for rally and racing. For normal driving auto is what the default and available now, most of the cars do not sell in manual transmission version.
> Linux is written in C, OpenBSD is written in C, GTK+ is object-oriented pure C, GNOME is written in C. Most of the Linux desktop apps are actually written in plain old C. So why try harder ? I know C
C is the lingua-franca of all other programming languages including Python, Julia, Rust, etc. Period.
D language has already bite the bullet and made C built-in by natively supporting it. Genius.
D language also has GC by default for more sane and intuitive programming, it's your call. It also one of the fastest compilation time and execution time languages in existence.
From the KC3 language website, "KC3 is a programming language with meta-programmation and a graph database embedded into the language."
Why you want to have a graph database embedded into the language? Just support associative array built-in since it has been proven to be the basis of all common data representations of spreadsheet, SQL, NoSQL, matrices, Graph database, etc.
[1] Associative Array Model of SQL, NoSQL, and NewSQL Databases:
https://arxiv.org/pdf/1606.05797
[2] Mathematics of Big Data: Spreadsheets, Databases, Matrices, and Graphs:
https://mitpress.mit.edu/9780262038393/mathematics-of-big-da...
Glib is c batteries included library I really like. Does anyone have any others they prefer?
yes, my life is still full of segfaults as many segfaults as ignorance and impatience. so its delightful because it helps me overcome those 2 things, slowly and steadily =)
a bit LOL, isn't it?
also the part about terraform, ansible and the other stuff.
Your work is genius! I hope KC3 can be adopted widely, there is great potential.
I wanted to do this on Linux, because I my main laptop is a Linux machine after my children confiscated my Windows laptop to play Minecraft with the only decent GPU in the house.
And I just couldn't get past the tooling. I could not get through to anything that felt like a build setup that I'd be able to replicate in my own.
On Windows, using Visual Studio, it's not that bad. It's a little annoying compared to a .NET project, and there are a lot more settings to worry about, but at the end of the day VS makes the two but very different from each other.
I actually didn't understand that until I tried to write C++ on Linux. I thought C++ on Windows was worlds different than C#. But now I've seen the light.
I honestly don't know how people do development with on Linux. Make, Cmake, all of that stuff, is so bad.
IDK, maybe someone will come along and tell me, "oh, no, do this and you'll have no problems". I hope so. But without that, what a disgusting waste of time C and C++ is on Linux.
I believe it. And I'd love to see it and hack on it, if it were open source.
This whole kc3 thing looks pretty interesting to me. I agree with the premise. It's really just another super-C that's not C++, but that's a pretty good idea for a lot of things because the C ABI is just so omnipresent.
Has it been fuzzed? Have you had someone who is very good at finding bugs in C code look at it carefully? It is understandable if the answer to one or both is "no". But we should be careful about the claims we make about code.
Related
The Liberating Experience of Common Lisp
The author critiques modern programming languages for their complexity, praising Common Lisp for its stability, unique developer experience, and creative freedom, making it preferable for software development.
C3 lang – A modern C alternative – 0.6.3 released
C3 is a programming language derived from C, emphasizing ergonomics and safety, with features like optionals, SIMD support, and a developing standard library for graphics and game frameworks.
On: Lisp, on Lisp, and MySelf
The author shares mixed feelings about Common Lisp, noting frustrations with its syntax but appreciating its flexibility. They recommend "On Lisp" and encourage experimentation and openness to new programming experiences.
A Closed and Common Lisp (2021)
The author enjoyed learning Common Lisp during the Advent of Code challenge, appreciating its unique blend of programming styles, powerful tooling, and multiple return values, despite some library limitations.
Why I Chose Common Lisp
The author switched from Clojure to Common Lisp after seven years, citing performance issues. Common Lisp met their needs for standalone executables, multithreading, and offered a supportive community and resources.