• lugal@lemmy.world
    link
    fedilink
    English
    arrow-up
    158
    arrow-down
    1
    ·
    2 months ago

    I don’t know if everyone gets the reference: RollerCoaster Tycoon is in fact writing mostly in assembly to use the hardware more efficiently

          • Fubber Nuckin'@lemmy.world
            link
            fedilink
            English
            arrow-up
            2
            ·
            2 months ago

            We created the world of monorail 1. Everything exists to bring more people to monorail 1. What is monorail 1? It is a 4 car monorail that takes the shortest possible path back to the start of the station. We have several other attractions at the park such as: The Pit; Memento Mori; Install CSS, but none of them are the main attraction.

      • Faresh@lemmy.ml
        link
        fedilink
        English
        arrow-up
        55
        arrow-down
        1
        ·
        edit-2
        2 months ago

        Writing it in assembly would make it pretty much the opposite of portable (not accounting for emulation), since you are directly giving instructions to a specific hardware and OS.

          • __dev@lemmy.world
            link
            fedilink
            English
            arrow-up
            12
            ·
            2 months ago

            That’s no less true than games written in C, or otherwise with few dependencies. Doom is way more portable than RCT precisely because it’s written in C instead of assembly.

          • Faresh@lemmy.ml
            link
            fedilink
            English
            arrow-up
            5
            ·
            edit-2
            2 months ago

            you’re not usually directly accessing/working on the hardware

            I mean, you are. Sure, there’s a layer of abstraction when doing tasks that require the intervention of the kernel, but you are still dealing with cpu registers and stuff like that. Merely by writing in assembly you are making your software less portable because you are writing for a specific ISA that only a certain family of processors can read, and talking with the kernel through an API or ABI that is specific to the kernel (standards like Posix mitigate the latter part somewhat, but some systems (windows) aren’t Posix compilant).

  • mlg@lemmy.world
    link
    fedilink
    English
    arrow-up
    4
    ·
    2 months ago

    I wanna see someone make a GPU accelerated game in assembly.

    Just throw the Vulkan and DX12 C APIs in the garbage and do it all yourself lol.

  • Valmond@lemmy.world
    link
    fedilink
    English
    arrow-up
    71
    arrow-down
    4
    ·
    2 months ago

    try writing it it in Assembly

    Small error, game crashes and takes whole PC with it burning a hole in the ground.

    • Flying Squid@lemmy.world
      link
      fedilink
      English
      arrow-up
      4
      ·
      2 months ago

      It was really easy to crash an Apple II game and get into the assembler. And my goodness am I glad I didn’t destroy my computer as a kid randomly typing things in to see what would happen.

      • Valmond@lemmy.world
        link
        fedilink
        English
        arrow-up
        3
        ·
        2 months ago

        Remember old Apple, had to use them when learning to program, there were 2 types, one with the OS on a diskette, one with a small hard drive, and they randomly showed a large bomb in the middle of the screen and you had to reload the OS. Probably the compuler that broke everything.

    • ayyy@sh.itjust.works
      link
      fedilink
      English
      arrow-up
      68
      ·
      2 months ago

      The game Roller Coaster Tycoon was famously hand written in raw CPU instructions (called assembly language). It’s only one step removed from writing literal ones and zeros. Normally computers are programmed using a human-friendly language which is then “compiled” into CPU instructions so that the humans don’t have to deal with the tedium and complication of writing CPU instructions.

        • ericbomb@lemmy.world
          link
          fedilink
          English
          arrow-up
          10
          arrow-down
          1
          ·
          2 months ago

          To send the point home even more, this is how in python you make a line of text display:

          print("Hello World")

          This is the same thing, in assembly (According to a blog I found. I can’t read this. I am not build better.)

            org  0x100        ; .com files always start 256 bytes into the segment
          
              ; int 21h is going to want...
          
              mov  dx, msg      ; the address of or message in dx
              mov  ah, 9        ; ah=9 - "print string" sub-function
              int  0x21         ; call dos services
          
              mov  ah, 0x4c     ; "terminate program" sub-function
              int  0x21         ; call dos services
          
              msg  db 'Hello, World!', 0x0d, 0x0a, '$'   ; $-terminated message
          

          But python turns that cute little line up top, into that mess at the bottom.

          I like python. Python is cute. Anyone can read python.

          • pivot_root@lemmy.world
            link
            fedilink
            English
            arrow-up
            6
            ·
            2 months ago

            That assembly is for a DOS application. It would be more verbose for a modern Linux or Win32 application and probably require a linker script.

            But python turns that cute little line up top, into that mess at the bottom.

            Technically, not quite. Python is interpreted, so it’s more like “call the print function with this string parameter” gets fed into another program, which calls it’s own functions to make it happen.

            • ericbomb@lemmy.world
              link
              fedilink
              English
              arrow-up
              1
              ·
              2 months ago

              Yeah over simplifying it a bit, and that’s funny that the stupid thing I found wasn’t even stupid enough.

              But was mostly trying to impart that we should be happy for modern languages, because for every line you write in a modern language, it’ll do a dozen things on the back end for you that in assembly you’d need to do by hand.

      • OsrsNeedsF2P@lemmy.ml
        link
        fedilink
        English
        arrow-up
        23
        ·
        edit-2
        2 months ago

        To further emphasize this, I had an assembly course in university. During my first lab, the instructor told us to add a comment explaining what every line of assembly code did, because if we didn’t, we would forget what we wrote.

        I listened to his advice, but one day I was in a rush, so I didn’t leave comments. I swear, I looked away from the computer for like 2 minutes, looked back, and had no idea what I wrote. I basically had to redo my work.

        It is not that much better than reading 1s and 0s. In fact in that course, we spent a lot of time converting 1s and 0s (by hand) to assembly and back. Got pretty good at it, would never even think of writing a game. I would literally rather create my own compiler and programming language than write a game in assembly.

        • pivot_root@lemmy.world
          link
          fedilink
          English
          arrow-up
          15
          ·
          2 months ago

          I’m probably completely insane and deranged, but I actually like assembly. With decent reverse engineering software like Ghidra, it’s not terribly difficult to understand the intent and operation of isolated functions.

          Mnemonics for the amd64 AVX extensions can go the fuck right off a bridge, though. VCVTTPS2UQQ might as well be my hands rolling across a keyboard, not a truncated conversation from packed single precision floats into packed unsigned quadword integers.

          • emergencybird@lemmy.world
            link
            fedilink
            English
            arrow-up
            9
            ·
            2 months ago

            I had a course in uni that taught us assembler on z/os. My advisor told me most students fail the course on the first try because it was so tough and my Prof for that course said if any of us managed to get at least a B in the course, he’d write us a rec letter for graduate school. That course was the most difficult and most fun I’ve ever had. I learned how to properly use registers to store my values for calculations, I learned how to use subroutines. Earned myself that B and went on to take the follow up course which was COBOL. You’re not crazy, I yearn to go back to doing low level programming, I’m mostly doing ruby for my job but I think my heart never left assembler hahaha

          • MonkderVierte@lemmy.ml
            link
            fedilink
            English
            arrow-up
            2
            ·
            edit-2
            2 months ago

            Ah yes, there was this guy in our tech school class that used to code golf in assembly. Was a crack in math and analytics too, which might explain it somewhat. Well, everyone is different i guess.

  • Wilzax@lemmy.world
    link
    fedilink
    English
    arrow-up
    143
    arrow-down
    5
    ·
    2 months ago

    Your game will actually likely be more efficient if written in C. The gcc compiler has become ridiculously optimized and probably knows more tricks than you do.

      • Wilzax@lemmy.world
        link
        fedilink
        English
        arrow-up
        31
        ·
        2 months ago

        If you’re writing sloppy C code your assembly code probably won’t work either

          • Buddahriffic@lemmy.world
            link
            fedilink
            English
            arrow-up
            1
            ·
            2 months ago

            A compiler making assumptions like that about undefined behaviour sounds just like a bug. Maybe the bug is in the spec rather than the compiler, but I can’t think of any time it would be better to optimize that code out entirely because UB is detected rather than just throwing an error or warning and otherwise ignoring the edge cases where the behaviour might break. It sounds like the worst possible option exactly for the reasons listed in that blog.

            • calcopiritus@lemmy.world
              link
              fedilink
              English
              arrow-up
              2
              ·
              2 months ago

              The thing about UB is that many optimizations are possible precisely because the spec specified it as UB. And the spec did so in order to make these optimizations possible.

              Codebases are not 6 lines long, they are hundreds of thousands. Without optimizations like those, many CPU cycles would be lost to unnecessary code being executed.

              If you write C/C++, it is because you either hate yourself or the application’s performance is important, and these optimizations are needed.

              The reason rust is so impressive nowadays is that you can write high performing code without risking accidentally doing UB. And if you are going to write code that might result in UB, you have to explicitly state so with unsafe. But for C/C++, there’s no saving. If you want your compiler to optimize code in those languages, you are going to have loaded guns pointing at your feet all the time.

          • calcopiritus@lemmy.world
            link
            fedilink
            English
            arrow-up
            2
            ·
            2 months ago

            I recently came across a rust book on how pointers aren’t just ints, because of UB.

            fn main() {
                a = &1
                b = &2
                a++
                if a == b {
                    *a = 3
                    print(b)
                }
            }
            

            This may either: not print anything, print 3 or print 2.

            Depending on the compiler, since b isn’t changed at all, it might optimize the print for print(2) instead of print(b). Even though everyone can agree that it should either not print anything or 3, but never 2.

    • dejected_warp_core@lemmy.world
      link
      fedilink
      English
      arrow-up
      46
      ·
      2 months ago

      Especially these days. Current-gen x86 architecture has all kinds of insane optimizations and special instruction sets that the Pentium I never had (e.g. SSE). You really do need a higher-level compiler at your back to make the most of it these days. And even then, there are cases where you have to resort to inline ASM or processor-specific intrinsics to optimize to the level that Roller Coaster Tycoon is/was. (original system specs)

      • KubeRoot@discuss.tchncs.de
        link
        fedilink
        English
        arrow-up
        2
        ·
        2 months ago

        I might be wrong, but doesn’t SSE require you to explicitly use it in C/C++? Laying out your data as arrays and specifically calling the SIMD operations on them?

        • dejected_warp_core@lemmy.world
          link
          fedilink
          English
          arrow-up
          1
          ·
          2 months ago

          Honestly, I’m not 100% sure. I would bet that a modern compiler would just “do the right thing” but I’ve never written code in such a high performance fashion before.

        • acockworkorange@mander.xyz
          link
          fedilink
          English
          arrow-up
          5
          ·
          2 months ago

          There’s absolutely nothing you can do in C that you can’t also do in assembly. Because assembly is just the bunch of bits that the compiler generates.

          That said, you’d have to be insane to write a game featuring SIMD instructions these days in assembly.

          • Buddahriffic@lemmy.world
            link
            fedilink
            English
            arrow-up
            4
            ·
            2 months ago

            I think they meant the other way around, that if you wanted to use it in C/C++, you’d have to either use assembly or some specific SSE construct otherwise the compiler wouldn’t bother.

            That probably was the case at one point, but I’d be surprised if it’s still the case. Though maybe that’s part of the reason why the Intel compiler can generate faster code. But I suspect it’s more of a case of better optimization by people who have a better understanding of how it works under the hood, and maybe better utilization of newer instruction set extensions.

            SSE has been around for a long time and is present in most (all?) x86 chips these days and I’d be very surprised if gcc and other popular compilers don’t use it effectively today. Some of the other extensions might be different though.

            • calcopiritus@lemmy.world
              link
              fedilink
              English
              arrow-up
              3
              ·
              2 months ago

              If you want to use instructions from an extension (for example SIMD), you either: provide 2 versions of the function, or just won’t run in some CPUs. It would be weird for someone that doesn’t know about that to compile it for x86 and then have it not run on another x86 machine. I don’t think compilers use those instructions if you don’t tell them too.

              Anyway, the SIMD the compilers will do is nowhere near the amount that it’s possible. If you manually use SIMD intrinsics/inline SIMD assembly, chances are that it will be faster than what the compiler would do. Especially because you are reducing the % of CPUs your program can run on.

            • acockworkorange@mander.xyz
              link
              fedilink
              English
              arrow-up
              2
              ·
              2 months ago

              Oh I see your point. Yeah, I think they meant that. And yes, there was a time you’d have to do trickery in C to force the use of SSE or whatever extensions you wanted to use.

          • Wilzax@lemmy.world
            link
            fedilink
            English
            arrow-up
            3
            ·
            2 months ago

            Technically assembly is a human-readable, paper-thin abstraction of the machine code. It really only implements one additional feature over raw machine code and that’s labels, which prevents you from having to rewrite jump and goto instructions EVERY TIME you refactor upstream code to have a different number of instructions.

            So not strictly the bunch of bits. But very close to it.

  • UnderpantsWeevil@lemmy.world
    link
    fedilink
    English
    arrow-up
    52
    arrow-down
    2
    ·
    2 months ago

    Step 1: Begin writing in Assembly

    Step 2: Write C

    Step 3: Use C to write C#

    Step 4: Implement Unity

    Step 5: Write your game

    Step 6: ???

    Step 7: Profit

  • einlander@lemmy.world
    link
    fedilink
    English
    arrow-up
    152
    arrow-down
    7
    ·
    2 months ago
    • Programming was never meant to be abstract so far from the hardware.
    • 640k is enough ram for everybody.
    • The come with names like rust, typescript, go, and python. Names thought up by imbeciles.
    • Dev environments, environmental variables, build and make scripts, and macros, from the minds of the utter deranged.

    They have played us for fools

    • mynameisigglepiggle@lemmy.world
      link
      fedilink
      English
      arrow-up
      16
      ·
      2 months ago

      I dabbled with making a fairly complex program for a microcontroller the other day and quickly hit the stack limit for a simple object.

      It wasn’t so much that it was a large object, but to provide flexibility I was amazed how fast I filled the memory.

      I’ve done heaps with memory managed languages in the past but shit as soon as I had to think about what I was doing under the hood everything got hard af.

      So serious question - does anyone have any good resources for a competent programmer, but with no clue whatsoever how to manage memory in a microcontroller space and avoid fragmentation etc?

      I got it to work but I’m sure I did shit job and want to be better at it.

      • BigDanishGuy@sh.itjust.works
        link
        fedilink
        English
        arrow-up
        8
        ·
        2 months ago

        The best book I’ve ever bought on programming, and the second best book I bought for a class in uni, was https://dl.acm.org/doi/book/10.5555/1824214 it may be worth checking out on libgen and buy if it suits your needs.

        Whenever I do low-level programming on the AVR architecture, I’ll make a memory map. As in I’ll map out where I’ll put what. It may not be suitable for more complex programs, but it does the job for me. And it has enabled teamwork in assembly in the past.

        If you want to work in a language that doesn’t offer memory management, but manually mapping memory isn’t feasible either, how about building your own memory management? Or perhaps use an RTOS? I’ve used freeRTOS before on various arm-based micros, and it does take a bit to get started, but after that it’s easy sailing.

        Sorry for the following tangent, all semi intelligent content in this comment is found above this line.
        BTW I tried CoOS once, I wouldn’t recommend it… OK it was 12 years ago, I can’t remember exactly what was wrong other than the documentation was crap, but I don’t need to remember why to hold a grudge.

    • CascadianGiraffe@lemmy.world
      link
      fedilink
      English
      arrow-up
      1
      ·
      2 months ago

      Adobe promised that Lingo was the future of ‘PC and internet gaming’

      Luckily by the time I had to learn to write that garbage I already coded in several other languages. Made it easier, but somehow more painful. I’m pretty sure that shit was designed so that executives could look at the code and pretend they understood what was going on. At least with ‘common terms’ it eliminated the need for commenting out most of the time. One line of code would take a paragraph of text lol.

  • Codex@lemmy.world
    link
    fedilink
    English
    arrow-up
    3
    ·
    2 months ago

    I don’t know exactly how much code reuse Sawyer had going back then, but if you’ve ever played Transport Tycoon (or more likely the open source version around today, OpenTTD) then you know that the interface and graphics are extremely similar. So it’s not like he started from scratch each game with nothing but a hot spinning disc and a magnetized needle.

    But yeah, the main reason to put up with all the modern framework bloat is the ever-ephemeral promise of being able to write your thing once and have it ported to run anywhere with minimal to no further effort.