commit ddac1a791ef47ebd855b5aa3d0d4eaa2a6d9cc8c Author: Chris Date: Wed Feb 4 20:51:17 2026 +0100 Initial commit diff --git a/.DS_Store b/.DS_Store new file mode 100644 index 0000000..2bab15a Binary files /dev/null and b/.DS_Store differ diff --git a/.gitattributes b/.gitattributes new file mode 100644 index 0000000..dfe0770 --- /dev/null +++ b/.gitattributes @@ -0,0 +1,2 @@ +# Auto detect text files and perform LF normalization +* text=auto diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..4472d59 --- /dev/null +++ b/LICENSE @@ -0,0 +1,676 @@ + GNU GENERAL PUBLIC LICENSE + Version 3, 29 June 2007 + + Copyright(C) Chris (boreddevnl) 2024-2026 + + Copyright (C) 2007 Free Software Foundation, Inc. + Everyone is permitted to copy and distribute verbatim copies + of this license document, but changing it is not allowed. + + Preamble + + The GNU General Public License is a free, copyleft license for +software and other kinds of works. + + The licenses for most software and other practical works are designed +to take away your freedom to share and change the works. By contrast, +the GNU General Public License is intended to guarantee your freedom to +share and change all versions of a program--to make sure it remains free +software for all its users. We, the Free Software Foundation, use the +GNU General Public License for most of our software; it applies also to +any other work released this way by its authors. You can apply it to +your programs, too. + + When we speak of free software, we are referring to freedom, 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 +them if you wish), that you receive source code or can get it if you +want it, that you can change the software or use pieces of it in new +free programs, and that you know you can do these things. + + To protect your rights, we need to prevent others from denying you +these rights or asking you to surrender the rights. Therefore, you have +certain responsibilities if you distribute copies of the software, or if +you modify it: responsibilities to respect the freedom of others. + + For example, if you distribute copies of such a program, whether +gratis or for a fee, you must pass on to the recipients the same +freedoms that you received. You must make sure that they, too, receive +or can get the source code. And you must show them these terms so they +know their rights. + + Developers that use the GNU GPL protect your rights with two steps: +(1) assert copyright on the software, and (2) offer you this License +giving you legal permission to copy, distribute and/or modify it. + + For the developers' and authors' protection, the GPL clearly explains +that there is no warranty for this free software. For both users' and +authors' sake, the GPL requires that modified versions be marked as +changed, so that their problems will not be attributed erroneously to +authors of previous versions. + + Some devices are designed to deny users access to install or run +modified versions of the software inside them, although the manufacturer +can do so. This is fundamentally incompatible with the aim of +protecting users' freedom to change the software. The systematic +pattern of such abuse occurs in the area of products for individuals to +use, which is precisely where it is most unacceptable. Therefore, we +have designed this version of the GPL to prohibit the practice for those +products. If such problems arise substantially in other domains, we +stand ready to extend this provision to those domains in future versions +of the GPL, as needed to protect the freedom of users. + + Finally, every program is threatened constantly by software patents. +States should not allow patents to restrict development and use of +software on general-purpose computers, but in those that do, we wish to +avoid the special danger that patents applied to a free program could +make it effectively proprietary. To prevent this, the GPL assures that +patents cannot be used to render the program non-free. + + The precise terms and conditions for copying, distribution and +modification follow. + + TERMS AND CONDITIONS + + 0. Definitions. + + "This License" refers to version 3 of the GNU General Public License. + + "Copyright" also means copyright-like laws that apply to other kinds of +works, such as semiconductor masks. + + "The Program" refers to any copyrightable work licensed under this +License. Each licensee is addressed as "you". "Licensees" and +"recipients" may be individuals or organizations. + + To "modify" a work means to copy from or adapt all or part of the work +in a fashion requiring copyright permission, other than the making of an +exact copy. The resulting work is called a "modified version" of the +earlier work or a work "based on" the earlier work. + + A "covered work" means either the unmodified Program or a work based +on the Program. + + To "propagate" a work means to do anything with it that, without +permission, would make you directly or secondarily liable for +infringement under applicable copyright law, except executing it on a +computer or modifying a private copy. Propagation includes copying, +distribution (with or without modification), making available to the +public, and in some countries other activities as well. + + To "convey" a work means any kind of propagation that enables other +parties to make or receive copies. Mere interaction with a user through +a computer network, with no transfer of a copy, is not conveying. + + An interactive user interface displays "Appropriate Legal Notices" +to the extent that it includes a convenient and prominently visible +feature that (1) displays an appropriate copyright notice, and (2) +tells the user that there is no warranty for the work (except to the +extent that warranties are provided), that licensees may convey the +work under this License, and how to view a copy of this License. If +the interface presents a list of user commands or options, such as a +menu, a prominent item in the list meets this criterion. + + 1. Source Code. + + The "source code" for a work means the preferred form of the work +for making modifications to it. "Object code" means any non-source +form of a work. + + A "Standard Interface" means an interface that either is an official +standard defined by a recognized standards body, or, in the case of +interfaces specified for a particular programming language, one that +is widely used among developers working in that language. + + The "System Libraries" of an executable work include anything, other +than the work as a whole, that (a) is included in the normal form of +packaging a Major Component, but which is not part of that Major +Component, and (b) serves only to enable use of the work with that +Major Component, or to implement a Standard Interface for which an +implementation is available to the public in source code form. A +"Major Component", in this context, means a major essential component +(kernel, window system, and so on) of the specific operating system +(if any) on which the executable work runs, or a compiler used to +produce the work, or an object code interpreter used to run it. + + The "Corresponding Source" for a work in object code form means all +the source code needed to generate, install, and (for an executable +work) run the object code and to modify the work, including scripts to +control those activities. However, it does not include the work's +System Libraries, or general-purpose tools or generally available free +programs which are used unmodified in performing those activities but +which are not part of the work. For example, Corresponding Source +includes interface definition files associated with source files for +the work, and the source code for shared libraries and dynamically +linked subprograms that the work is specifically designed to require, +such as by intimate data communication or control flow between those +subprograms and other parts of the work. + + The Corresponding Source need not include anything that users +can regenerate automatically from other parts of the Corresponding +Source. + + The Corresponding Source for a work in source code form is that +same work. + + 2. Basic Permissions. + + All rights granted under this License are granted for the term of +copyright on the Program, and are irrevocable provided the stated +conditions are met. This License explicitly affirms your unlimited +permission to run the unmodified Program. The output from running a +covered work is covered by this License only if the output, given its +content, constitutes a covered work. This License acknowledges your +rights of fair use or other equivalent, as provided by copyright law. + + You may make, run and propagate covered works that you do not +convey, without conditions so long as your license otherwise remains +in force. You may convey covered works to others for the sole purpose +of having them make modifications exclusively for you, or provide you +with facilities for running those works, provided that you comply with +the terms of this License in conveying all material for which you do +not control copyright. Those thus making or running the covered works +for you must do so exclusively on your behalf, under your direction +and control, on terms that prohibit them from making any copies of +your copyrighted material outside their relationship with you. + + Conveying under any other circumstances is permitted solely under +the conditions stated below. Sublicensing is not allowed; section 10 +makes it unnecessary. + + 3. Protecting Users' Legal Rights From Anti-Circumvention Law. + + No covered work shall be deemed part of an effective technological +measure under any applicable law fulfilling obligations under article +11 of the WIPO copyright treaty adopted on 20 December 1996, or +similar laws prohibiting or restricting circumvention of such +measures. + + When you convey a covered work, you waive any legal power to forbid +circumvention of technological measures to the extent such circumvention +is effected by exercising rights under this License with respect to +the covered work, and you disclaim any intention to limit operation or +modification of the work as a means of enforcing, against the work's +users, your or third parties' legal rights to forbid circumvention of +technological measures. + + 4. Conveying Verbatim Copies. + + You may convey verbatim copies of the Program's source code as you +receive it, in any medium, provided that you conspicuously and +appropriately publish on each copy an appropriate copyright notice; +keep intact all notices stating that this License and any +non-permissive terms added in accord with section 7 apply to the code; +keep intact all notices of the absence of any warranty; and give all +recipients a copy of this License along with the Program. + + You may charge any price or no price for each copy that you convey, +and you may offer support or warranty protection for a fee. + + 5. Conveying Modified Source Versions. + + You may convey a work based on the Program, or the modifications to +produce it from the Program, in the form of source code under the +terms of section 4, provided that you also meet all of these conditions: + + a) The work must carry prominent notices stating that you modified + it, and giving a relevant date. + + b) The work must carry prominent notices stating that it is + released under this License and any conditions added under section + 7. This requirement modifies the requirement in section 4 to + "keep intact all notices". + + c) You must license the entire work, as a whole, under this + License to anyone who comes into possession of a copy. This + License will therefore apply, along with any applicable section 7 + additional terms, to the whole of the work, and all its parts, + regardless of how they are packaged. This License gives no + permission to license the work in any other way, but it does not + invalidate such permission if you have separately received it. + + d) If the work has interactive user interfaces, each must display + Appropriate Legal Notices; however, if the Program has interactive + interfaces that do not display Appropriate Legal Notices, your + work need not make them do so. + + A compilation of a covered work with other separate and independent +works, which are not by their nature extensions of the covered work, +and which are not combined with it such as to form a larger program, +in or on a volume of a storage or distribution medium, is called an +"aggregate" if the compilation and its resulting copyright are not +used to limit the access or legal rights of the compilation's users +beyond what the individual works permit. Inclusion of a covered work +in an aggregate does not cause this License to apply to the other +parts of the aggregate. + + 6. Conveying Non-Source Forms. + + You may convey a covered work in object code form under the terms +of sections 4 and 5, provided that you also convey the +machine-readable Corresponding Source under the terms of this License, +in one of these ways: + + a) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by the + Corresponding Source fixed on a durable physical medium + customarily used for software interchange. + + b) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by a + written offer, valid for at least three years and valid for as + long as you offer spare parts or customer support for that product + model, to give anyone who possesses the object code either (1) a + copy of the Corresponding Source for all the software in the + product that is covered by this License, on a durable physical + medium customarily used for software interchange, for a price no + more than your reasonable cost of physically performing this + conveying of source, or (2) access to copy the + Corresponding Source from a network server at no charge. + + c) Convey individual copies of the object code with a copy of the + written offer to provide the Corresponding Source. This + alternative is allowed only occasionally and noncommercially, and + only if you received the object code with such an offer, in accord + with subsection 6b. + + d) Convey the object code by offering access from a designated + place (gratis or for a charge), and offer equivalent access to the + Corresponding Source in the same way through the same place at no + further charge. You need not require recipients to copy the + Corresponding Source along with the object code. If the place to + copy the object code is a network server, the Corresponding Source + may be on a different server (operated by you or a third party) + that supports equivalent copying facilities, provided you maintain + clear directions next to the object code saying where to find the + Corresponding Source. Regardless of what server hosts the + Corresponding Source, you remain obligated to ensure that it is + available for as long as needed to satisfy these requirements. + + e) Convey the object code using peer-to-peer transmission, provided + you inform other peers where the object code and Corresponding + Source of the work are being offered to the general public at no + charge under subsection 6d. + + A separable portion of the object code, whose source code is excluded +from the Corresponding Source as a System Library, need not be +included in conveying the object code work. + + A "User Product" is either (1) a "consumer product", which means any +tangible personal property which is normally used for personal, family, +or household purposes, or (2) anything designed or sold for incorporation +into a dwelling. In determining whether a product is a consumer product, +doubtful cases shall be resolved in favor of coverage. For a particular +product received by a particular user, "normally used" refers to a +typical or common use of that class of product, regardless of the status +of the particular user or of the way in which the particular user +actually uses, or expects or is expected to use, the product. A product +is a consumer product regardless of whether the product has substantial +commercial, industrial or non-consumer uses, unless such uses represent +the only significant mode of use of the product. + + "Installation Information" for a User Product means any methods, +procedures, authorization keys, or other information required to install +and execute modified versions of a covered work in that User Product from +a modified version of its Corresponding Source. The information must +suffice to ensure that the continued functioning of the modified object +code is in no case prevented or interfered with solely because +modification has been made. + + If you convey an object code work under this section in, or with, or +specifically for use in, a User Product, and the conveying occurs as +part of a transaction in which the right of possession and use of the +User Product is transferred to the recipient in perpetuity or for a +fixed term (regardless of how the transaction is characterized), the +Corresponding Source conveyed under this section must be accompanied +by the Installation Information. But this requirement does not apply +if neither you nor any third party retains the ability to install +modified object code on the User Product (for example, the work has +been installed in ROM). + + The requirement to provide Installation Information does not include a +requirement to continue to provide support service, warranty, or updates +for a work that has been modified or installed by the recipient, or for +the User Product in which it has been modified or installed. Access to a +network may be denied when the modification itself materially and +adversely affects the operation of the network or violates the rules and +protocols for communication across the network. + + Corresponding Source conveyed, and Installation Information provided, +in accord with this section must be in a format that is publicly +documented (and with an implementation available to the public in +source code form), and must require no special password or key for +unpacking, reading or copying. + + 7. Additional Terms. + + "Additional permissions" are terms that supplement the terms of this +License by making exceptions from one or more of its conditions. +Additional permissions that are applicable to the entire Program shall +be treated as though they were included in this License, to the extent +that they are valid under applicable law. If additional permissions +apply only to part of the Program, that part may be used separately +under those permissions, but the entire Program remains governed by +this License without regard to the additional permissions. + + When you convey a copy of a covered work, you may at your option +remove any additional permissions from that copy, or from any part of +it. (Additional permissions may be written to require their own +removal in certain cases when you modify the work.) You may place +additional permissions on material, added by you to a covered work, +for which you have or can give appropriate copyright permission. + + Notwithstanding any other provision of this License, for material you +add to a covered work, you may (if authorized by the copyright holders of +that material) supplement the terms of this License with terms: + + a) Disclaiming warranty or limiting liability differently from the + terms of sections 15 and 16 of this License; or + + b) Requiring preservation of specified reasonable legal notices or + author attributions in that material or in the Appropriate Legal + Notices displayed by works containing it; or + + c) Prohibiting misrepresentation of the origin of that material, or + requiring that modified versions of such material be marked in + reasonable ways as different from the original version; or + + d) Limiting the use for publicity purposes of names of licensors or + authors of the material; or + + e) Declining to grant rights under trademark law for use of some + trade names, trademarks, or service marks; or + + f) Requiring indemnification of licensors and authors of that + material by anyone who conveys the material (or modified versions of + it) with contractual assumptions of liability to the recipient, for + any liability that these contractual assumptions directly impose on + those licensors and authors. + + All other non-permissive additional terms are considered "further +restrictions" within the meaning of section 10. If the Program as you +received it, or any part of it, contains a notice stating that it is +governed by this License along with a term that is a further +restriction, you may remove that term. If a license document contains +a further restriction but permits relicensing or conveying under this +License, you may add to a covered work material governed by the terms +of that license document, provided that the further restriction does +not survive such relicensing or conveying. + + If you add terms to a covered work in accord with this section, you +must place, in the relevant source files, a statement of the +additional terms that apply to those files, or a notice indicating +where to find the applicable terms. + + Additional terms, permissive or non-permissive, may be stated in the +form of a separately written license, or stated as exceptions; +the above requirements apply either way. + + 8. Termination. + + You may not propagate or modify a covered work except as expressly +provided under this License. Any attempt otherwise to propagate or +modify it is void, and will automatically terminate your rights under +this License (including any patent licenses granted under the third +paragraph of section 11). + + However, if you cease all violation of this License, then your +license from a particular copyright holder is reinstated (a) +provisionally, unless and until the copyright holder explicitly and +finally terminates your license, and (b) permanently, if the copyright +holder fails to notify you of the violation by some reasonable means +prior to 60 days after the cessation. + + Moreover, your license from a particular copyright holder is +reinstated permanently if the copyright holder notifies you of the +violation by some reasonable means, this is the first time you have +received notice of violation of this License (for any work) from that +copyright holder, and you cure the violation prior to 30 days after +your receipt of the notice. + + Termination of your rights under this section does not terminate the +licenses of parties who have received copies or rights from you under +this License. If your rights have been terminated and not permanently +reinstated, you do not qualify to receive new licenses for the same +material under section 10. + + 9. Acceptance Not Required for Having Copies. + + You are not required to accept this License in order to receive or +run a copy of the Program. Ancillary propagation of a covered work +occurring solely as a consequence of using peer-to-peer transmission +to receive a copy likewise does not require acceptance. However, +nothing other than this License grants you permission to propagate or +modify any covered work. These actions infringe copyright if you do +not accept this License. Therefore, by modifying or propagating a +covered work, you indicate your acceptance of this License to do so. + + 10. Automatic Licensing of Downstream Recipients. + + Each time you convey a covered work, the recipient automatically +receives a license from the original licensors, to run, modify and +propagate that work, subject to this License. You are not responsible +for enforcing compliance by third parties with this License. + + An "entity transaction" is a transaction transferring control of an +organization, or substantially all assets of one, or subdividing an +organization, or merging organizations. If propagation of a covered +work results from an entity transaction, each party to that +transaction who receives a copy of the work also receives whatever +licenses to the work the party's predecessor in interest had or could +give under the previous paragraph, plus a right to possession of the +Corresponding Source of the work from the predecessor in interest, if +the predecessor has it or can get it with reasonable efforts. + + You may not impose any further restrictions on the exercise of the +rights granted or affirmed under this License. For example, you may +not impose a license fee, royalty, or other charge for exercise of +rights granted under this License, and you may not initiate litigation +(including a cross-claim or counterclaim in a lawsuit) alleging that +any patent claim is infringed by making, using, selling, offering for +sale, or importing the Program or any portion of it. + + 11. Patents. + + A "contributor" is a copyright holder who authorizes use under this +License of the Program or a work on which the Program is based. The +work thus licensed is called the contributor's "contributor version". + + A contributor's "essential patent claims" are all patent claims +owned or controlled by the contributor, whether already acquired or +hereafter acquired, that would be infringed by some manner, permitted +by this License, of making, using, or selling its contributor version, +but do not include claims that would be infringed only as a +consequence of further modification of the contributor version. For +purposes of this definition, "control" includes the right to grant +patent sublicenses in a manner consistent with the requirements of +this License. + + Each contributor grants you a non-exclusive, worldwide, royalty-free +patent license under the contributor's essential patent claims, to +make, use, sell, offer for sale, import and otherwise run, modify and +propagate the contents of its contributor version. + + In the following three paragraphs, a "patent license" is any express +agreement or commitment, however denominated, not to enforce a patent +(such as an express permission to practice a patent or covenant not to +sue for patent infringement). To "grant" such a patent license to a +party means to make such an agreement or commitment not to enforce a +patent against the party. + + If you convey a covered work, knowingly relying on a patent license, +and the Corresponding Source of the work is not available for anyone +to copy, free of charge and under the terms of this License, through a +publicly available network server or other readily accessible means, +then you must either (1) cause the Corresponding Source to be so +available, or (2) arrange to deprive yourself of the benefit of the +patent license for this particular work, or (3) arrange, in a manner +consistent with the requirements of this License, to extend the patent +license to downstream recipients. "Knowingly relying" means you have +actual knowledge that, but for the patent license, your conveying the +covered work in a country, or your recipient's use of the covered work +in a country, would infringe one or more identifiable patents in that +country that you have reason to believe are valid. + + If, pursuant to or in connection with a single transaction or +arrangement, you convey, or propagate by procuring conveyance of, a +covered work, and grant a patent license to some of the parties +receiving the covered work authorizing them to use, propagate, modify +or convey a specific copy of the covered work, then the patent license +you grant is automatically extended to all recipients of the covered +work and works based on it. + + A patent license is "discriminatory" if it does not include within +the scope of its coverage, prohibits the exercise of, or is +conditioned on the non-exercise of one or more of the rights that are +specifically granted under this License. You may not convey a covered +work if you are a party to an arrangement with a third party that is +in the business of distributing software, under which you make payment +to the third party based on the extent of your activity of conveying +the work, and under which the third party grants, to any of the +parties who would receive the covered work from you, a discriminatory +patent license (a) in connection with copies of the covered work +conveyed by you (or copies made from those copies), or (b) primarily +for and in connection with specific products or compilations that +contain the covered work, unless you entered into that arrangement, +or that patent license was granted, prior to 28 March 2007. + + Nothing in this License shall be construed as excluding or limiting +any implied license or other defenses to infringement that may +otherwise be available to you under applicable patent law. + + 12. No Surrender of Others' Freedom. + + If 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 convey a +covered work so as to satisfy simultaneously your obligations under this +License and any other pertinent obligations, then as a consequence you may +not convey it at all. For example, if you agree to terms that obligate you +to collect a royalty for further conveying from those to whom you convey +the Program, the only way you could satisfy both those terms and this +License would be to refrain entirely from conveying the Program. + + 13. Use with the GNU Affero General Public License. + + Notwithstanding any other provision of this License, you have +permission to link or combine any covered work with a work licensed +under version 3 of the GNU Affero General Public License into a single +combined work, and to convey the resulting work. The terms of this +License will continue to apply to the part which is the covered work, +but the special requirements of the GNU Affero General Public License, +section 13, concerning interaction through a network will apply to the +combination as such. + + 14. Revised Versions of this License. + + The Free Software Foundation may publish revised and/or new versions of +the GNU 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 +Program specifies that a certain numbered version of the GNU General +Public License "or any later version" applies to it, you have the +option of following the terms and conditions either of that numbered +version or of any later version published by the Free Software +Foundation. If the Program does not specify a version number of the +GNU General Public License, you may choose any version ever published +by the Free Software Foundation. + + If the Program specifies that a proxy can decide which future +versions of the GNU General Public License can be used, that proxy's +public statement of acceptance of a version permanently authorizes you +to choose that version for the Program. + + Later license versions may give you additional or different +permissions. However, no additional obligations are imposed on any +author or copyright holder as a result of your choosing to follow a +later version. + + 15. Disclaimer of Warranty. + + THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY +APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT +HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "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 PROGRAM +IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF +ALL NECESSARY SERVICING, REPAIR OR CORRECTION. + + 16. Limitation of Liability. + + IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING +WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS +THE PROGRAM 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 PROGRAM (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 PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), +EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF +SUCH DAMAGES. + + 17. Interpretation of Sections 15 and 16. + + If the disclaimer of warranty and limitation of liability provided +above cannot be given local legal effect according to their terms, +reviewing courts shall apply local law that most closely approximates +an absolute waiver of all civil liability in connection with the +Program, unless a warranty or assumption of liability accompanies a +copy of the Program in return for a fee. + + END OF TERMS AND CONDITIONS + + How to Apply These Terms to Your New Programs + + If you develop a new program, and you want it to be of the greatest +possible use to the public, the best way to achieve this is to make it +free software which everyone can redistribute and change under these terms. + + To do so, attach the following notices to the program. It is safest +to attach them to the start of each source file to most effectively +state the exclusion of warranty; and each file should have at least +the "copyright" line and a pointer to where the full notice is found. + + + Copyright (C) + + 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 3 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. If not, see . + +Also add information on how to contact you by electronic and paper mail. + + If the program does terminal interaction, make it output a short +notice like this when it starts in an interactive mode: + + Copyright (C) + This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. + This is free software, and you are welcome to redistribute it + under certain conditions; type `show c' for details. + +The hypothetical commands `show w' and `show c' should show the appropriate +parts of the General Public License. Of course, your program's commands +might be different; for a GUI interface, you would use an "about box". + + You should also get your employer (if you work as a programmer) or school, +if any, to sign a "copyright disclaimer" for the program, if necessary. +For more information on this, and how to apply and follow the GNU GPL, see +. + + The GNU General Public License does not permit incorporating your program +into proprietary programs. If your program is a subroutine library, you +may consider it more useful to permit linking proprietary applications with +the library. If this is what you want to do, use the GNU Lesser General +Public License instead of this License. But first, please read +. diff --git a/Makefile b/Makefile new file mode 100644 index 0000000..f4e8aa0 --- /dev/null +++ b/Makefile @@ -0,0 +1,122 @@ +# BrewOS Makefile +# Target Architecture: x86_64 +# Host: macOS + +# Toolchain Definitions +CC = x86_64-elf-gcc +LD = x86_64-elf-ld +NASM = nasm +XORRISO = xorriso + +# Directories +SRC_DIR = src/kernel +BUILD_DIR = build +ISO_DIR = iso_root + +# Output +KERNEL_ELF = $(BUILD_DIR)/brewos.elf +ISO_IMAGE = brewos.iso + +# Source Files +C_SOURCES = $(wildcard $(SRC_DIR)/*.c) +CLI_APP_SOURCES = $(wildcard $(SRC_DIR)/cli_apps/*.c) +ASM_SOURCES = $(wildcard $(SRC_DIR)/*.asm) +OBJ_FILES = $(patsubst $(SRC_DIR)/%.c, $(BUILD_DIR)/%.o, $(C_SOURCES)) \ + $(patsubst $(SRC_DIR)/cli_apps/%.c, $(BUILD_DIR)/cli_apps/%.o, $(CLI_APP_SOURCES)) \ + $(patsubst $(SRC_DIR)/%.asm, $(BUILD_DIR)/%.o, $(ASM_SOURCES)) + +# Flags +CFLAGS = -g -O2 -pipe -Wall -Wextra -std=gnu11 -ffreestanding \ + -fno-stack-protector -fno-stack-check -fno-lto -fPIE \ + -m64 -march=x86-64 -mno-80387 -mno-mmx -mno-sse -mno-sse2 -mno-red-zone \ + -I$(SRC_DIR) -I$(SRC_DIR)/cli_apps + +LDFLAGS = -m elf_x86_64 -nostdlib -static -pie --no-dynamic-linker \ + -z text -z max-page-size=0x1000 -T linker.ld + +NASMFLAGS = -f elf64 + +# Limine Version +LIMINE_VERSION = 7.0.0 +LIMINE_URL_BASE = https://github.com/limine-bootloader/limine/raw/v$(LIMINE_VERSION) + +.PHONY: all clean run limine-setup + +all: $(ISO_IMAGE) + +# Ensure build directories exist +$(BUILD_DIR): + mkdir -p $(BUILD_DIR) + mkdir -p $(BUILD_DIR)/cli_apps + +# Download Limine Binaries via Git +limine-setup: + @if [ ! -f limine/limine-bios.sys ]; then \ + echo "Limine binaries missing or invalid. Cloning v$(LIMINE_VERSION)-binary..."; \ + rm -rf limine; \ + git clone https://github.com/limine-bootloader/limine.git --branch=v$(LIMINE_VERSION)-binary --depth=1 limine; \ + fi + @if [ ! -f $(SRC_DIR)/limine.h ]; then \ + echo "Copying limine.h..."; \ + cp limine/limine.h $(SRC_DIR)/limine.h; \ + fi + @echo "Building Limine host utility..."; \ + $(MAKE) -C limine + +# Compile C Sources +$(BUILD_DIR)/%.o: $(SRC_DIR)/%.c | $(BUILD_DIR) limine-setup + $(CC) $(CFLAGS) -c $< -o $@ + +# Compile CLI Apps C Sources +$(BUILD_DIR)/cli_apps/%.o: $(SRC_DIR)/cli_apps/%.c | $(BUILD_DIR) limine-setup + $(CC) $(CFLAGS) -c $< -o $@ + +# Assemble ASM Sources +$(BUILD_DIR)/%.o: $(SRC_DIR)/%.asm | $(BUILD_DIR) + $(NASM) $(NASMFLAGS) $< -o $@ + +# Link Kernel +$(KERNEL_ELF): $(OBJ_FILES) + $(LD) $(LDFLAGS) -o $@ $(OBJ_FILES) + +# Create ISO +$(ISO_IMAGE): $(KERNEL_ELF) limine.cfg limine-setup + rm -rf $(ISO_DIR) + mkdir -p $(ISO_DIR) + mkdir -p $(ISO_DIR)/EFI/BOOT + + # Copy Kernel and Config + cp $(KERNEL_ELF) $(ISO_DIR)/ + cp limine.cfg $(ISO_DIR)/ + + # Copy README + cp README.md $(ISO_DIR)/ + + # Copy Wallpaper (if it exists) + @if [ -f src/kernel/wallpaper.ppm ]; then cp src/kernel/wallpaper.ppm $(ISO_DIR)/; fi + + # Copy Limine Bootloader Files (flat structure in binary branch) + cp limine/limine-bios.sys $(ISO_DIR)/ + cp limine/limine-bios-cd.bin $(ISO_DIR)/ + cp limine/limine-uefi-cd.bin $(ISO_DIR)/ + + # Create EFI Boot Files + cp limine/BOOTX64.EFI $(ISO_DIR)/EFI/BOOT/ + cp limine/BOOTIA32.EFI $(ISO_DIR)/EFI/BOOT/ + + # Generate ISO + $(XORRISO) -as mkisofs -b limine-bios-cd.bin \ + -no-emul-boot -boot-load-size 4 -boot-info-table \ + --efi-boot limine-uefi-cd.bin \ + -efi-boot-part --efi-boot-image --protective-msdos-label \ + $(ISO_DIR) -o $(ISO_IMAGE) + + # Install Limine to ISO (for BIOS boot) + ./limine/limine bios-install $(ISO_IMAGE) + +clean: + rm -rf $(BUILD_DIR) $(ISO_DIR) $(ISO_IMAGE) + +run: $(ISO_IMAGE) + qemu-system-x86_64 -m 2G -serial stdio -cdrom $(ISO_IMAGE) -boot d \ + -audiodev coreaudio,id=audio0 -machine pcspk-audiodev=audio0 diff --git a/README.md b/README.md new file mode 100644 index 0000000..916502b --- /dev/null +++ b/README.md @@ -0,0 +1,148 @@ +# Brew OS 1.01 Alpha + +## Brewkernel is now BrewOS! +Brewkernel will from now on be deprecated as it's core became too messy. I have built a less bloated kernel and wrote a DE above it, which is why it is now an OS instead of a kernel (in my opinion). + + +
+Brew Kernel is a simple x86_64 hobbyist operating system. +It features a DE (and WM), a FAT32 filesystem, customizable UI and much much more! + +## Features +- Brew WM +- Fat 32 FS +- 64-bit long mode support +- Multiboot2 compliant +- Text editor +- IDT +- Ability to run on actual x86_64 hardware +- CLI + +## Prerequisites + +To build BrewOS, you'll need the following tools installed: + +- **x86_64 ELF Toolchain**: `x86_64-elf-gcc`, `x86_64-elf-ld` +- **NASM**: Netwide Assembler for compiling assembly code +- **xorriso**: For creating bootable ISO images +- **QEMU** (optional): For testing the kernel in an emulator + +On macOS, you can install these using Homebrew: +```sh +brew install x86_64-elf-binutils x86_64-elf-gcc nasm xorriso qemu +``` + +## Building + +Simply run `make` from the project root: + +```sh +make +``` + +This will: +1. Compile all kernel C sources and assembly files +2. Link the kernel ELF binary +3. Generate a bootable ISO image (`brewos.iso`) + +The build output is organized as follows: +- Compiled object files: `build/` +- ISO root filesystem: `iso_root/` +- Final ISO image: `brewos.iso` + +## Running + +### QEMU Emulation + +Run the kernel in QEMU: + +```sh +make run +``` + +Or manually: +```sh +qemu-system-x86_64 -m 2G -serial stdio -cdrom brewos.iso -boot d +``` + +### Running on Real Hardware + +*Warning: This is at YOUR OWN RISK. This software comes with ZERO warranty and may break your system.* + +1. **Create bootable USB**: Use [Balena Etcher](https://www.balena.io/etcher/) to flash `brewos.iso` to a USB drive + +2. **Prepare the system**: + - Enable legacy (BIOS) boot in your system BIOS/UEFI settings + - Disable Secure Boot if needed + +3. **Boot**: Insert the USB drive and select it in the boot menu during startup + +4. **Tested Hardware**: + - HP EliteDesk 705 G4 DM (AMD Ryzen 5 PRO 2400G, Radeon Vega) + - Lenovo ThinkPad A475 20KL002VMH (AMD Pro A12-8830B, Radeon R7) + + +## Project Structure + +- `src/kernel/` - Main kernel implementation + - `boot.asm` - Boot assembly code + - `main.c` - Kernel entry point + - `*.c / *.h` - Core kernel modules (graphics, interrupts, filesystem, etc.) + - `cli_apps/` - Command-line applications + - `wallpaper.ppm` - Default desktop wallpaper +- `build/` - Compiled object files (generated during build) +- `iso_root/` - ISO filesystem layout (generated during build) +- `limine/` - Limine bootloader files (downloaded automatically) +- `linker.ld` - Linker script for x86_64 ELF +- `limine.cfg` - Limine bootloader configuration +- `Makefile` - Build configuration and targets + + + + +### +### + +

Help me brew some coffee! ☕️

+ +### + +

+ If you enjoy this project, and like what i'm doing here, consider buying me a coffee! +

+ + Buy Me A Coffee + +

+ +### + + +## License + +Copyright (C) 2024-2026 boreddevnl + +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 3 of the License, or (at your option) any later version. + +NOTICE +------ + +This product includes software developed by Chris ("boreddevnl") as part of the BrewKernel project. + +Copyright (C) 2024–2026 Chris / boreddevnl (previously boreddevhq) + +All source files in this repository contain copyright and license +headers that must be preserved in redistributions and derivative works. + +If you distribute or modify this project (in whole or in part), +you MUST: + + - Retain all copyright and license headers at the top of each file. + - Include this NOTICE file along with any redistributions or + derivative works. + - Provide clear attribution to the original author in documentation + or credits where appropriate. + +The above attribution requirements are informational and intended to +ensure proper credit is given. They do not alter or supersede the +terms of the GNU General Public License (GPL), which governs this work. diff --git a/asciiart.png b/asciiart.png new file mode 100644 index 0000000..427d554 Binary files /dev/null and b/asciiart.png differ diff --git a/brewos.iso b/brewos.iso new file mode 100644 index 0000000..9b1c660 Binary files /dev/null and b/brewos.iso differ diff --git a/build/about.o b/build/about.o new file mode 100644 index 0000000..a2970ea Binary files /dev/null and b/build/about.o differ diff --git a/build/boot.o b/build/boot.o new file mode 100644 index 0000000..2d5e0e8 Binary files /dev/null and b/build/boot.o differ diff --git a/build/brewos.elf b/build/brewos.elf new file mode 100755 index 0000000..89a352a Binary files /dev/null and b/build/brewos.elf differ diff --git a/build/calculator.o b/build/calculator.o new file mode 100644 index 0000000..40330b3 Binary files /dev/null and b/build/calculator.o differ diff --git a/build/cli_apps/about.o b/build/cli_apps/about.o new file mode 100644 index 0000000..8bd3096 Binary files /dev/null and b/build/cli_apps/about.o differ diff --git a/build/cli_apps/beep.o b/build/cli_apps/beep.o new file mode 100644 index 0000000..27522fa Binary files /dev/null and b/build/cli_apps/beep.o differ diff --git a/build/cli_apps/blind.o b/build/cli_apps/blind.o new file mode 100644 index 0000000..0b3e866 Binary files /dev/null and b/build/cli_apps/blind.o differ diff --git a/build/cli_apps/clear.o b/build/cli_apps/clear.o new file mode 100644 index 0000000..ff6ff00 Binary files /dev/null and b/build/cli_apps/clear.o differ diff --git a/build/cli_apps/cli_utils.o b/build/cli_apps/cli_utils.o new file mode 100644 index 0000000..eb10a93 Binary files /dev/null and b/build/cli_apps/cli_utils.o differ diff --git a/build/cli_apps/cowsay.o b/build/cli_apps/cowsay.o new file mode 100644 index 0000000..9f174a8 Binary files /dev/null and b/build/cli_apps/cowsay.o differ diff --git a/build/cli_apps/date.o b/build/cli_apps/date.o new file mode 100644 index 0000000..58b1106 Binary files /dev/null and b/build/cli_apps/date.o differ diff --git a/build/cli_apps/exit.o b/build/cli_apps/exit.o new file mode 100644 index 0000000..64d31fc Binary files /dev/null and b/build/cli_apps/exit.o differ diff --git a/build/cli_apps/fs_commands.o b/build/cli_apps/fs_commands.o new file mode 100644 index 0000000..beeec0e Binary files /dev/null and b/build/cli_apps/fs_commands.o differ diff --git a/build/cli_apps/help.o b/build/cli_apps/help.o new file mode 100644 index 0000000..1a9d226 Binary files /dev/null and b/build/cli_apps/help.o differ diff --git a/build/cli_apps/license.o b/build/cli_apps/license.o new file mode 100644 index 0000000..2c9dad6 Binary files /dev/null and b/build/cli_apps/license.o differ diff --git a/build/cli_apps/man.o b/build/cli_apps/man.o new file mode 100644 index 0000000..3470d6e Binary files /dev/null and b/build/cli_apps/man.o differ diff --git a/build/cli_apps/math.o b/build/cli_apps/math.o new file mode 100644 index 0000000..b0e6922 Binary files /dev/null and b/build/cli_apps/math.o differ diff --git a/build/cli_apps/memcmd.o b/build/cli_apps/memcmd.o new file mode 100644 index 0000000..d3f1401 Binary files /dev/null and b/build/cli_apps/memcmd.o differ diff --git a/build/cli_apps/meminfo.o b/build/cli_apps/meminfo.o new file mode 100644 index 0000000..ce739e1 Binary files /dev/null and b/build/cli_apps/meminfo.o differ diff --git a/build/cli_apps/readtheman.o b/build/cli_apps/readtheman.o new file mode 100644 index 0000000..320bef1 Binary files /dev/null and b/build/cli_apps/readtheman.o differ diff --git a/build/cli_apps/reboot.o b/build/cli_apps/reboot.o new file mode 100644 index 0000000..2734c35 Binary files /dev/null and b/build/cli_apps/reboot.o differ diff --git a/build/cli_apps/shutdown.o b/build/cli_apps/shutdown.o new file mode 100644 index 0000000..16fa3ce Binary files /dev/null and b/build/cli_apps/shutdown.o differ diff --git a/build/cli_apps/txtedit.o b/build/cli_apps/txtedit.o new file mode 100644 index 0000000..acf204e Binary files /dev/null and b/build/cli_apps/txtedit.o differ diff --git a/build/cli_apps/uptime.o b/build/cli_apps/uptime.o new file mode 100644 index 0000000..e155568 Binary files /dev/null and b/build/cli_apps/uptime.o differ diff --git a/build/cmd.o b/build/cmd.o new file mode 100644 index 0000000..145d237 Binary files /dev/null and b/build/cmd.o differ diff --git a/build/control_panel.o b/build/control_panel.o new file mode 100644 index 0000000..577a8e9 Binary files /dev/null and b/build/control_panel.o differ diff --git a/build/editor.o b/build/editor.o new file mode 100644 index 0000000..fd9ca9c Binary files /dev/null and b/build/editor.o differ diff --git a/build/explorer.o b/build/explorer.o new file mode 100644 index 0000000..6979cbb Binary files /dev/null and b/build/explorer.o differ diff --git a/build/fat32.o b/build/fat32.o new file mode 100644 index 0000000..061ad77 Binary files /dev/null and b/build/fat32.o differ diff --git a/build/graphics.o b/build/graphics.o new file mode 100644 index 0000000..d78a9de Binary files /dev/null and b/build/graphics.o differ diff --git a/build/idt.o b/build/idt.o new file mode 100644 index 0000000..fe1073b Binary files /dev/null and b/build/idt.o differ diff --git a/build/interrupts.o b/build/interrupts.o new file mode 100644 index 0000000..1432665 Binary files /dev/null and b/build/interrupts.o differ diff --git a/build/licensewr.o b/build/licensewr.o new file mode 100644 index 0000000..48e9b82 Binary files /dev/null and b/build/licensewr.o differ diff --git a/build/main.o b/build/main.o new file mode 100644 index 0000000..8abd45d Binary files /dev/null and b/build/main.o differ diff --git a/build/memory_manager.o b/build/memory_manager.o new file mode 100644 index 0000000..f6354a2 Binary files /dev/null and b/build/memory_manager.o differ diff --git a/build/minesweeper.o b/build/minesweeper.o new file mode 100644 index 0000000..06b6d90 Binary files /dev/null and b/build/minesweeper.o differ diff --git a/build/notepad.o b/build/notepad.o new file mode 100644 index 0000000..f9822fa Binary files /dev/null and b/build/notepad.o differ diff --git a/build/ps2.o b/build/ps2.o new file mode 100644 index 0000000..8ea15fd Binary files /dev/null and b/build/ps2.o differ diff --git a/build/rtc.o b/build/rtc.o new file mode 100644 index 0000000..88921b4 Binary files /dev/null and b/build/rtc.o differ diff --git a/build/wm.o b/build/wm.o new file mode 100644 index 0000000..5035cff Binary files /dev/null and b/build/wm.o differ diff --git a/iso_root/EFI/BOOT/BOOTIA32.EFI b/iso_root/EFI/BOOT/BOOTIA32.EFI new file mode 100644 index 0000000..27d59d4 Binary files /dev/null and b/iso_root/EFI/BOOT/BOOTIA32.EFI differ diff --git a/iso_root/EFI/BOOT/BOOTX64.EFI b/iso_root/EFI/BOOT/BOOTX64.EFI new file mode 100644 index 0000000..a5a4883 Binary files /dev/null and b/iso_root/EFI/BOOT/BOOTX64.EFI differ diff --git a/iso_root/README.md b/iso_root/README.md new file mode 100644 index 0000000..916502b --- /dev/null +++ b/iso_root/README.md @@ -0,0 +1,148 @@ +# Brew OS 1.01 Alpha + +## Brewkernel is now BrewOS! +Brewkernel will from now on be deprecated as it's core became too messy. I have built a less bloated kernel and wrote a DE above it, which is why it is now an OS instead of a kernel (in my opinion). + + +
+Brew Kernel is a simple x86_64 hobbyist operating system. +It features a DE (and WM), a FAT32 filesystem, customizable UI and much much more! + +## Features +- Brew WM +- Fat 32 FS +- 64-bit long mode support +- Multiboot2 compliant +- Text editor +- IDT +- Ability to run on actual x86_64 hardware +- CLI + +## Prerequisites + +To build BrewOS, you'll need the following tools installed: + +- **x86_64 ELF Toolchain**: `x86_64-elf-gcc`, `x86_64-elf-ld` +- **NASM**: Netwide Assembler for compiling assembly code +- **xorriso**: For creating bootable ISO images +- **QEMU** (optional): For testing the kernel in an emulator + +On macOS, you can install these using Homebrew: +```sh +brew install x86_64-elf-binutils x86_64-elf-gcc nasm xorriso qemu +``` + +## Building + +Simply run `make` from the project root: + +```sh +make +``` + +This will: +1. Compile all kernel C sources and assembly files +2. Link the kernel ELF binary +3. Generate a bootable ISO image (`brewos.iso`) + +The build output is organized as follows: +- Compiled object files: `build/` +- ISO root filesystem: `iso_root/` +- Final ISO image: `brewos.iso` + +## Running + +### QEMU Emulation + +Run the kernel in QEMU: + +```sh +make run +``` + +Or manually: +```sh +qemu-system-x86_64 -m 2G -serial stdio -cdrom brewos.iso -boot d +``` + +### Running on Real Hardware + +*Warning: This is at YOUR OWN RISK. This software comes with ZERO warranty and may break your system.* + +1. **Create bootable USB**: Use [Balena Etcher](https://www.balena.io/etcher/) to flash `brewos.iso` to a USB drive + +2. **Prepare the system**: + - Enable legacy (BIOS) boot in your system BIOS/UEFI settings + - Disable Secure Boot if needed + +3. **Boot**: Insert the USB drive and select it in the boot menu during startup + +4. **Tested Hardware**: + - HP EliteDesk 705 G4 DM (AMD Ryzen 5 PRO 2400G, Radeon Vega) + - Lenovo ThinkPad A475 20KL002VMH (AMD Pro A12-8830B, Radeon R7) + + +## Project Structure + +- `src/kernel/` - Main kernel implementation + - `boot.asm` - Boot assembly code + - `main.c` - Kernel entry point + - `*.c / *.h` - Core kernel modules (graphics, interrupts, filesystem, etc.) + - `cli_apps/` - Command-line applications + - `wallpaper.ppm` - Default desktop wallpaper +- `build/` - Compiled object files (generated during build) +- `iso_root/` - ISO filesystem layout (generated during build) +- `limine/` - Limine bootloader files (downloaded automatically) +- `linker.ld` - Linker script for x86_64 ELF +- `limine.cfg` - Limine bootloader configuration +- `Makefile` - Build configuration and targets + + + + +### +### + +

Help me brew some coffee! ☕️

+ +### + +

+ If you enjoy this project, and like what i'm doing here, consider buying me a coffee! +

+ + Buy Me A Coffee + +

+ +### + + +## License + +Copyright (C) 2024-2026 boreddevnl + +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 3 of the License, or (at your option) any later version. + +NOTICE +------ + +This product includes software developed by Chris ("boreddevnl") as part of the BrewKernel project. + +Copyright (C) 2024–2026 Chris / boreddevnl (previously boreddevhq) + +All source files in this repository contain copyright and license +headers that must be preserved in redistributions and derivative works. + +If you distribute or modify this project (in whole or in part), +you MUST: + + - Retain all copyright and license headers at the top of each file. + - Include this NOTICE file along with any redistributions or + derivative works. + - Provide clear attribution to the original author in documentation + or credits where appropriate. + +The above attribution requirements are informational and intended to +ensure proper credit is given. They do not alter or supersede the +terms of the GNU General Public License (GPL), which governs this work. diff --git a/iso_root/brewos.elf b/iso_root/brewos.elf new file mode 100755 index 0000000..89a352a Binary files /dev/null and b/iso_root/brewos.elf differ diff --git a/iso_root/limine-bios-cd.bin b/iso_root/limine-bios-cd.bin new file mode 100644 index 0000000..645bf55 Binary files /dev/null and b/iso_root/limine-bios-cd.bin differ diff --git a/iso_root/limine-bios.sys b/iso_root/limine-bios.sys new file mode 100644 index 0000000..6d95511 Binary files /dev/null and b/iso_root/limine-bios.sys differ diff --git a/iso_root/limine-uefi-cd.bin b/iso_root/limine-uefi-cd.bin new file mode 100644 index 0000000..873b059 Binary files /dev/null and b/iso_root/limine-uefi-cd.bin differ diff --git a/iso_root/limine.cfg b/iso_root/limine.cfg new file mode 100644 index 0000000..9a3e839 --- /dev/null +++ b/iso_root/limine.cfg @@ -0,0 +1,14 @@ +# Timeout in seconds that Limine will wait before automatically booting. +TIMEOUT=3 + +# The entry name that will be displayed in the boot menu. +:BrewOS + # We use the Limine boot protocol. + PROTOCOL=limine + + # Path to the kernel to boot. boot:/// represents the partition on which limine.cfg is located. + KERNEL_PATH=boot:///brewos.elf + + # Set screen resolution to 1920x1080 + FRAMEBUFFER_WIDTH=1920 + FRAMEBUFFER_HEIGHT=1080 diff --git a/limine.cfg b/limine.cfg new file mode 100644 index 0000000..9a3e839 --- /dev/null +++ b/limine.cfg @@ -0,0 +1,14 @@ +# Timeout in seconds that Limine will wait before automatically booting. +TIMEOUT=3 + +# The entry name that will be displayed in the boot menu. +:BrewOS + # We use the Limine boot protocol. + PROTOCOL=limine + + # Path to the kernel to boot. boot:/// represents the partition on which limine.cfg is located. + KERNEL_PATH=boot:///brewos.elf + + # Set screen resolution to 1920x1080 + FRAMEBUFFER_WIDTH=1920 + FRAMEBUFFER_HEIGHT=1080 diff --git a/linker.ld b/linker.ld new file mode 100644 index 0000000..a593f55 --- /dev/null +++ b/linker.ld @@ -0,0 +1,43 @@ +/* Tell the linker that we want an x86_64 ELF64 output file */ +OUTPUT_FORMAT(elf64-x86-64) +OUTPUT_ARCH(i386:x86-64) + +/* We want the symbol _start to be our entry point */ +ENTRY(_start) + +/* Define the memory layout for the kernel */ +SECTIONS +{ + /* We want to be loaded in the upper half of memory */ + . = 0xffffffff80000000; + + .text : { + *(.text*) + } + + .rodata : { + *(.rodata*) + } + + .data : { + *(.data*) + } + + .bss : { + *(COMMON) + *(.bss*) + } + + /* Limine requests section */ + .requests : { + KEEP(*(.requests_start)) + KEEP(*(.requests)) + KEEP(*(.requests_end)) + } + + /* Discard unnecessary sections */ + /DISCARD/ : { + *(.eh_frame) + *(.note .note.*) + } +} diff --git a/src/kernel/about.c b/src/kernel/about.c new file mode 100644 index 0000000..604d4a1 --- /dev/null +++ b/src/kernel/about.c @@ -0,0 +1,65 @@ +#include "about.h" +#include "graphics.h" +#include "wm.h" +#include + +Window win_about; + +// Color definitions +#define COLOR_BLUE_LOGO 0xFF1E8AF5 +#define COLOR_GREEN_LOGO 0xFF6DD651 +#define COLOR_YELLOW_LOGO 0xFFF5BE34 +#define COLOR_RED_LOGO 0xFFF05456 +#define COLOR_PURPLE_LOGO 0xFFA65DC2 +#define COLOR_CYAN_LOGO 0xFF368DF7 + +static void about_paint(Window *win) { + // Background + draw_rect(win->x + 4, win->y + 24, win->w - 8, win->h - 28, COLOR_LTGRAY); + + int offset_x = win->x + 15; + int offset_y = win->y + 35; + + // Draw brewkernel ASCII logo + draw_string(offset_x, offset_y, "( (", COLOR_BLUE_LOGO); + + draw_string(offset_x, offset_y + 15, " ) )", COLOR_GREEN_LOGO); + + draw_string(offset_x, offset_y + 30, " ........", COLOR_YELLOW_LOGO); + + draw_string(offset_x, offset_y + 45, " | |]", COLOR_RED_LOGO); + + draw_string(offset_x, offset_y + 60, " \\ /", COLOR_PURPLE_LOGO); + + draw_string(offset_x, offset_y + 75, " `----'", COLOR_CYAN_LOGO); + + // Version info + draw_string(offset_x, offset_y + 105, "BrewOS", COLOR_BLACK); + draw_string(offset_x, offset_y + 120, "Version 1.0", COLOR_BLACK); + + // Copyright + draw_string(offset_x, offset_y + 150, "(C) 2026 boreddevnl.", COLOR_BLACK); + draw_string(offset_x, offset_y + 165, "All rights reserved.", COLOR_BLACK); +} + +static void about_click(Window *win, int x, int y) { + (void)win; + (void)x; + (void)y; + // No interactive elements needed for About dialog +} + +void about_init(void) { + win_about.title = "About BrewOS"; + win_about.x = 250; + win_about.y = 180; + win_about.w = 185; + win_about.h = 240; + win_about.visible = false; + win_about.focused = false; + win_about.z_index = 0; + win_about.paint = about_paint; + win_about.handle_click = about_click; + win_about.handle_right_click = NULL; + win_about.handle_key = NULL; +} diff --git a/src/kernel/about.h b/src/kernel/about.h new file mode 100644 index 0000000..2a95f71 --- /dev/null +++ b/src/kernel/about.h @@ -0,0 +1,10 @@ +#ifndef ABOUT_H +#define ABOUT_H + +#include "wm.h" + +extern Window win_about; + +void about_init(void); + +#endif diff --git a/src/kernel/boot.asm b/src/kernel/boot.asm new file mode 100644 index 0000000..598878d --- /dev/null +++ b/src/kernel/boot.asm @@ -0,0 +1,23 @@ +; brew-os/src/kernel/boot.asm +; 64-bit Entry Point for BrewOS + +section .text +global _start +extern kmain + +bits 64 + +_start: + ; Ensure interrupts are disabled + cli + + ; Setup stack is handled by Limine, but we can re-align if paranoid + ; (Limine guarantees 16-byte alignment) + + ; Call the C kernel entry point + call kmain + + ; Halt if kmain returns + hlt +.loop: + jmp .loop diff --git a/src/kernel/calculator.c b/src/kernel/calculator.c new file mode 100644 index 0000000..b723816 --- /dev/null +++ b/src/kernel/calculator.c @@ -0,0 +1,170 @@ +#include "calculator.h" +#include "graphics.h" +#include "wm.h" +#include +#include + +Window win_calculator; + +static long long calc_acc = 0; +static long long calc_curr = 0; +static char calc_op = 0; +static bool calc_new_entry = true; +static bool calc_error = false; + +// Helper to convert int to string +static void int_to_str(long long n, char *buf) { + if (n == 0) { + buf[0] = '0'; buf[1] = 0; return; + } + + int i = 0; + bool neg = n < 0; + if (neg) n = -n; + + char temp[32]; + while (n > 0) { + temp[i++] = '0' + (n % 10); + n /= 10; + } + if (neg) temp[i++] = '-'; + + int j = 0; + while (i > 0) { + buf[j++] = temp[--i]; + } + buf[j] = 0; +} + +static void update_display(Window *win) { + if (calc_error) { + char *err = "Error"; + int i = 0; while(err[i]) { win->buffer[i] = err[i]; i++; } + win->buffer[i] = 0; + } else { + int_to_str(calc_curr, win->buffer); + } + win->buf_len = 0; while(win->buffer[win->buf_len]) win->buf_len++; +} + +static void calculator_paint(Window *win) { + // Background + draw_rect(win->x + 4, win->y + 24, win->w - 8, win->h - 28, COLOR_LTGRAY); + + // Display Area + draw_bevel_rect(win->x + 10, win->y + 30, win->w - 20, 25, true); + // Right align text + int text_w = win->buf_len * 8; + int text_x = win->x + win->w - 15 - text_w; + draw_string(text_x, win->y + 38, win->buffer, COLOR_BLACK); + + // Buttons + const char *labels[] = { + "7", "8", "9", "/", + "4", "5", "6", "*", + "1", "2", "3", "-", + "0", "C", "=", "+" + }; + + int bw = 30; + int bh = 25; + int gap = 5; + int start_x = win->x + 10; + int start_y = win->y + 65; + + for (int i = 0; i < 16; i++) { + int r = i / 4; + int c = i % 4; + draw_button(start_x + c*(bw+gap), start_y + r*(bh+gap), bw, bh, labels[i], false); + } +} + +static void do_op(void) { + if (calc_op == '+') calc_acc += calc_curr; + else if (calc_op == '-') calc_acc -= calc_curr; + else if (calc_op == '*') calc_acc *= calc_curr; + else if (calc_op == '/') { + if (calc_curr == 0) calc_error = true; + else calc_acc /= calc_curr; + } else { + calc_acc = calc_curr; + } +} + +static void calculator_click(Window *win, int x, int y) { + int bw = 30; + int bh = 25; + int gap = 5; + int start_x = 10; + int start_y = 65; + + for (int i = 0; i < 16; i++) { + int r = i / 4; + int c = i % 4; + int bx = start_x + c*(bw+gap); + int by = start_y + r*(bh+gap); + + if (x >= bx && x < bx + bw && y >= by && y < by + bh) { + // Clicked button i + const char *labels[] = { + "7", "8", "9", "/", + "4", "5", "6", "*", + "1", "2", "3", "-", + "0", "C", "=", "+" + }; + char lbl = labels[i][0]; + + if (lbl >= '0' && lbl <= '9') { + if (calc_new_entry || calc_curr == 0) { + calc_curr = lbl - '0'; + calc_new_entry = false; + } else { + calc_curr = calc_curr * 10 + (lbl - '0'); + } + calc_error = false; + } else if (lbl == 'C') { + calc_curr = 0; + calc_acc = 0; + calc_op = 0; + calc_new_entry = true; + calc_error = false; + } else if (lbl == '=') { + do_op(); + calc_curr = calc_acc; + calc_op = 0; + calc_new_entry = true; + } else { + if (!calc_new_entry) { + if (calc_op) do_op(); + else calc_acc = calc_curr; + } + calc_op = lbl; + calc_new_entry = true; + } + + update_display(win); + wm_paint(); // Request repaint + return; + } + } +} + +void calculator_init(void) { + win_calculator.title = "Calculator"; + win_calculator.x = 200; + win_calculator.y = 200; + win_calculator.w = 160; + win_calculator.h = 200; + win_calculator.visible = false; + win_calculator.focused = false; + win_calculator.z_index = 0; + win_calculator.paint = calculator_paint; + win_calculator.handle_click = calculator_click; + win_calculator.handle_right_click = NULL; + + calc_curr = 0; + calc_acc = 0; + calc_op = 0; + calc_new_entry = true; + update_display(&win_calculator); +} \ No newline at end of file diff --git a/src/kernel/calculator.h b/src/kernel/calculator.h new file mode 100644 index 0000000..33bb9b7 --- /dev/null +++ b/src/kernel/calculator.h @@ -0,0 +1,10 @@ +#ifndef CALCULATOR_H +#define CALCULATOR_H + +#include "wm.h" + +extern Window win_calculator; + +void calculator_init(void); + +#endif \ No newline at end of file diff --git a/src/kernel/cli_apps/about.c b/src/kernel/cli_apps/about.c new file mode 100644 index 0000000..ae1bb43 --- /dev/null +++ b/src/kernel/cli_apps/about.c @@ -0,0 +1,6 @@ +#include "cli_utils.h" + +void cli_cmd_about(char *args) { + (void)args; + cli_write("BrewOS Desktop v0.01 Alpha\n"); +} diff --git a/src/kernel/cli_apps/beep.c b/src/kernel/cli_apps/beep.c new file mode 100644 index 0000000..7946571 --- /dev/null +++ b/src/kernel/cli_apps/beep.c @@ -0,0 +1,16 @@ +#include "cli_utils.h" +#include "io.h" + +void cli_cmd_beep(char *args) { + (void)args; + cli_write("BEEP!\n"); + outb(0x43, 0xB6); + int freq = 1000; + int div = 1193180 / freq; + outb(0x42, div & 0xFF); + outb(0x42, (div >> 8) & 0xFF); + + outb(0x61, inb(0x61) | 0x03); + cli_delay(10000000); + outb(0x61, inb(0x61) & 0xFC); +} diff --git a/src/kernel/cli_apps/blind.c b/src/kernel/cli_apps/blind.c new file mode 100644 index 0000000..d1fe69b --- /dev/null +++ b/src/kernel/cli_apps/blind.c @@ -0,0 +1,8 @@ +#include "cli_utils.h" + +void cli_cmd_blind(char *args) { + (void)args; + cli_write("Woah.. is this heaven?\n"); + cli_write("no.\n"); + cli_write("This isn't TempleOS you fucking retard.\n"); +} diff --git a/src/kernel/cli_apps/clear.c b/src/kernel/cli_apps/clear.c new file mode 100644 index 0000000..32e17b6 --- /dev/null +++ b/src/kernel/cli_apps/clear.c @@ -0,0 +1,9 @@ +#include "cli_utils.h" + +// Public declaration from cmd.c +extern void cmd_screen_clear(void); + +void cli_cmd_clear(char *args) { + (void)args; + cmd_screen_clear(); +} diff --git a/src/kernel/cli_apps/cli_apps.h b/src/kernel/cli_apps/cli_apps.h new file mode 100644 index 0000000..e796f3b --- /dev/null +++ b/src/kernel/cli_apps/cli_apps.h @@ -0,0 +1,39 @@ +#ifndef CLI_APPS_H +#define CLI_APPS_H + +// All CLI command function declarations +void cli_cmd_help(char *args); +void cli_cmd_date(char *args); +void cli_cmd_math(char *args); +void cli_cmd_beep(char *args); +void cli_cmd_cowsay(char *args); +void cli_cmd_reboot(char *args); +void cli_cmd_shutdown(char *args); +void cli_cmd_uptime(char *args); +void cli_cmd_man(char *args); +void cli_cmd_license(char *args); +void cli_cmd_txtedit(char *args); +void cli_cmd_blind(char *args); +void cli_cmd_readtheman(char *args); +void cli_cmd_about(char *args); +void cli_cmd_clear(char *args); +void cli_cmd_exit(char *args); + +// Filesystem commands +void cli_cmd_cd(char *args); +void cli_cmd_pwd(char *args); +void cli_cmd_ls(char *args); +void cli_cmd_mkdir(char *args); +void cli_cmd_rm(char *args); +void cli_cmd_echo(char *args); +void cli_cmd_cat(char *args); + +// Memory management commands +void cli_cmd_meminfo(char *args); +void cli_cmd_malloc(char *args); +void cli_cmd_free_mem(char *args); +void cli_cmd_memblock(char *args); +void cli_cmd_memvalid(char *args); +void cli_cmd_memtest(char *args); + +#endif diff --git a/src/kernel/cli_apps/cli_command.h b/src/kernel/cli_apps/cli_command.h new file mode 100644 index 0000000..7174c01 --- /dev/null +++ b/src/kernel/cli_apps/cli_command.h @@ -0,0 +1,22 @@ +#ifndef CLI_COMMAND_H +#define CLI_COMMAND_H + +#include + +// Standard interface for CLI command output +// Commands should call these functions to write to the terminal +extern void cli_write(const char *str); +extern void cli_write_int(int n); +extern void cli_putchar(char c); + +// Callback function type for command execution +typedef void (*cmd_callback_t)(char *args); + +// Command entry in dispatch table +typedef struct { + const char *name; + cmd_callback_t callback; + const char *help_text; +} CLI_Command; + +#endif diff --git a/src/kernel/cli_apps/cli_utils.c b/src/kernel/cli_apps/cli_utils.c new file mode 100644 index 0000000..07b06b8 --- /dev/null +++ b/src/kernel/cli_apps/cli_utils.c @@ -0,0 +1,80 @@ +#include "cli_utils.h" + +// Forward declarations - these will be provided by cmd.c +extern void cmd_putchar(char c); +extern void cmd_write(const char *str); +extern void cmd_write_int(int n); + +void cli_memset(void *dest, int val, size_t len) { + unsigned char *ptr = dest; + while (len-- > 0) *ptr++ = val; +} + +size_t cli_strlen(const char *str) { + size_t len = 0; + while (str[len]) len++; + return len; +} + +int cli_strcmp(const char *s1, const char *s2) { + while (*s1 && (*s1 == *s2)) { + s1++; + s2++; + } + return *(const unsigned char*)s1 - *(const unsigned char*)s2; +} + +void cli_strcpy(char *dest, const char *src) { + while (*src) *dest++ = *src++; + *dest = 0; +} + +int cli_atoi(const char *str) { + int res = 0; + int sign = 1; + if (*str == '-') { sign = -1; str++; } + while (*str >= '0' && *str <= '9') { + res = res * 10 + (*str - '0'); + str++; + } + return res * sign; +} + +void cli_itoa(int n, char *buf) { + if (n == 0) { + buf[0] = '0'; buf[1] = 0; return; + } + int i = 0; + int sign = n < 0; + if (sign) n = -n; + while (n > 0) { + buf[i++] = (n % 10) + '0'; + n /= 10; + } + if (sign) buf[i++] = '-'; + buf[i] = 0; + // Reverse + for (int j = 0; j < i / 2; j++) { + char t = buf[j]; + buf[j] = buf[i - 1 - j]; + buf[i - 1 - j] = t; + } +} + +void cli_write(const char *str) { + cmd_write(str); +} + +void cli_write_int(int n) { + cmd_write_int(n); +} + +void cli_putchar(char c) { + cmd_putchar(c); +} + +void cli_delay(int iterations) { + for (volatile int i = 0; i < iterations; i++) { + __asm__ __volatile__("nop"); + } +} diff --git a/src/kernel/cli_apps/cli_utils.h b/src/kernel/cli_apps/cli_utils.h new file mode 100644 index 0000000..2823043 --- /dev/null +++ b/src/kernel/cli_apps/cli_utils.h @@ -0,0 +1,27 @@ +#ifndef CLI_UTILS_H +#define CLI_UTILS_H + +#include +#include + +// String utilities +void cli_memset(void *dest, int val, size_t len); +size_t cli_strlen(const char *str); +int cli_strcmp(const char *s1, const char *s2); +void cli_strcpy(char *dest, const char *src); +int cli_atoi(const char *str); +void cli_itoa(int n, char *buf); + +// IO utilities +void cli_write(const char *str); +void cli_write_int(int n); +void cli_putchar(char c); + +// Timing utility +void cli_delay(int iterations); + +// CLI Command declarations +void cli_cmd_shutdown(char *args); +void cli_cmd_reboot(char *args); + +#endif diff --git a/src/kernel/cli_apps/cowsay.c b/src/kernel/cli_apps/cowsay.c new file mode 100644 index 0000000..f10782b --- /dev/null +++ b/src/kernel/cli_apps/cowsay.c @@ -0,0 +1,17 @@ +#include "cli_utils.h" + +void cli_cmd_cowsay(char *args) { + if (!args || !*args) args = (char*)"Brew!"; + size_t len = cli_strlen(args); + + cli_write(" "); + for(size_t i=0; i\n "); + for(size_t i=0; i\n"); + return; + } + + char dirname[256]; + int i = 0; + while (args[i] && args[i] != ' ' && args[i] != '\t') { + dirname[i] = args[i]; + i++; + } + dirname[i] = 0; + + if (fat32_mkdir(dirname)) { + cli_write("Created directory: "); + cli_write(dirname); + cli_write("\n"); + } else { + cli_write("Error: Cannot create directory\n"); + } +} + +void cli_cmd_rm(char *args) { + if (!args || args[0] == 0) { + cli_write("Usage: rm \n"); + return; + } + + char filename[256]; + int i = 0; + while (args[i] && args[i] != ' ' && args[i] != '\t') { + filename[i] = args[i]; + i++; + } + filename[i] = 0; + + if (fat32_delete(filename)) { + cli_write("Deleted: "); + cli_write(filename); + cli_write("\n"); + } else { + cli_write("Error: Cannot delete file\n"); + } +} + +void cli_cmd_echo(char *args) { + if (!args || args[0] == 0) { + cli_write("\n"); + return; + } + + // Check for redirection operators + char *redirect_ptr = NULL; + char redirect_mode = 0; // '>' for write, 'a' for append + char output_file[256] = {0}; + char echo_text[512] = {0}; + + // Find > or >> + for (int i = 0; args[i]; i++) { + if (args[i] == '>' && args[i+1] == '>') { + redirect_ptr = args + i + 2; + redirect_mode = 'a'; // append + // Copy text before redirection + for (int j = 0; j < i; j++) { + echo_text[j] = args[j]; + } + echo_text[i] = 0; + break; + } else if (args[i] == '>') { + redirect_ptr = args + i + 1; + redirect_mode = '>'; // write + // Copy text before redirection + for (int j = 0; j < i; j++) { + echo_text[j] = args[j]; + } + echo_text[i] = 0; + break; + } + } + + // If no redirection, just print the text + if (!redirect_ptr) { + cli_write(args); + cli_write("\n"); + return; + } + + // Parse output filename + int i = 0; + while (redirect_ptr[i] && (redirect_ptr[i] == ' ' || redirect_ptr[i] == '\t')) { + i++; + } + + int j = 0; + while (redirect_ptr[i] && redirect_ptr[i] != ' ' && redirect_ptr[i] != '\t') { + output_file[j++] = redirect_ptr[i++]; + } + output_file[j] = 0; + + if (!output_file[0]) { + cli_write("Error: No output file specified\n"); + return; + } + + // Open file + const char *mode = (redirect_mode == 'a') ? "a" : "w"; + FAT32_FileHandle *fh = fat32_open(output_file, mode); + if (!fh) { + cli_write("Error: Cannot open file for writing\n"); + return; + } + + // Write text + int text_len = 0; + while (echo_text[text_len]) text_len++; + + fat32_write(fh, echo_text, text_len); + fat32_write(fh, "\n", 1); + fat32_close(fh); + + cli_write("Wrote to: "); + cli_write(output_file); + cli_write("\n"); +} + +void cli_cmd_cat(char *args) { + if (!args || args[0] == 0) { + cli_write("Usage: cat \n"); + return; + } + + char filename[256]; + int i = 0; + while (args[i] && args[i] != ' ' && args[i] != '\t') { + filename[i] = args[i]; + i++; + } + filename[i] = 0; + + FAT32_FileHandle *fh = fat32_open(filename, "r"); + if (!fh) { + cli_write("Error: Cannot open file\n"); + return; + } + + // Read and display file + char buffer[4096]; + int bytes_read; + while ((bytes_read = fat32_read(fh, buffer, sizeof(buffer))) > 0) { + for (int j = 0; j < bytes_read; j++) { + cli_putchar(buffer[j]); + } + } + + fat32_close(fh); +} diff --git a/src/kernel/cli_apps/help.c b/src/kernel/cli_apps/help.c new file mode 100644 index 0000000..dc99843 --- /dev/null +++ b/src/kernel/cli_apps/help.c @@ -0,0 +1,18 @@ +#include "cli_utils.h" + +void cli_cmd_help(char *args) { + (void)args; + cli_write("Available commands:\n"); + cli_write(" HELP - Display this help message\n"); + cli_write(" DATE - Display current date and time\n"); + cli_write(" CLEAR - Clear the screen\n"); + cli_write(" ABOUT - System info\n"); + cli_write(" MATH - math (e.g. math + 1 2)\n"); + cli_write(" MAN - Show user manual (interactive)\n"); + cli_write(" LICENSE - Show license (interactive)\n"); + cli_write(" UPTIME - System uptime\n"); + cli_write(" BEEP - Make a sound\n"); + cli_write(" COWSAY - cowsay \n"); + cli_write(" REBOOT - Reboot system\n"); + cli_write(" SHUTDOWN- Shutdown system\n"); +} diff --git a/src/kernel/cli_apps/license.c b/src/kernel/cli_apps/license.c new file mode 100644 index 0000000..965de42 --- /dev/null +++ b/src/kernel/cli_apps/license.c @@ -0,0 +1,21 @@ +#include "cli_utils.h" + +// Forward declaration from cmd.c +extern void pager_wrap_content(const char **lines, int count); +extern void pager_set_mode(void); + +const char* license_pages[] = { + " GNU GENERAL PUBLIC LICENSE", + " Version 3, 29 June 2007", + " Copyright (C) 2024-2026 boreddevnl", + "", + " (License text abbreviated for build size. See https://www.gnu.org/licenses/gpl-3.0.txt)", + "--- End of License ---" +}; +const int license_num_lines = sizeof(license_pages) / sizeof(char*); + +void cli_cmd_license(char *args) { + (void)args; + pager_wrap_content(license_pages, license_num_lines); + pager_set_mode(); +} diff --git a/src/kernel/cli_apps/man.c b/src/kernel/cli_apps/man.c new file mode 100644 index 0000000..e2ae3a3 --- /dev/null +++ b/src/kernel/cli_apps/man.c @@ -0,0 +1,46 @@ +#include "cli_utils.h" + +// Forward declaration from cmd.c +extern void pager_wrap_content(const char **lines, int count); +extern void pager_set_mode(void); + +const char* manual_pages[] = { + "BrewKernel User Manual", + "----------------------", + "", + "Welcome to the BrewKernel.", + "", + "== Features ==", + "* Ramdisk-based Filesystem: A simple in-memory filesystem.", + "* VGA Text Mode Driver: Full control over text/colors.", + "* PS/2 Keyboard Driver: Handles key presses.", + "* Simple CLI: A basic shell.", + "", + "== Available Commands ==", + "HELP: Displays a short list of available commands.", + "MAN: Shows this detailed user manual.", + "ABOUT: Displays information about the kernel.", + "MATH: A simple calculator.", + "DATE: Displays the current date and time.", + "TXTEDIT: A simple text editor with file path support.", + " USAGE: txtedit ", + " EXAMPLES:", + " txtedit file.txt (relative path in current directory)", + " txtedit /file.txt (absolute path in root)", + " txtedit /docs/note.txt (absolute path with subdirectories)", + " FEATURES: Create/Edit files, Save (to RAM), Navigation.", + "CLEAR: Clears the entire screen.", + "EXIT: Exits the CLI mode.", + "LICENSE: Displays the full GNU GPL v3.", + "COWSAY: Moo!", + "UPTIME: Shows how long the system has been running.", + "BEEP: Makes a beep sound.", + "--- End of Manual ---" +}; +const int manual_num_lines = sizeof(manual_pages) / sizeof(char*); + +void cli_cmd_man(char *args) { + (void)args; + pager_wrap_content(manual_pages, manual_num_lines); + pager_set_mode(); +} diff --git a/src/kernel/cli_apps/math.c b/src/kernel/cli_apps/math.c new file mode 100644 index 0000000..258c145 --- /dev/null +++ b/src/kernel/cli_apps/math.c @@ -0,0 +1,34 @@ +#include "cli_utils.h" + +void cli_cmd_math(char *args) { + while (*args == ' ') args++; + if (!*args) { + cli_write("Usage: math \n"); + return; + } + char op = *args; + args++; + while (*args == ' ') args++; + + char *end = args; + while (*end && *end != ' ') end++; + int saved = *end; + *end = 0; + int n1 = cli_atoi(args); + if (saved) *end = saved; + + args = end; + while (*args == ' ') args++; + + int n2 = cli_atoi(args); + + int res = 0; + switch(op) { + case '+': res = n1 + n2; break; + case '-': res = n1 - n2; break; + case '*': res = n1 * n2; break; + case '/': if(n2!=0) res = n1/n2; else { cli_write("Div by zero\n"); return; } break; + default: cli_write("Invalid op.\n"); return; + } + cli_write("Result: "); cli_write_int(res); cli_write("\n"); +} diff --git a/src/kernel/cli_apps/memcmd.c b/src/kernel/cli_apps/memcmd.c new file mode 100644 index 0000000..b357b14 --- /dev/null +++ b/src/kernel/cli_apps/memcmd.c @@ -0,0 +1,143 @@ +#include "cli_utils.h" +#include "../memory_manager.h" + +#define MAX_TEST_ALLOCS 64 +static void *test_allocs[MAX_TEST_ALLOCS]; +static int test_alloc_count = 0; + +void cli_cmd_malloc(char *args) { + if (!args || !args[0]) { + cli_write("Usage: malloc \n"); + cli_write("Example: malloc 10\n"); + return; + } + + // Parse size in KB + int size_kb = cli_atoi(args); + if (size_kb <= 0 || size_kb > 1024) { + cli_write("Invalid size. Use 1-1024 KB\n"); + return; + } + + size_t size = size_kb * 1024; + void *ptr = kmalloc(size); + + if (ptr == NULL) { + cli_write("Allocation failed!\n"); + return; + } + + // Track allocation for later freeing + if (test_alloc_count < MAX_TEST_ALLOCS) { + test_allocs[test_alloc_count++] = ptr; + } + + cli_write("Allocated "); + cli_write_int(size_kb); + cli_write("KB at address 0x"); + cli_write_int((uintptr_t)ptr / 1024); + cli_write("\n"); + cli_write("Test allocation index: "); + cli_write_int(test_alloc_count - 1); + cli_write("\n"); + + memory_print_stats(); +} + +void cli_cmd_free_mem(char *args) { + if (!args || !args[0]) { + cli_write("Usage: freemem \n"); + cli_write("Specify the allocation index from malloc output\n"); + return; + } + + int idx = cli_atoi(args); + if (idx < 0 || idx >= test_alloc_count) { + cli_write("Invalid index. Must be 0-"); + cli_write_int(test_alloc_count - 1); + cli_write("\n"); + return; + } + + void *ptr = test_allocs[idx]; + if (ptr == NULL) { + cli_write("Allocation at index "); + cli_write_int(idx); + cli_write(" is NULL\n"); + return; + } + + kfree(ptr); + test_allocs[idx] = NULL; + + cli_write("Freed allocation at index "); + cli_write_int(idx); + cli_write("\n"); + + memory_print_stats(); +} + +void cli_cmd_memblock(char *args) { + (void)args; + + cli_write("Detailed block information:\n"); + memory_print_detailed(); +} + +void cli_cmd_memvalid(char *args) { + (void)args; + + cli_write("Validating memory integrity...\n"); + memory_validate(); +} + +void cli_cmd_memtest(char *args) { + (void)args; + + cli_write("\n=== MEMORY STRESS TEST ===\n"); + + // Allocate multiple blocks + cli_write("Allocating 10 blocks of 256KB each...\n"); + void *test_ptrs[10]; + + for (int i = 0; i < 10; i++) { + test_ptrs[i] = kmalloc(256 * 1024); + if (test_ptrs[i] == NULL) { + cli_write("Allocation "); + cli_write_int(i); + cli_write(" failed\n"); + + // Free previous allocations + for (int j = 0; j < i; j++) { + kfree(test_ptrs[j]); + } + return; + } + cli_write("Allocated block "); + cli_write_int(i); + cli_write("\n"); + } + + memory_print_stats(); + + // Free every other block (create fragmentation) + cli_write("\nFreeing alternate blocks to create fragmentation...\n"); + for (int i = 0; i < 10; i += 2) { + kfree(test_ptrs[i]); + cli_write("Freed block "); + cli_write_int(i); + cli_write("\n"); + } + + memory_print_stats(); + + // Free remaining blocks + cli_write("\nFreeing remaining blocks...\n"); + for (int i = 1; i < 10; i += 2) { + kfree(test_ptrs[i]); + } + + memory_print_stats(); + + cli_write("=== TEST COMPLETE ===\n\n"); +} diff --git a/src/kernel/cli_apps/meminfo.c b/src/kernel/cli_apps/meminfo.c new file mode 100644 index 0000000..ab81ecd --- /dev/null +++ b/src/kernel/cli_apps/meminfo.c @@ -0,0 +1,9 @@ +#include "cli_utils.h" +#include "../memory_manager.h" + +void cli_cmd_meminfo(char *args) { + (void)args; + + // Print memory statistics + memory_print_stats(); +} diff --git a/src/kernel/cli_apps/readtheman.c b/src/kernel/cli_apps/readtheman.c new file mode 100644 index 0000000..6c238f3 --- /dev/null +++ b/src/kernel/cli_apps/readtheman.c @@ -0,0 +1,14 @@ +#include "cli_utils.h" + +// Forward declaration from cmd.c +extern void cli_cmd_beep(char *args); + +void cli_cmd_readtheman(char *args) { + (void)args; + cli_write("\nYou read the manual? NERD. you know what?\n"); + cli_write("Fuck you.\n"); + for(int i=0; i<3; i++) { + cli_cmd_beep(NULL); + cli_delay(1000000); + } +} diff --git a/src/kernel/cli_apps/reboot.c b/src/kernel/cli_apps/reboot.c new file mode 100644 index 0000000..bc38a5a --- /dev/null +++ b/src/kernel/cli_apps/reboot.c @@ -0,0 +1,11 @@ +#include "cli_utils.h" +#include "io.h" + +void cli_cmd_reboot(char *args) { + (void)args; + cli_write("Rebooting...\n"); + cli_delay(10000000); + while ((inb(0x64) & 2) != 0) cli_delay(1000); + outb(0x64, 0xFE); + asm volatile ("int $0x3"); +} diff --git a/src/kernel/cli_apps/shutdown.c b/src/kernel/cli_apps/shutdown.c new file mode 100644 index 0000000..e642ea8 --- /dev/null +++ b/src/kernel/cli_apps/shutdown.c @@ -0,0 +1,11 @@ +#include "cli_utils.h" +#include "io.h" + +void cli_cmd_shutdown(char *args) { + (void)args; + cli_write("Shutting down...\n"); + cli_delay(10000000); + outb(0x64, 0xFE); + outw(0x604, 0x2000); + outw(0xB004, 0x2000); +} diff --git a/src/kernel/cli_apps/txtedit.c b/src/kernel/cli_apps/txtedit.c new file mode 100644 index 0000000..ea1be39 --- /dev/null +++ b/src/kernel/cli_apps/txtedit.c @@ -0,0 +1,61 @@ +#include "cli_utils.h" +#include "fat32.h" +#include "wm.h" + +// Forward declarations from editor.h and wm.c +extern void editor_open_file(const char *filename); +extern void editor_init(void); +extern Window win_editor; +extern Window win_explorer; +extern Window win_cmd; +extern Window win_notepad; +extern Window win_calculator; + +void cli_cmd_txtedit(char *args) { + // Parse the file path argument + char filepath[256]; + int i = 0; + + // Skip leading whitespace + while (args && args[i] && (args[i] == ' ' || args[i] == '\t')) { + i++; + } + + // Extract filepath + int j = 0; + while (args && args[i] && args[i] != ' ' && args[i] != '\t' && j < 255) { + filepath[j++] = args[i++]; + } + filepath[j] = 0; + + // If no filepath provided, create a new empty file + if (j == 0) { + cli_write("Usage: txtedit \n"); + cli_write("Example: txtedit myfile.txt\n"); + cli_write(" txtedit /document.txt\n"); + return; + } + + // Normalize the path (handles relative and absolute paths) + char normalized_path[256]; + fat32_normalize_path(filepath, normalized_path); + + // Open the file in the GUI editor + editor_open_file(normalized_path); + + // Make editor window visible and focused, bring to front + win_editor.visible = true; + win_editor.focused = true; + + // Calculate max z_index to bring window to front + int max_z = 0; + if (win_explorer.z_index > max_z) max_z = win_explorer.z_index; + if (win_cmd.z_index > max_z) max_z = win_cmd.z_index; + if (win_notepad.z_index > max_z) max_z = win_notepad.z_index; + if (win_calculator.z_index > max_z) max_z = win_calculator.z_index; + win_editor.z_index = max_z + 1; + + cli_write("Opening: "); + cli_write(normalized_path); + cli_write("\n"); +} diff --git a/src/kernel/cli_apps/uptime.c b/src/kernel/cli_apps/uptime.c new file mode 100644 index 0000000..c2dd133 --- /dev/null +++ b/src/kernel/cli_apps/uptime.c @@ -0,0 +1,26 @@ +#include "cli_utils.h" + +// Forward declarations from cmd.c +extern void rtc_get_datetime(int *y, int *m, int *d, int *h, int *min, int *s); +extern int boot_time_init; +extern int boot_year, boot_month, boot_day, boot_hour, boot_min, boot_sec; + +void cli_cmd_uptime(char *args) { + (void)args; + int y, m, d, h, min, s; + rtc_get_datetime(&y, &m, &d, &h, &min, &s); + + int start_sec = boot_hour * 3600 + boot_min * 60 + boot_sec; + int curr_sec = h * 3600 + min * 60 + s; + if (curr_sec < start_sec) curr_sec += 24 * 3600; + + int diff = curr_sec - start_sec; + int up_h = diff / 3600; + int up_m = (diff % 3600) / 60; + int up_s = diff % 60; + + cli_write("Uptime: "); + cli_write_int(up_h); cli_write("h "); + cli_write_int(up_m); cli_write("m "); + cli_write_int(up_s); cli_write("s\n"); +} diff --git a/src/kernel/cli_apps_old/beep.h b/src/kernel/cli_apps_old/beep.h new file mode 100644 index 0000000..81aeaa8 --- /dev/null +++ b/src/kernel/cli_apps_old/beep.h @@ -0,0 +1,81 @@ +/* + * Brew Kernel + * Copyright (C) 2024-2026 boreddevnl + * + * 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 3 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. If not, see . + */ +#ifndef BEEP_H +#define BEEP_H + +#include "../print.h" +#include "../io.h" + +// Forward declaration of brewing function from main.c +void brewing(int iterations); + +void beep_command() { + brew_str("\n"); + brew_str("BEEP!"); + // Add a 500ms delay before the beep + brewing(5000000); + + // Set the PIT to the desired frequency (1000 Hz for high pitch) + outb(0x43, 0xB6); // Command byte: channel 2, mode 3, binary + int frequency = 1000; + int divisor = 1193180 / frequency; + outb(0x42, divisor & 0xFF); + outb(0x42, (divisor >> 8) & 0xFF); + + // Turn speaker on + outb(0x61, inb(0x61) | 0x03); + + // Keep the sound for 1 second + brewing(10000000); + + // Turn speaker off + outb(0x61, inb(0x61) & 0xFC); + + brewing(1000000); + + outb(0x61, inb(0x61) | 0x03); + + // Keep the sound for 1 second + brewing(50000000); + + // Turn speaker off + outb(0x61, inb(0x61) & 0xFC); + + // Turn speaker on + outb(0x61, inb(0x61) | 0x03); + + // Keep the sound for 1 second + brewing(10000000); + + // Turn speaker off + outb(0x61, inb(0x61) & 0xFC); + + brewing(1000000); + + outb(0x61, inb(0x61) | 0x03); + + // Keep the sound for 1 second + brewing(50000000); + + // Turn speaker off + outb(0x61, inb(0x61) & 0xFC); + + +} + +#endif \ No newline at end of file diff --git a/src/kernel/cli_apps_old/blind.h b/src/kernel/cli_apps_old/blind.h new file mode 100644 index 0000000..34f67aa --- /dev/null +++ b/src/kernel/cli_apps_old/blind.h @@ -0,0 +1,33 @@ +/* + * Brew Kernel + * Copyright (C) 2024-2026 boreddevnl + * + * 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 3 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. If not, see . + */ +#ifndef APPS_BLIND_H +#define APPS_BLIND_H + +#include "print.h" + +static void blindme() { + print_set_color(PRINT_INDEX_0, PRINT_INDEX_15); + print_clear(); + brew_str("Woah.. is this heaven?\n"); + brew_str("no.\n"); + brew_str("This isn't TempleOS you fucking retard.\n"); + + +} + +#endif // APPS_BLIND_H diff --git a/src/kernel/cli_apps_old/calc.h b/src/kernel/cli_apps_old/calc.h new file mode 100644 index 0000000..3fbd9dc --- /dev/null +++ b/src/kernel/cli_apps_old/calc.h @@ -0,0 +1,291 @@ +/* + * Brew Kernel + * Copyright (C) 2024-2026 boreddevnl + * + * 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 3 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. If not, see . + */ +#ifndef APPS_CALC_H +#define APPS_CALC_H + +#include "print.h" +#include "keyboard.h" + +extern int buffer_pos; + +// Calculator state structure +struct { + char display[64]; // Display buffer for calculation + int display_len; + long current_val; + long prev_val; + char operation; + int cursor_row; // 0-3 for numpad rows + int cursor_col; // 0-3 for numpad cols + int has_operation; +} calc_state; + +// Redraw the calculator UI +static void calc_redraw_ui() { + // Clear screen by printing newlines (but fewer to avoid deadspace) + for (int i = 0; i < 24; i++) { + brew_str("\n"); + } + + // Display header + brew_str("===== CALCULATOR =====\n"); + brew_str("Display: "); + brew_str(calc_state.display); + brew_str("\n\n"); + + // Draw numpad with cursor + const char* buttons[4][4] = { + {"7", "8", "9", "/"}, + {"4", "5", "6", "*"}, + {"1", "2", "3", "-"}, + {"0", ".", "=", "+"} + }; + + for (int row = 0; row < 4; row++) { + brew_str(" "); + for (int col = 0; col < 4; col++) { + if (row == calc_state.cursor_row && col == calc_state.cursor_col) { + brew_str("["); + brew_str(buttons[row][col]); + brew_str("]"); + } else { + brew_str(" "); + brew_str(buttons[row][col]); + brew_str(" "); + } + brew_str(" "); + } + brew_str("\n"); + } + + brew_str("\nNavigate: Arrow Keys | Select: Enter | Clear: C | Quit: Q/ESC\n"); +} + +// Get button at current cursor position +static const char* calc_get_button() { + const char* buttons[4][4] = { + {"7", "8", "9", "/"}, + {"4", "5", "6", "*"}, + {"1", "2", "3", "-"}, + {"0", ".", "=", "+"} + }; + return buttons[calc_state.cursor_row][calc_state.cursor_col]; +} + +// Append character to display +static void calc_append_to_display(const char* text) { + int i = 0; + while (text[i] && calc_state.display_len < 63) { + calc_state.display[calc_state.display_len++] = text[i++]; + } + calc_state.display[calc_state.display_len] = '\0'; +} + +// Parse and calculate result +static void calc_perform_calculation() { + if (calc_state.display_len == 0) return; + + // Parse the current display value + long val = 0; + int i = 0; + int is_negative = 0; + + if (calc_state.display[0] == '-') { + is_negative = 1; + i = 1; + } + + while (i < calc_state.display_len && calc_state.display[i] != '.') { + if (calc_state.display[i] >= '0' && calc_state.display[i] <= '9') { + val = val * 10 + (calc_state.display[i] - '0'); + } + i++; + } + + if (is_negative) val = -val; + + long result = val; + + if (calc_state.has_operation) { + switch (calc_state.operation) { + case '+': result = calc_state.prev_val + val; break; + case '-': result = calc_state.prev_val - val; break; + case '*': result = calc_state.prev_val * val; break; + case '/': + if (val != 0) result = calc_state.prev_val / val; + else { + calc_state.display_len = 0; + calc_append_to_display("ERROR"); + return; + } + break; + } + } + + // Convert result back to string + calc_state.display_len = 0; + if (result < 0) { + calc_append_to_display("-"); + result = -result; + } + + if (result == 0) { + calc_append_to_display("0"); + } else { + char temp[32]; + int len = 0; + long temp_val = result; + while (temp_val > 0) { + temp[len++] = '0' + (temp_val % 10); + temp_val /= 10; + } + + // Reverse and append + for (int j = len - 1; j >= 0; j--) { + char c[2] = {temp[j], '\0'}; + calc_append_to_display(c); + } + } + + calc_state.prev_val = result; +} + +// Main calculator command +static void calc_cmd() { + // Initialize calculator state + calc_state.display[0] = '\0'; + calc_state.display_len = 0; + calc_state.current_val = 0; + calc_state.prev_val = 0; + calc_state.operation = '\0'; + calc_state.cursor_row = 0; + calc_state.cursor_col = 0; + calc_state.has_operation = 0; + + calc_redraw_ui(); + + buffer_pos = 0; + + while (1) { + if (check_keyboard()) { + unsigned char scan_code = read_scan_code(); + + // Handle arrow keys for navigation + if (scan_code == 0x48) { // UP arrow + if (calc_state.cursor_row > 0) { + calc_state.cursor_row--; + calc_redraw_ui(); + } + continue; + } else if (scan_code == 0x50) { // DOWN arrow + if (calc_state.cursor_row < 3) { + calc_state.cursor_row++; + calc_redraw_ui(); + } + continue; + } else if (scan_code == 0x4B) { // LEFT arrow + if (calc_state.cursor_col > 0) { + calc_state.cursor_col--; + calc_redraw_ui(); + } + continue; + } else if (scan_code == 0x4D) { // RIGHT arrow + if (calc_state.cursor_col < 3) { + calc_state.cursor_col++; + calc_redraw_ui(); + } + continue; + } + + char ascii_char = scan_code_to_ascii(scan_code); + + // Handle C key to clear + if (ascii_char == 'c' || ascii_char == 'C') { + calc_state.display[0] = '\0'; + calc_state.display_len = 0; + calc_state.current_val = 0; + calc_state.prev_val = 0; + calc_state.operation = '\0'; + calc_state.has_operation = 0; + calc_redraw_ui(); + continue; + } + + // Handle Q key or ESC to quit + if (ascii_char == 'q' || ascii_char == 'Q' || scan_code == 0x01) { + brew_str("\n"); + return; + } + + // Handle Enter to select button + if (ascii_char == '\n' || scan_code == 0x1C) { + const char* button = calc_get_button(); + + if (button[0] >= '0' && button[0] <= '9') { + // Number button + calc_append_to_display(button); + calc_redraw_ui(); + } else if (button[0] == '.') { + // Decimal point + int has_dot = 0; + for (int i = 0; i < calc_state.display_len; i++) { + if (calc_state.display[i] == '.') { + has_dot = 1; + break; + } + } + if (!has_dot && calc_state.display_len > 0) { + calc_append_to_display("."); + calc_redraw_ui(); + } + } else if (button[0] == '=' || button[0] == '+' || + button[0] == '-' || button[0] == '*' || button[0] == '/') { + + if (button[0] == '=') { + if (calc_state.has_operation && calc_state.display_len > 0) { + calc_perform_calculation(); + calc_state.operation = '\0'; + calc_state.has_operation = 0; + calc_redraw_ui(); + } + } else { + // Operation button pressed + if (calc_state.display_len > 0) { + // If we already have an operation pending, calculate first + if (calc_state.has_operation) { + calc_perform_calculation(); + } else { + // First operation: save display value to prev_val + calc_perform_calculation(); + } + // Set new operation + calc_state.operation = button[0]; + calc_state.has_operation = 1; + calc_state.display_len = 0; + calc_state.display[0] = '\0'; + calc_redraw_ui(); + } + } + continue; + } + } + } + } +} + +#endif // APPS_CALC_H \ No newline at end of file diff --git a/src/kernel/cli_apps_old/cowsay.h b/src/kernel/cli_apps_old/cowsay.h new file mode 100644 index 0000000..ccc77f5 --- /dev/null +++ b/src/kernel/cli_apps_old/cowsay.h @@ -0,0 +1,109 @@ +/* + * Brew Kernel Cowsay Utility + * Copyright (C) 2024-2026 boreddevnl + * + * + * 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 3 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. If not, see . + */ +// +// Implementation of cowsay from GNU/LINUX for Brew Kernel. +// Props to Tony Monroe for the original concept. +// One of these days we WONT port it to Windows. :3 +// +#ifndef APPS_COWSAY_H +#define APPS_COWSAY_H + +#include "print.h" + +// Function to calculate string length since we don't have string.h +static int brew_strlen(const char* str) { + int len = 0; + while (str[len]) len++; + return len; +} + +// String comparison for parsing command +static int strncmp_kernel(const char* s1, const char* s2, int n) { + while (n-- > 0) { + if (*s1 != *s2) { + return (*s1 - *s2); + } + if (*s1 == '\0') { + return 0; + } + s1++; + s2++; + } + return 0; +} + +// Function to find where the command arguments start +static const char* find_args(const char* cmd) { + while (*cmd && *cmd != ' ') cmd++; + while (*cmd == ' ') cmd++; + return cmd; +} + +// Function to draw the top border of the speech bubble +static void draw_top_border(int width) { + brew_str(" "); + for (int i = 0; i < width + 2; i++) { + brew_str("_"); + } + brew_str("\n"); +} + +// Function to draw the bottom border of the speech bubble +static void draw_bottom_border(int width) { + brew_str(" "); + for (int i = 0; i < width + 2; i++) { + brew_str("-"); + } + brew_str("\n"); +} + +// Main cowsay function +static void cowsay(const char* message) { + int len = brew_strlen(message); + + // Draw speech bubble + brew_str("\n"); // Start with a newline for better spacing + draw_top_border(len); + brew_str("< "); + brew_str(message); + brew_str(" >\n"); + draw_bottom_border(len); + + // Draw the cow + brew_str(" \\ ^__^\n"); + brew_str(" \\ (oo)\\_______\n"); + brew_str(" (__)\\ )\\/\\\n"); + brew_str(" ||----w |\n"); + brew_str(" || ||\n\n"); +} + +// Entry point for the cowsay command +static void display_cowsay(const char* command) { + const char* message; + if (!command || brew_strlen(command) == 0) { + message = "Brew!"; + } else { + message = find_args(command); + if (brew_strlen(message) == 0) { + message = "Brew!"; + } + } + cowsay(message); +} +#endif // APPS_COWSAY_H diff --git a/src/kernel/cli_apps_old/date.h b/src/kernel/cli_apps_old/date.h new file mode 100644 index 0000000..417278c --- /dev/null +++ b/src/kernel/cli_apps_old/date.h @@ -0,0 +1,196 @@ +/* + * Brew Kernel + * Copyright (C) 2024-2026 boreddevnl + * + * 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 3 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. If not, see . + */ +#ifndef DATE_H +#define DATE_H + +#include "print.h" +#include "keyboard.h" +#include "rtc.h" +#include "timezones.h" + +// String comparison function for kernel +static int strcmp_kernel_date(const char *s1, const char *s2) { + while (*s1 && (*s1 == *s2)) { + s1++; + s2++; + } + return *(const unsigned char*)s1 - *(const unsigned char*)s2; +} + +// Simple delay function +static inline void brewing_date(int iterations) { + for (volatile int i = 0; i < iterations; i++) { + __asm__ __volatile__("nop"); + } +} + +// --- Timezone Selection --- +static void select_timezone_for_continent_date(const char* continent, int* timezone_offset_h, int* timezone_offset_m) { + int selected_timezone = 0; + int needs_redraw = 1; + + while (1) { + if (needs_redraw) { + print_clear(); + brew_str("Select a timezone:\n"); + int current_timezone = 0; + for (int i = 0; i < num_timezones; i++) { + if (strcmp_kernel_date(timezones[i].continent, continent) == 0) { + if (current_timezone == selected_timezone) { + print_set_color(PRINT_INDEX_0, PRINT_INDEX_7); + } + brew_str(timezones[i].name); + brew_str("\n"); + print_set_color(PRINT_INDEX_7, PRINT_INDEX_0); + current_timezone++; + } + } + needs_redraw = 0; + } + + while (!check_keyboard()) { /* Do nothing */ } + + unsigned char sc = read_scan_code(); + if (sc == 0x48) { // Up Arrow + if (selected_timezone > 0) { + selected_timezone--; + needs_redraw = 1; + } + } else if (sc == 0x50) { // Down Arrow + int max_timezone = 0; + for (int i = 0; i < num_timezones; i++) { + if (strcmp_kernel_date(timezones[i].continent, continent) == 0) { + max_timezone++; + } + } + if (selected_timezone < max_timezone - 1) { + selected_timezone++; + needs_redraw = 1; + } + } else if (scan_code_to_ascii(sc) == '\n' || scan_code_to_ascii(sc) == '\r') { + int current_timezone = 0; + for (int i = 0; i < num_timezones; i++) { + if (strcmp_kernel_date(timezones[i].continent, continent) == 0) { + if (current_timezone == selected_timezone) { + *timezone_offset_h = timezones[i].offset_h; // DST + *timezone_offset_m = timezones[i].offset_m; + return; + } + current_timezone++; + } + } + } + brewing_date(10000000); + } +} + +static void select_continent_date(int* timezone_offset_h, int* timezone_offset_m) { + const char* continents[] = {"North America", "South America", "Europe", "Asia", "Oceania", "Africa"}; + int num_continents = sizeof(continents) / sizeof(char*); + int selected_continent = 0; + int needs_redraw = 1; + + while (1) { + if (needs_redraw) { + print_clear(); + brew_str("Select a continent:\n"); + for (int i = 0; i < num_continents; i++) { + if (i == selected_continent) { + print_set_color(PRINT_INDEX_0, PRINT_INDEX_7); + } + brew_str(continents[i]); + brew_str("\n"); + print_set_color(PRINT_INDEX_7, PRINT_INDEX_0); + } + needs_redraw = 0; + } + + while (!check_keyboard()) { /* Do nothing */ } + + unsigned char sc = read_scan_code(); + if (sc == 0x48) { // Up Arrow + if (selected_continent > 0) { + selected_continent--; + needs_redraw = 1; + } + } else if (sc == 0x50) { // Down Arrow + if (selected_continent < num_continents - 1) { + selected_continent++; + needs_redraw = 1; + } + } else if (scan_code_to_ascii(sc) == '\n' || scan_code_to_ascii(sc) == '\r') { + select_timezone_for_continent_date(continents[selected_continent], timezone_offset_h, timezone_offset_m); + return; + } + brewing_date(10000000); + } +} + +// Function to print a zero-padded integer +static void brew_int_padded_date(int n) { + if (n >= 0 && n < 10) { + print_char('0'); + } + brew_int(n); +} + +static void date_command(int* timezone_offset_h, int* timezone_offset_m) { + select_continent_date(timezone_offset_h, timezone_offset_m); + int year, month, day, hour, minute, second; + get_datetime(&year, &month, &day, &hour, &minute, &second); + + // Apply timezone offset + hour += *timezone_offset_h; + minute += *timezone_offset_m; + + // Handle minute overflow/underflow + if (minute >= 60) { + hour++; + minute -= 60; + } else if (minute < 0) { + hour--; + minute += 60; + } + + // Handle hour overflow/underflow + if (hour >= 24) { + day++; + hour -= 24; + } else if (hour < 0) { + day--; + hour += 24; + } + + // Note: This doesn't handle day/month/year wrapping correctly. It's a simplified implementation. + + brew_str("\nCurrent Date and Time:\n"); + brew_int(year); + brew_str("-"); + brew_int_padded_date(month); + brew_str("-"); + brew_int_padded_date(day); + brew_str(" "); + brew_int_padded_date(hour); + brew_str(":"); + brew_int_padded_date(minute); + brew_str(":"); + brew_int_padded_date(second); + brew_str("\n"); +} + +#endif // DATE_H diff --git a/src/kernel/cli_apps_old/help.h b/src/kernel/cli_apps_old/help.h new file mode 100644 index 0000000..8ff6b7a --- /dev/null +++ b/src/kernel/cli_apps_old/help.h @@ -0,0 +1,54 @@ +/* + * Brew Kernel + * Copyright (C) 2024-2026 boreddevnl + * + * 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 3 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. If not, see . + */ +#ifndef APPS_HELP_H +#define APPS_HELP_H + +#include "../print.h" + +static void display_help() { + brew_str("\nAvailable commands:\n"); + brew_str(" HELP - Display this help message\n"); + brew_str(" DATE - Display the current date and time\n"); + brew_str(" EXIT - Exit CLI mode and return to regular typing\n"); + brew_str(" CLEAR - Clear the screen\n"); + brew_str(" ABOUT - Display system information\n"); + brew_str(" MATH - Perform basic arithmetic\n"); + brew_str(" MAN - Show the detailed user manual\n"); + brew_str(" LICENSE - Display the GNU GPLv3 license\n"); + brew_str(" UPTIME - Show how long the system has been running\n"); + brew_str(" MEMORY - Display memory usage statistics\n"); + brew_str(" BEEP - Makes a beep sound using the PC speaker\n"); + brew_str(" TXTEDIT - Open the text editor\n"); + brew_str(" COWSAY. - MOO!\n"); + brew_str(" LS - List files in current directory\n"); + brew_str(" CD - Change current directory\n"); + brew_str(" PWD - Print working directory\n"); + brew_str(" MKDIR - Create one or more directories\n"); + brew_str(" RM - Remove a file or empty directory\n"); + brew_str(" CAT - Display file contents\n"); + brew_str(" TOUCH - Create an empty file\n"); + brew_str(" ECHO - Print text (can redirect to file with >)\n"); + brew_str(" NETINIT - Initialize network card\n"); + brew_str(" NETINFO - Show network status (MAC, IP)\n"); + brew_str(" UDPTEST - Start UDP echo server on port 12345 (broken)\n"); + brew_str(" UDPSEND - Send UDP packet (UDPSEND )\n"); + brew_str("\nPipe support:\n"); + brew_str(" CAT | UDPSEND - Send file contents via UDP\n"); +} + +#endif // APPS_HELP_H diff --git a/src/kernel/cli_apps_old/license.h b/src/kernel/cli_apps_old/license.h new file mode 100644 index 0000000..170f195 --- /dev/null +++ b/src/kernel/cli_apps_old/license.h @@ -0,0 +1,738 @@ +/* + * Brew Kernel + * Copyright (C) 2024-2026 boreddevnl + * + * 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 3 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. If not, see . + */ +#ifndef APPS_LICENSE_H +#define APPS_LICENSE_H + +#include "print.h" +#include "keyboard.h" + +const char* license_pages[] = { + " GNU GENERAL PUBLIC LICENSE", + " Version 3, 29 June 2007", + " Copyright (C) 2024-2026 boreddevnl", + "", + " ", + " Copyright (C) 2007 Free Software Foundation, Inc. ", + " Everyone is permitted to copy and distribute verbatim copies", + " of this license document, but changing it is not allowed.", + "", + " Preamble", + "", + " The GNU General Public License is a free, copyleft license for", + "software and other kinds of works.", + "", + " The licenses for most software and other practical works are designed", + "to take away your freedom to share and change the works. By contrast,", + "the GNU General Public License is intended to guarantee your freedom to", + "share and change all versions of a program--to make sure it remains free", + "software for all its users. We, the Free Software Foundation, use the", + "GNU General Public License for most of our software; it applies also to", + "any other work released this way by its authors. You can apply it to", + "your programs, too.", + "", + " When we speak of free software, we are referring to freedom, 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", + "them if you wish), that you receive source code or can get it if you", + "want it, that you can change the software or use pieces of it in new", + "free programs, and that you know you can do these things.", + "", + " To protect your rights, we need to prevent others from denying you", + "these rights or asking you to surrender the rights. Therefore, you have", + "certain responsibilities if you distribute copies of the software, or if", + "you modify it: responsibilities to respect the freedom of others.", + "", + " For example, if you distribute copies of such a program, whether", + "gratis or for a fee, you must pass on to the recipients the same", + "freedoms that you received. You must make sure that they, too, receive", + "or can get the source code. And you must show them these terms so they", + "know their rights.", + "", + " Developers that use the GNU GPL protect your rights with two steps:", + "(1) assert copyright on the software, and (2) offer you this License", + "giving you legal permission to copy, distribute and/or modify it.", + "", + " For the developers' and authors' protection, the GPL clearly explains", + "that there is no warranty for this free software. For both users' and", + "authors' sake, the GPL requires that modified versions be marked as", + "changed, so that their problems will not be attributed erroneously to", + "authors of previous versions.", + "", + " Some devices are designed to deny users access to install or run", + "modified versions of the software inside them, although the manufacturer", + "can do so. This is fundamentally incompatible with the aim of", + "protecting users' freedom to change the software. The systematic", + "pattern of such abuse occurs in the area of products for individuals to", + "use, which is precisely where it is most unacceptable. Therefore, we", + "have designed this version of the GPL to prohibit the practice for those", + "products. If such problems arise substantially in other domains, we", + "stand ready to extend this provision to those domains in future versions", + "of the GPL, as needed to protect the freedom of users.", + "", + " Finally, every program is threatened constantly by software patents.", + "States should not allow patents to restrict development and use of", + "software on general-purpose computers, but in those that do, we wish to", + "avoid the special danger that patents applied to a free program could", + "make it effectively proprietary. To prevent this, the GPL assures that", + "patents cannot be used to render the program non-free.", + "", + " The precise terms and conditions for copying, distribution and", + "modification follow.", + "", + " TERMS AND CONDITIONS", + "", + " 0. Definitions.", + "", + " 'This License' refers to version 3 of the GNU General Public License.", + "", + " 'Copyright' also means copyright-like laws that apply to other kinds of", + "works, such as semiconductor masks.", + "", + " 'The Program' refers to any copyrightable work licensed under this", + "License. Each licensee is addressed as 'you'. 'Licensees' and", + "'recipients' may be individuals or organizations.", + "", + " To 'modify' a work means to copy from or adapt all or part of the work", + "in a fashion requiring copyright permission, other than the making of an", + "exact copy. The resulting work is called a 'modified version' of the", + "earlier work or a work 'based on' the earlier work.", + "", + " A 'covered work' means either the unmodified Program or a work based", + "on the Program.", + "", + " To 'propagate' a work means to do anything with it that, without", + "permission, would make you directly or secondarily liable for", + "infringement under applicable copyright law, except executing it on a", + "computer or modifying a private copy. Propagation includes copying,", + "distribution (with or without modification), making available to the", + "public, and in some countries other activities as well.", + "", + " To 'convey' a work means any kind of propagation that enables other", + "parties to make or receive copies. Mere interaction with a user through", + "a computer network, with no transfer of a copy, is not conveying.", + "", + " An interactive user interface displays 'Appropriate Legal Notices'", + "to the extent that it includes a convenient and prominently visible", + "feature that (1) displays an appropriate copyright notice, and (2)", + "tells the user that there is no warranty for the work (except to the", + "extent that warranties are provided), that licensees may convey the", + "work under this License, and how to view a copy of this License. If", + "the interface presents a list of user commands or options, such as a", + "menu, a prominent item in the list meets this criterion.", + "", + " 1. Source Code.", + "", + " The 'source code' for a work means the preferred form of the work", + "for making modifications to it. 'Object code' means any non-source", + "form of a work.", + "", + " A 'Standard Interface' means an interface that either is an official", + "standard defined by a recognized standards body, or, in the case of", + "interfaces specified for a particular programming language, one that", + "is widely used among developers working in that language.", + "", + " The 'System Libraries' of an executable work include anything, other", + "than the work as a whole, that (a) is included in the normal form of", + "packaging a Major Component, but which is not part of that Major", + "Component, and (b) serves only to enable use of the work with that", + "Major Component, or to implement a Standard Interface for which an", + "implementation is available to the public in source code form. A", + "'Major Component', in this context, means a major essential component", + "(kernel, window system, and so on) of the specific operating system", + "(if any) on which the executable work runs, or a compiler used to", + "produce the work, or an object code interpreter used to run it.", + "", + " The 'Corresponding Source' for a work in object code form means all", + "the source code needed to generate, install, and (for an executable", + "work) run the object code and to modify the work, including scripts to", + "control those activities. However, it does not include the work's", + "System Libraries, or general-purpose tools or generally available free", + "programs which are used unmodified in performing those activities but", + "which are not part of the work. For example, Corresponding Source", + "includes interface definition files associated with source files for", + "the work, and the source code for shared libraries and dynamically", + "linked subprograms that the work is specifically designed to require,", + "such as by intimate data communication or control flow between those", + "subprograms and other parts of the work.", + "", + " The Corresponding Source need not include anything that users", + "can regenerate automatically from other parts of the Corresponding", + "Source.", + "", + " The Corresponding Source for a work in source code form is that", + "same work.", + "", + " 2. Basic Permissions.", + "", + " All rights granted under this License are granted for the term of", + "copyright on the Program, and are irrevocable provided the stated", + "conditions are met. This License explicitly affirms your unlimited", + "permission to run the unmodified Program. The output from running a", + "covered work is covered by this License only if the output, given its", + "content, constitutes a covered work. This License acknowledges your", + "rights of fair use or other equivalent, as provided by copyright law.", + "", + " You may make, run and propagate covered works that you do not", + "convey, without conditions so long as your license otherwise remains", + "in force. You may convey covered works to others for the sole purpose", + "of having them make modifications exclusively for you, or provide you", + "with facilities for running those works, provided that you comply with", + "the terms of this License in conveying all material for which you do", + "not control copyright. Those thus making or running the covered works", + "for you must do so exclusively on your behalf, under your direction", + "and control, on terms that prohibit them from making any copies of", + "your copyrighted material outside their relationship with you.", + "", + " Conveying under any other circumstances is permitted solely under", + "the conditions stated below. Sublicensing is not allowed; section 10", + "makes it unnecessary.", + "", + " 3. Protecting Users' Legal Rights From Anti-Circumvention Law.", + "", + " No covered work shall be deemed part of an effective technological", + "measure under any applicable law fulfilling obligations under article", + "11 of the WIPO copyright treaty adopted on 20 December 1996, or", + "similar laws prohibiting or restricting circumvention of such", + "measures.", + "", + " When you convey a covered work, you waive any legal power to forbid", + "circumvention of technological measures to the extent such circumvention", + "is effected by exercising rights under this License with respect to", + "the covered work, and you disclaim any intention to limit operation or", + "modification of the work as a means of enforcing, against the work's", + "users, your or third parties' legal rights to forbid circumvention of", + "technological measures.", + "", + " 4. Conveying Verbatim Copies.", + "", + " You may convey verbatim copies of the Program's source code as you", + "receive it, in any medium, provided that you conspicuously and", + "appropriately publish on each copy an appropriate copyright notice;", + "keep intact all notices stating that this License and any", + "non-permissive terms added in accord with section 7 apply to the code;", + "keep intact all notices of the absence of any warranty; and give all", + "recipients a copy of this License along with the Program.", + "", + " You may charge any price or no price for each copy that you convey,", + "and you may offer support or warranty protection for a fee.", + "", + " 5. Conveying Modified Source Versions.", + "", + " You may convey a work based on the Program, or the modifications to", + "produce it from the Program, in the form of source code under the", + "terms of section 4, provided that you also meet all of these conditions:", + "", + " a) The work must carry prominent notices stating that you modified", + " it, and giving a relevant date.", + "", + " b) The work must carry prominent notices stating that it is", + " released under this License and any conditions added under section", + " 7. This requirement modifies the requirement in section 4 to", + " 'keep intact all notices'.", + "", + " c) You must license the entire work, as a whole, under this", + " License to anyone who comes into possession of a copy. This", + " License will therefore apply, along with any applicable section 7", + " additional terms, to the whole of the work, and all its parts,", + " regardless of how they are packaged. This License gives no", + " permission to license the work in any other way, but it does not", + " invalidate such permission if you have separately received it.", + "", + " d) If the work has interactive user interfaces, each must display", + " Appropriate Legal Notices; however, if the Program has interactive", + " interfaces that do not display Appropriate Legal Notices, your", + " work need not make them do so.", + "", + " A compilation of a covered work with other separate and independent", + "works, which are not by their nature extensions of the covered work,", + "and which are not combined with it such as to form a larger program,", + "in or on a volume of a storage or distribution medium, is called an", + "'aggregate' if the compilation and its resulting copyright are not", + "used to limit the access or legal rights of the compilation's users", + "beyond what the individual works permit. Inclusion of a covered work", + "in an aggregate does not cause this License to apply to the other", + "parts of the aggregate.", + "", + " 6. Conveying Non-Source Forms.", + "", + " You may convey a covered work in object code form under the terms", + "of sections 4 and 5, provided that you also convey the", + "machine-readable Corresponding Source under the terms of this License,", + "in one of these ways:", + "", + " a) Convey the object code in, or embodied in, a physical product", + " (including a physical distribution medium), accompanied by the", + " Corresponding Source fixed on a durable physical medium", + " customarily used for software interchange.", + "", + " b) Convey the object code in, or embodied in, a physical product", + " (including a physical distribution medium), accompanied by a", + " written offer, valid for at least three years and valid for as", + " long as you offer spare parts or customer support for that product", + " model, to give anyone who possesses the object code either (1) a", + " copy of the Corresponding Source for all the software in the", + " product that is covered by this License, on a durable physical", + " medium customarily used for software interchange, for a price no", + " more than your reasonable cost of physically performing this", + " conveying of source, or (2) access to copy the", + " Corresponding Source from a network server at no charge.", + "", + " c) Convey individual copies of the object code with a copy of the", + " written offer to provide the Corresponding Source. This", + " alternative is allowed only occasionally and noncommercially, and", + " only if you received the object code with such an offer, in accord", + " with subsection 6b.", + "", + " d) Convey the object code by offering access from a designated", + " place (gratis or for a charge), and offer equivalent access to the", + " Corresponding Source in the same way through the same place at no", + " further charge. You need not require recipients to copy the", + " Corresponding Source along with the object code. If the place to", + " copy the object code is a network server, the Corresponding Source", + " may be on a different server (operated by you or a third party)", + " that supports equivalent copying facilities, provided you maintain", + " clear directions next to the object code saying where to find the", + " Corresponding Source. Regardless of what server hosts the", + " Corresponding Source, you remain obligated to ensure that it is", + " available for as long as needed to satisfy these requirements.", + "", + " e) Convey the object code using peer-to-peer transmission, provided", + " you inform other peers where the object code and Corresponding", + " Source of the work are being offered to the general public at no", + " charge under subsection 6d.", + "", + " A separable portion of the object code, whose source code is excluded", + "from the Corresponding Source as a System Library, need not be", + "included in conveying the object code work.", + "", + " A 'User Product' is either (1) a 'consumer product', which means any", + "tangible personal property which is normally used for personal, family,", + "or household purposes, or (2) anything designed or sold for incorporation", + "into a dwelling. In determining whether a product is a consumer product,", + "doubtful cases shall be resolved in favor of coverage. For a particular", + "product received by a particular user, 'normally used' refers to a", + "typical or common use of that class of product, regardless of the status", + "of the particular user or of the way in which the particular user", + "actually uses, or expects or is expected to use, the product. A product", + "is a consumer product regardless of whether the product has substantial", + "commercial, industrial or non-consumer uses, unless such uses represent", + "the only significant mode of use of the product.", + "", + " 'Installation Information' for a User Product means any methods,", + "procedures, authorization keys, or other information required to install", + "and execute modified versions of a covered work in that User Product from", + "a modified version of its Corresponding Source. The information must", + "suffice to ensure that the continued functioning of the modified object", + "code is in no case prevented or interfered with solely because", + "modification has been made.", + "", + " If you convey an object code work under this section in, or with, or", + "specifically for use in, a User Product, and the conveying occurs as", + "part of a transaction in which the right of possession and use of the", + "User Product is transferred to the recipient in perpetuity or for a", + "fixed term (regardless of how the transaction is characterized), the", + "Corresponding Source conveyed under this section must be accompanied", + "by the Installation Information. But this requirement does not apply", + "if neither you nor any third party retains the ability to install", + "modified object code on the User Product (for example, the work has", + "been installed in ROM).", + "", + " The requirement to provide Installation Information does not include a", + "requirement to continue to provide support service, warranty, or updates", + "for a work that has been modified or installed by the recipient, or for", + "the User Product in which it has been modified or installed. Access to a", + "network may be denied when the modification itself materially and", + "adversely affects the operation of the network or violates the rules and", + "protocols for communication across the network.", + "", + " Corresponding Source conveyed, and Installation Information provided,", + "in accord with this section must be in a format that is publicly", + "documented (and with an implementation available to the public in", + "source code form), and must require no special password or key for", + "unpacking, reading or copying.", + "", + " 7. Additional Terms.", + "", + " 'Additional permissions' are terms that supplement the terms of this", + "License by making exceptions from one or more of its conditions.", + "Additional permissions that are applicable to the entire Program shall", + "be treated as though they were included in this License, to the extent", + "that they are valid under applicable law. If additional permissions", + "apply only to part of the Program, that part may be used separately", + "under those permissions, but the entire Program remains governed by", + "this License without regard to the additional permissions.", + "", + " When you convey a copy of a covered work, you may at your option", + "remove any additional permissions from that copy, or from any part of", + "it. (Additional permissions may be written to require their own", + "removal in certain cases when you modify the work.) You may place", + "additional permissions on material, added by you to a covered work,", + "for which you have or can give appropriate copyright permission.", + "", + " Notwithstanding any other provision of this License, for material you", + "add to a covered work, you may (if authorized by the copyright holders of", + "that material) supplement the terms of this License with terms:", + "", + " a) Disclaiming warranty or limiting liability differently from the", + " terms of sections 15 and 16 of this License; or", + "", + " b) Requiring preservation of specified reasonable legal notices or", + " author attributions in that material or in the Appropriate Legal", + " Notices displayed by works containing it; or", + "", + " c) Prohibiting misrepresentation of the origin of that material, or", + " requiring that modified versions of such material be marked in", + " reasonable ways as different from the original version; or", + "", + " d) Limiting the use for publicity purposes of names of licensors or", + " authors of the material; or", + "", + " e) Declining to grant rights under trademark law for use of some", + " trade names, trademarks, or service marks; or", + "", + " f) Requiring indemnification of licensors and authors of that", + " material by anyone who conveys the material (or modified versions of", + " it) with contractual assumptions of liability to the recipient, for", + " any liability that these contractual assumptions directly impose on", + " those licensors and authors.", + "", + " All other non-permissive additional terms are considered 'further", + "restrictions' within the meaning of section 10. If the Program as you", + "received it, or any part of it, contains a notice stating that it is", + "governed by this License along with a term that is a further", + "restriction, you may remove that term. If a license document contains", + "a further restriction but permits relicensing or conveying under this", + "License, you may add to a covered work material governed by the terms", + "of that license document, provided that the further restriction does", + "not survive such relicensing or conveying.", + "", + " If you add terms to a covered work in accord with this section, you", + "must place, in the relevant source files, a statement of the", + "additional terms that apply to those files, or a notice indicating", + "where to find the applicable terms.", + "", + " Additional terms, permissive or non-permissive, may be stated in the", + "form of a separately written license, or stated as exceptions;", + "the above requirements apply either way.", + "", + " 8. Termination.", + "", + " You may not propagate or modify a covered work except as expressly", + "provided under this License. Any attempt otherwise to propagate or", + "modify it is void, and will automatically terminate your rights under", + "this License (including any patent licenses granted under the third", + "paragraph of section 11).", + "", + " However, if you cease all violation of this License, then your", + "license from a particular copyright holder is reinstated (a)", + "provisionally, unless and until the copyright holder explicitly and", + "finally terminates your license, and (b) permanently, if the copyright", + "holder fails to notify you of the violation by some reasonable means", + "prior to 60 days after the cessation.", + "", + " Moreover, your license from a particular copyright holder is", + "reinstated permanently if the copyright holder notifies you of the", + "violation by some reasonable means, this is the first time you have", + "received notice of violation of this License (for any work) from that", + "copyright holder, and you cure the violation prior to 30 days after", + "your receipt of the notice.", + "", + " Termination of your rights under this section does not terminate the", + "licenses of parties who have received copies or rights from you under", + "this License. If your rights have been terminated and not permanently", + "reinstated, you do not qualify to receive new licenses for the same", + "material under section 10.", + "", + " 9. Acceptance Not Required for Having Copies.", + "", + " You are not required to accept this License in order to receive or", + "run a copy of the Program. Ancillary propagation of a covered work", + "occurring solely as a consequence of using peer-to-peer transmission", + "to receive a copy likewise does not require acceptance. However,", + "nothing other than this License grants you permission to propagate or", + "modify any covered work. These actions infringe copyright if you do", + "not accept this License. Therefore, by modifying or propagating a", + "covered work, you indicate your acceptance of this License to do so.", + "", + " 10. Automatic Licensing of Downstream Recipients.", + "", + " Each time you convey a covered work, the recipient automatically", + "receives a license from the original licensors, to run, modify and", + "propagate that work, subject to this License. You are not responsible", + "for enforcing compliance by third parties with this License.", + "", + " An 'entity transaction' is a transaction transferring control of an", + "organization, or substantially all assets of one, or subdividing an", + "organization, or merging organizations. If propagation of a covered", + "work results from an entity transaction, each party to that", + "transaction who receives a copy of the work also receives whatever", + "licenses to the work the party's predecessor in interest had or could", + "give under the previous paragraph, plus a right to possession of the", + "Corresponding Source of the work from the predecessor in interest, if", + "the predecessor has it or can get it with reasonable efforts.", + "", + " You may not impose any further restrictions on the exercise of the", + "rights granted or affirmed under this License. For example, you may", + "not impose a license fee, royalty, or other charge for exercise of", + "rights granted under this License, and you may not initiate litigation", + "(including a cross-claim or counterclaim in a lawsuit) alleging that", + "any patent claim is infringed by making, using, selling, offering for", + "sale, or importing the Program or any portion of it.", + "", + " 11. Patents.", + "", + " A 'contributor' is a copyright holder who authorizes use under this", + "License of the Program or a work on which the Program is based. The", + "work thus licensed is called the contributor's 'contributor version'.", + "", + " A contributor's 'essential patent claims' are all patent claims", + "owned or controlled by the contributor, whether already acquired or", + "hereafter acquired, that would be infringed by some manner, permitted", + "by this License, of making, using, or selling its contributor version,", + "but do not include claims that would be infringed only as a", + "consequence of further modification of the contributor version. For", + "purposes of this definition, 'control' includes the right to grant", + "patent sublicenses in a manner consistent with the requirements of", + "this License.", + "", + " Each contributor grants you a non-exclusive, worldwide, royalty-free", + "patent license under the contributor's essential patent claims, to", + "make, use, sell, offer for sale, import and otherwise run, modify and", + "propagate the contents of its contributor version.", + "", + " In the following three paragraphs, a 'patent license' is any express", + "agreement or commitment, however denominated, not to enforce a patent", + "(such as an express permission to practice a patent or covenant not to", + "sue for patent infringement). To 'grant' such a patent license to a", + "party means to make such an agreement or commitment not to enforce a", + "patent against the party.", + "", + " If you convey a covered work, knowingly relying on a patent license,", + "and the Corresponding Source of the work is not available for anyone", + "to copy, free of charge and under the terms of this License, through a", + "publicly available network server or other readily accessible means,", + "then you must either (1) cause the Corresponding Source to be so", + "available, or (2) arrange to deprive yourself of the benefit of the", + "patent license for this particular work, or (3) arrange, in a manner", + "consistent with the requirements of this License, to extend the patent", + "license to downstream recipients. 'Knowingly relying' means you have", + "actual knowledge that, but for the patent license, your conveying the", + "covered work in a country, or your recipient's use of the covered work", + "in a country, would infringe one or more identifiable patents in that", + "country that you have reason to believe are valid.", + "", + " If, pursuant to or in connection with a single transaction or", + "arrangement, you convey, or propagate by procuring conveyance of, a", + "covered work, and grant a patent license to some of the parties", + "receiving the covered work authorizing them to use, propagate, modify", + "or convey a specific copy of the covered work, then the patent license", + "you grant is automatically extended to all recipients of the covered", + "work and works based on it.", + "", + " A patent license is 'discriminatory' if it does not include within", + "the scope of its coverage, prohibits the exercise of, or is", + "conditioned on the non-exercise of one or more of the rights that are", + "specifically granted under this License. You may not convey a covered", + "work if you are a party to an arrangement with a third party that is", + "in the business of distributing software, under which you make payment", + "to the third party based on the extent of your activity of conveying", + "the work, and under which the third party grants, to any of the", + "parties who would receive the covered work from you, a discriminatory", + "patent license (a) in connection with copies of the covered work", + "conveyed by you (or copies made from those copies), or (b) primarily", + "for and in connection with specific products or compilations that", + "contain the covered work, unless you entered into that arrangement,", + "or that patent license was granted, prior to 28 March 2007.", + "", + " Nothing in this License shall be construed as excluding or limiting", + "any implied license or other defenses to infringement that may", + "otherwise be available to you under applicable patent law.", + "", + " 12. No Surrender of Others' Freedom.", + "", + " If 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 convey a", + "covered work so as to satisfy simultaneously your obligations under this", + "License and any other pertinent obligations, then as a consequence you may", + "not convey it at all. For example, if you agree to terms that obligate you", + "to collect a royalty for further conveying from those to whom you convey", + "the Program, the only way you could satisfy both those terms and this", + "License would be to refrain entirely from conveying the Program.", + "", + " 13. Use with the GNU Affero General Public License.", + "", + " Notwithstanding any other provision of this License, you have", + "permission to link or combine any covered work with a work licensed", + "under version 3 of the GNU Affero General Public License into a single", + "combined work, and to convey the resulting work. The terms of this", + "License will continue to apply to the part which is the covered work,", + "but the special requirements of the GNU Affero General Public License,", + "section 13, concerning interaction through a network will apply to the", + "combination as such.", + "", + " 14. Revised Versions of this License.", + "", + " The Free Software Foundation may publish revised and/or new versions of", + "the GNU 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", + "Program specifies that a certain numbered version of the GNU General", + "Public License 'or any later version' applies to it, you have the", + "option of following the terms and conditions either of that numbered", + "version or of any later version published by the Free Software", + "Foundation. If the Program does not specify a version number of the", + "GNU General Public License, you may choose any version ever published", + "by the Free Software Foundation.", + "", + " If the Program specifies that a proxy can decide which future", + "versions of the GNU General Public License can be used, that proxy's", + "public statement of acceptance of a version permanently authorizes you", + "to choose that version for the Program.", + "", + " Later license versions may give you additional or different", + "permissions. However, no additional obligations are imposed on any", + "author or copyright holder as a result of your choosing to follow a", + "later version.", + "", + " 15. Disclaimer of Warranty.", + "", + " THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY", + "APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT", + "HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM '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 PROGRAM", + "IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF", + "ALL NECESSARY SERVICING, REPAIR OR CORRECTION.", + "", + " 16. Limitation of Liability.", + "", + " IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING", + "WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS", + "THE PROGRAM 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 PROGRAM (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 PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),", + "EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF", + "SUCH DAMAGES.", + "", + " 17. Interpretation of Sections 15 and 16.", + "", + " If the disclaimer of warranty and limitation of liability provided", + "above cannot be given local legal effect according to their terms,", + "reviewing courts shall apply local law that most closely approximates", + "an absolute waiver of all civil liability in connection with the", + "Program, unless a warranty or assumption of liability accompanies a", + "copy of the Program in return for a fee.", + "", + " END OF TERMS AND CONDITIONS", + "", + " How to Apply These Terms to Your New Programs", + "", + " If you develop a new program, and you want it to be of the greatest", + "possible use to the public, the best way to achieve this is to make it", + "free software which everyone can redistribute and change under these terms.", + "", + " To do so, attach the following notices to the program. It is safest", + "to attach them to the start of each source file to most effectively", + "state the exclusion of warranty; and each file should have at least", + "the 'copyright' line and a pointer to where the full notice is found.", + "", + " ", + " Copyright (C) ", + "", + " 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 3 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. If not, see .", + "", + "Also add information on how to contact you by electronic and paper mail.", + "", + " If the program does terminal interaction, make it output a short", + "notice like this when it starts in an interactive mode:", + "", + " Copyright (C) ", + " This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.", + " This is free software, and you are welcome to redistribute it", + " under certain conditions; type `show c' for details.", + "", + "The hypothetical commands `show w' and `show c' should show the appropriate", + "parts of the General Public License. Of course, your program's commands", + "might be different; for a GUI interface, you would use an 'about box'.", + "", + " You should also get your employer (if you work as a programmer) or school,", + "if any, to sign a 'copyright disclaimer' for the program, if necessary.", + "For more information on this, and how to apply and follow the GNU GPL, see", + ".", + "", + " The GNU General Public License does not permit incorporating your program", + "into proprietary programs. If your program is a subroutine library, you", + "may consider it more useful to permit linking proprietary applications with", + "the library. If this is what you want to do, use the GNU Lesser General", + "Public License instead of this License. But first, please read", + ".", + "--- End of License ---" +}; +const int license_num_lines = sizeof(license_pages) / sizeof(char*); + +static void show_license() { + int top_line = 0; + int needs_redraw = 1; + + while (1) { + if (needs_redraw) { + print_clear(); + print_set_color(PRINT_INDEX_7, PRINT_INDEX_0); + for (int i = 0; i < 24 && (top_line + i) < license_num_lines; i++) { + brew_str(license_pages[top_line + i]); + brew_str("\n"); + } + print_set_color(PRINT_INDEX_15, PRINT_INDEX_9); + brew_str("-- (Up/Down to scroll, 'q' to quit) --"); + print_set_color(PRINT_INDEX_7, PRINT_INDEX_0); + needs_redraw = 0; + } + + // Wait for a key press before doing anything + while (!check_keyboard()) { /* Do nothing */ } + + unsigned char sc = read_scan_code(); // Now read the key + if (sc == 0x48 && top_line > 0) { top_line--; needs_redraw = 1; } // Up Arrow + else if (sc == 0x50 && (top_line + 24) < license_num_lines) { top_line++; needs_redraw = 1; } // Down Arrow + else if (scan_code_to_ascii(sc) == 'q') { + print_set_color(PRINT_INDEX_7, PRINT_INDEX_0); + break; + } + brewing(10000000); // Delay to handle keyboard typematic rate (key-repeat) + } +} + + +#endif // APPS_MAN_H diff --git a/src/kernel/cli_apps_old/man.h b/src/kernel/cli_apps_old/man.h new file mode 100644 index 0000000..7ff0ca5 --- /dev/null +++ b/src/kernel/cli_apps_old/man.h @@ -0,0 +1,156 @@ +/* + * Brew Kernel + * Copyright (C) 2024-2026 boreddevnl + * + * 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 3 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. If not, see . + */ +#ifndef APPS_MAN_H +#define APPS_MAN_H + +#include "print.h" +#include "keyboard.h" + +const char* manual_pages[] = { + "BrewKernel User Manual", + "----------------------", + "", + "Welcome to the BrewKernel, a simple hobby operating system kernel designed", + "for x86_64 architecture. This manual provides an overview of the system,", + "its features, and available commands.", + "", + "== System Overview ==", + "BrewKernel boots into a VGA text mode display (80x25 characters). It", + "initializes a custom color palette and provides basic keyboard input", + "handling. The primary user interface is a simple command-line", + "interface (CLI).", + "", + "== Features ==", + "* Ramdisk-based Filesystem: A simple in-memory filesystem supporting", + " files and directories.", + "* VGA Text Mode Driver: Full control over text and background colors.", + "* PS/2 Keyboard Driver: Handles key presses, including modifier keys", + " like Shift.", + "* Simple CLI: A basic shell to interact with the kernel.", + "* Command History (sort of): The last entered command remains in the", + " buffer but is not yet a full history feature.", + "", + "== How to Use the CLI ==", + "Upon boot, you can type 'CLI' and press Enter to start the command-line", + "interface. Once in the CLI, you can type commands followed by Enter.", + "Commands are case-insensitive.", + "", + "== Special Keys ==", + "* Up/down arrow keys, scrolls through command history. (CLI mode only)", + "", + "== File System ==", + "The BrewKernel includes a simple ramdisk-based filesystem. You can create,", + "read, write, and list files and directories using the following commands:", + "This filesystem will NOT save to disk and only saves to RAM.", + "This filesystem is UNIX like, using '/' as the directory separator.", + "Commands like ls, cd, mkdir work with absolute and relative paths.", + "", + "== Brew Language==", + "== This version of brewkernel contains a simple interpreter for the", + "== brew programming language. Currently the only implemented function is", + "== brew_str, which prints a string to the screen. More features will be", + "== added in future versions. Brew files have the extension .brew", + "== you can run a brew file using the command: brewer >filename.brew<", + "== Available Commands ==", + "HELP: Displays a short list of available commands.", + "LS [path]: Lists files in the specified directory or current directory if", + " no path is given.", + "", + "CD [path]: Changes the current directory to 'path'.", + "", + "PWD: Prints the current working directory path.", + "", + "MKDIR [path]: Creates a new directory at the specified path.", + "", + "MAN: Shows this detailed user manual. Use UP/DOWN arrow keys to", + " scroll and 'q' to quit the manual viewer.", + "", + "ABOUT: Displays information about the kernel, including version, build", + " date, architecture, and compiler.", + "", + "MATH:", + "A simple calculator for basic arithmetic operations", + "(add, subtract, multiply, divide) on integers.", + + "", + "DATE: Displays the current date and time, with an option to select your", + " timezone.", + "", + "TXTEDIT: A simple text editor. Features:", + " - Create and edit multiple text files", + " - Files are preserved between editor sessions (until reboot)", + " - Navigate with arrow keys", + " - Save/load files with custom names", + " - ESC to exit (with save prompt)", + "USAGE: txtedit >filename< or:", + "txtedit and choose name on save", + "", + "IREADTHEMANUAL: Wow. You actually read the manual. Run this command", + " for a special surprise!", + "", + "CLEAR: Clears the entire screen and moves the cursor to the top-left.", + "", + "EXIT: Exits the CLI mode and returns to the initial kernel screen.", + "", + "LICENSE: Displays the full GNU General Public License v3, under which", + " BrewKernel is distributed. Use UP/DOWN to scroll, 'q' to quit.", + "", + "COWSAY: Moo! Displays a cow saying a message. Usage: COWSAY [message]", + " Inspired by GNU/LINUX", + "", + "UPTIME: Shows how long the system has been running since boot.", + "BEEP: Makes a beep sound using the PC speaker.", + "--- End of Manual ---" +}; +const int manual_num_lines = sizeof(manual_pages) / sizeof(char*); + +static void show_manual() { + int top_line = 0; + int needs_redraw = 1; + + while (1) { + if (needs_redraw) { + print_clear(); + print_set_color(PRINT_INDEX_7, PRINT_INDEX_0); + for (int i = 0; i < 24 && (top_line + i) < manual_num_lines; i++) { + brew_str(manual_pages[top_line + i]); + brew_str("\n"); + } + print_set_color(PRINT_INDEX_15, PRINT_INDEX_9); + brew_str("-- (Up/Down to scroll, 'q' to quit) --"); + print_set_color(PRINT_INDEX_7, PRINT_INDEX_0); + needs_redraw = 0; + } + + // Wait for a key press before doing anything + while (!check_keyboard()) { /* Do nothing */ } + + unsigned char sc = read_scan_code(); // Now read the key + if (sc == 0x48 && top_line > 0) { top_line--; needs_redraw = 1; } // Up Arrow + else if (sc == 0x50 && (top_line + 24) < manual_num_lines) { top_line++; needs_redraw = 1; } // Down Arrow + else if (scan_code_to_ascii(sc) == 'q') { + print_set_color(PRINT_INDEX_7, PRINT_INDEX_0); + break; + } + + brewing(10000000); // Delay to handle keyboard typematic rate (key-repeat) + } +} + + +#endif // APPS_MAN_H diff --git a/src/kernel/cli_apps_old/math.h b/src/kernel/cli_apps_old/math.h new file mode 100644 index 0000000..6b40a29 --- /dev/null +++ b/src/kernel/cli_apps_old/math.h @@ -0,0 +1,131 @@ +/* + * Brew Kernel + * Copyright (C) 2024-2026 boreddevnl + * + * 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 3 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. If not, see . + */ +#ifndef APPS_MATH_H +#define APPS_MATH_H + +#include "print.h" +#include "keyboard.h" + +extern int buffer_pos; + +static void math_cmd() { + brew_str("\nMath Calculator\n"); + brew_str("Choose operation:\n"); + brew_str("1. Addition (+)\n"); + brew_str("2. Subtraction (-)\n"); + brew_str("3. Multiplication (*)\n"); + brew_str("4. Division (/)\n"); + brew_str("\nEnter operation number: "); + + // Clear buffer for new input + buffer_pos = 0; + + // Wait for operation choice + while (1) { + if (check_keyboard()) { + unsigned char scan_code = read_scan_code(); + char ascii_char = scan_code_to_ascii(scan_code); + + if (ascii_char >= '1' && ascii_char <= '4') { + print_char(ascii_char); + char operation = ascii_char; + brew_str("\nEnter first number: "); + + // Get first number + buffer_pos = 0; + int first_num = 0; + while (1) { + if (check_keyboard()) { + scan_code = read_scan_code(); + ascii_char = scan_code_to_ascii(scan_code); + + if (ascii_char >= '0' && ascii_char <= '9') { + print_char(ascii_char); + first_num = first_num * 10 + (ascii_char - '0'); + } else if (ascii_char == '\n' || ascii_char == '\r') { + break; + } + } + } + + brew_str("\nEnter second number: "); + + // Get second number + buffer_pos = 0; + int second_num = 0; + while (1) { + if (check_keyboard()) { + scan_code = read_scan_code(); + ascii_char = scan_code_to_ascii(scan_code); + + if (ascii_char >= '0' && ascii_char <= '9') { + print_char(ascii_char); + second_num = second_num * 10 + (ascii_char - '0'); + } else if (ascii_char == '\n' || ascii_char == '\r') { + break; + } + } + } + + brew_str("\nResult: "); + switch (operation) { + case '1': + brew_str("\n"); + brew_int(first_num); + brew_str(" + "); + brew_int(second_num); + brew_str(" = "); + brew_int(first_num + second_num); + break; + case '2': + brew_str("\n"); + brew_int(first_num); + brew_str(" - "); + brew_int(second_num); + brew_str(" = "); + brew_int(first_num - second_num); + break; + case '3': + brew_str("\n"); + brew_int(first_num); + brew_str(" * "); + brew_int(second_num); + brew_str(" = "); + brew_int(first_num * second_num); + break; + case '4': + if (second_num == 0) { + brew_str("Error: Division by zero!\n"); + } else { + brew_str("\n"); + brew_int(first_num); + brew_str(" / "); + brew_int(second_num); + brew_str(" = "); + brew_int(first_num / second_num); + } + break; + } + brew_str("\n"); + break; + } + } + } + } + +#endif // APPS_MATH_H \ No newline at end of file diff --git a/src/kernel/cli_apps_old/readtheman.h b/src/kernel/cli_apps_old/readtheman.h new file mode 100644 index 0000000..44a72a2 --- /dev/null +++ b/src/kernel/cli_apps_old/readtheman.h @@ -0,0 +1,52 @@ +/* + * Brew Kernel + * Copyright (C) 2024-2026 boreddevnl + * + * 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 3 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. If not, see . + */ +#ifndef APPS_NERD_H +#define APPS_NERD_H + +#include "../io.h" +#include "../print.h" + +static void nerd() { + brew_str("\n"); + brew_str("You read the manual? NERD. you know what?\n"); + brew_str("Fuck you.\n"); + brewing(5000000000); + + + // Set the PIT to the desired frequency (1000 Hz for high pitch) + outb(0x43, 0xB6); // Command byte: channel 2, mode 3, binary + int frequency = 1000; + int divisor = 1193180 / frequency; + outb(0x42, divisor & 0xFF); + outb(0x42, (divisor >> 8) & 0xFF); + + outb(0x61, inb(0x61) | 0x03); + + for(int i = 0; i < 1000000000000000000; i++) { + print_set_color(PRINT_INDEX_0, PRINT_INDEX_15); // black on white + print_clear(); + brewing(500000); + + print_set_color(PRINT_INDEX_15, PRINT_INDEX_0); // white on black + print_clear(); + brewing(500000); + } + +} + +#endif // APPS_NERD_H diff --git a/src/kernel/cli_apps_old/reboot.h b/src/kernel/cli_apps_old/reboot.h new file mode 100644 index 0000000..4fd0bb1 --- /dev/null +++ b/src/kernel/cli_apps_old/reboot.h @@ -0,0 +1,40 @@ +/* + * Brew Kernel + * Copyright (C) 2024-2026 boreddevnl + * + * 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 3 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. If not, see . + */ +#ifndef REBOOT_H +#define REBOOT_H + +#include "../print.h" +#include "../io.h" + +void brewing(int iterations); + +void reboot_command() { + brew_str("\nInitiating system reboot...\n"); + brewing(10000000); + while ((inb(0x64) & 2) != 0) { + brewing(1000); + } + outb(0x64, 0xFE); + brewing(50000000); + asm volatile ("lidt %0" : : "m"(*(char*)0)); + asm volatile ("int $0x3"); + brew_str("WARNING: System reboot failed.\n"); + brew_str("Please reset your computer manually.\n"); +} + +#endif \ No newline at end of file diff --git a/src/kernel/cli_apps_old/shutdown.h b/src/kernel/cli_apps_old/shutdown.h new file mode 100644 index 0000000..b04bf94 --- /dev/null +++ b/src/kernel/cli_apps_old/shutdown.h @@ -0,0 +1,35 @@ +/* + * Brew Kernel + * Copyright (C) 2024-2026 boreddevnl + * + * 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 3 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. If not, see . + */ +#ifndef SHUTDOWN_H +#define SHUTDOWN_H + +#include "../print.h" +#include "../io.h" + +void brewing(int iterations); + +void shutdown_command() { + brew_str("\nInitiating system shutdown...\n"); + brewing(10000000); + outw(0xB004, 0x2000); + outb(0x64, 0xFE); + brew_str("WARNING: System shutdown failed.\n"); + brew_str("It is now safe to turn off your computer.\n"); +} + +#endif \ No newline at end of file diff --git a/src/kernel/cli_apps_old/txtedit.h b/src/kernel/cli_apps_old/txtedit.h new file mode 100644 index 0000000..0adce92 --- /dev/null +++ b/src/kernel/cli_apps_old/txtedit.h @@ -0,0 +1,420 @@ +/* + * Brew Kernel + * Copyright (C) 2024-2026 boreddevnl + * + * 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 3 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. If not, see . + */ + +#ifndef TXTEDIT_H +#define TXTEDIT_H + +#include "../print.h" +#include "../keyboard.h" +#include "../filesys.h" +#include "../file.h" + +#define BUFFER_SIZE 4096 +#define MAX_LINES 25 +#define MAX_LINE_LENGTH 80 + +// Helper string functions for txtedit +static int strcmp_txtedit(const char *s1, const char *s2) { + while (*s1 && (*s1 == *s2)) { + s1++; + s2++; + } + return *(const unsigned char*)s1 - *(const unsigned char*)s2; +} + +static size_t strlen_txtedit(const char* str) { + size_t len = 0; + while (str[len]) len++; + return len; +} + +// Buffer for the current file being edited +static char text_buffer[BUFFER_SIZE]; +static int buffer_size = 0; +static int cursor_pos = 0; +static int cursor_row = 0; +static int cursor_col = 0; +static int scroll_offset = 0; // Line number at top of visible area +static int has_unsaved_changes = 0; // Flag to track if file has been modified +#define ESC_KEY 0x01 +#define ENTER_KEY 0x1C +#define BACKSPACE_KEY 0x0E +#define SCAN_CODE_LEFT_ARROW 0x4B +#define SCAN_CODE_RIGHT_ARROW 0x4D +#define VISIBLE_LINES 23 // Lines 0-22 visible (line 23 is status) + +// Status messages +static const char* MSG_SAVED = "Text saved to RAM buffer"; +static const char* MSG_LOADED = "Text loaded from RAM buffer"; +static const char* MSG_HELP = "ESC:Exit ENTER:NewLine Arrows:Navigate"; + +static void draw_status_line(const char* msg) { + size_t row, col; + print_get_cursor_pos(&row, &col); + print_set_cursor_pos(24, 0); // Bottom line + print_set_color(PRINT_INDEX_0, PRINT_INDEX_7); // Black on light grey + for(int i = 0; i < 80; i++) print_char(' '); // Clear line + print_set_cursor_pos(24, 0); + brew_str(msg); + print_set_cursor_pos(row, col); + print_set_color(PRINT_INDEX_7, PRINT_INDEX_0); // Reset colors to latte +} + +static void get_filename(char* filename) { + int pos = 0; + print_clear(); + brew_str("Enter filename (e.g., file.txt): "); + + while(1) { + if(check_keyboard()) { + unsigned char scan_code = read_scan_code(); + + if(scan_code == 0x1C) { // Enter + filename[pos] = '\0'; + break; + } + else if(scan_code == 0x0E) { // Backspace + if(pos > 0) { + pos--; + print_backspace(); + } + } + else { + char c = scan_code_to_ascii(scan_code); + if(c && pos < FS_MAX_FILENAME - 1) { + filename[pos++] = c; + print_char(c); + } + } + } + } +} + +static char current_filename[FS_MAX_FILENAME]; + +static void save_current_buffer(const char* text_buffer, int buffer_size, bool prompt_for_filename) { + // If we don't have a current filename and need to prompt, ask for one + if (current_filename[0] == '\0' && prompt_for_filename) { + get_filename(current_filename); + } else if (current_filename[0] == '\0') { + draw_status_line("Error: No filename specified"); + return; + } + + // Create a new file in the current directory + File* file = fs_find_file(current_filename); + if (!file) { + file = fs_create_file(current_filename); + } + + if (file && file_write_content(file, text_buffer, buffer_size)) { + draw_status_line("File saved successfully"); + } else { + draw_status_line("Error: Could not save file"); + } +} + +static void calculate_cursor_position() { + // Calculate cursor position based on buffer position + cursor_row = 0; + cursor_col = 0; + int current_line = 0; + + for(int i = 0; i < cursor_pos && i < buffer_size; i++) { + if(text_buffer[i] == '\n') { + current_line++; + cursor_col = 0; + } else { + cursor_col++; + } + } + + // Adjust display row based on scroll offset + cursor_row = current_line - scroll_offset; + + // Auto-scroll up if cursor goes above visible area + if(cursor_row < 0) { + scroll_offset += cursor_row; + cursor_row = 0; + } + + // Auto-scroll down if cursor goes below visible area + if(cursor_row >= VISIBLE_LINES) { + scroll_offset += (cursor_row - VISIBLE_LINES + 1); + cursor_row = VISIBLE_LINES - 1; + } +} + +static void redraw_screen() { + print_disable_cursor(); + + print_clear(); + print_set_cursor_pos(0, 0); + + // Draw the text buffer, starting from scroll_offset + int current_line = 0; + int display_line = 0; + + for(int i = 0; i < buffer_size && display_line < VISIBLE_LINES; i++) { + // Skip lines until we reach scroll_offset + if(current_line < scroll_offset) { + if(text_buffer[i] == '\n') { + current_line++; + } + continue; + } + + // We're now in the visible area + if(text_buffer[i] == '\n') { + print_char('\n'); + current_line++; + display_line++; + if(display_line >= VISIBLE_LINES) break; + } else { + print_char(text_buffer[i]); + } + } + + // Calculate cursor position based on cursor_pos in buffer + calculate_cursor_position(); + + draw_status_line(MSG_HELP); + // Position cursor correctly and re-enable it + print_set_cursor_pos(cursor_row, cursor_col); + print_enable_cursor(); +} + +static void insert_char(char c) { + if(buffer_size >= BUFFER_SIZE - 1) return; + + // Move following text forward + for(int i = buffer_size; i > cursor_pos; i--) { + text_buffer[i] = text_buffer[i-1]; + } + + text_buffer[cursor_pos] = c; + cursor_pos++; + buffer_size++; + has_unsaved_changes = 1; // Mark file as modified + + // Redraw the screen to properly display the inserted character + redraw_screen(); +} + +static void delete_char() { + if(cursor_pos <= 0) return; + + cursor_pos--; + // Move following text backward + for(int i = cursor_pos; i < buffer_size - 1; i++) { + text_buffer[i] = text_buffer[i+1]; + } + buffer_size--; + has_unsaved_changes = 1; // Mark file as modified + + // Redraw the screen to properly display after deletion + redraw_screen(); +} + +static void handle_special_key(unsigned char scan_code) { + switch(scan_code) { + case SCAN_CODE_UP_ARROW: { + // Find the start of the current line + int line_start = cursor_pos; + while(line_start > 0 && text_buffer[line_start - 1] != '\n') { + line_start--; + } + + // If we're on the first line, can't go up + if(line_start == 0) break; + + // Find the start of the previous line + int prev_line_start = line_start - 1; + while(prev_line_start > 0 && text_buffer[prev_line_start - 1] != '\n') { + prev_line_start--; + } + + // Calculate target column position + int target_col = cursor_pos - line_start; + + // Find the end of the previous line + int prev_line_end = prev_line_start; + while(prev_line_end < line_start - 1 && text_buffer[prev_line_end] != '\n') { + prev_line_end++; + } + + // Move cursor to same column in previous line (or end of line if shorter) + int prev_line_length = prev_line_end - prev_line_start; + cursor_pos = prev_line_start + (target_col < prev_line_length ? target_col : prev_line_length); + calculate_cursor_position(); + redraw_screen(); + break; + } + + case SCAN_CODE_DOWN_ARROW: { + // Find the start of the current line + int line_start = cursor_pos; + while(line_start > 0 && text_buffer[line_start - 1] != '\n') { + line_start--; + } + + // Find the end of the current line + int line_end = cursor_pos; + while(line_end < buffer_size && text_buffer[line_end] != '\n') { + line_end++; + } + + // If we're on the last line, can't go down + if(line_end >= buffer_size) break; + + // Move to start of next line + int next_line_start = line_end + 1; + if(next_line_start >= buffer_size) break; + + // Find the end of the next line + int next_line_end = next_line_start; + while(next_line_end < buffer_size && text_buffer[next_line_end] != '\n') { + next_line_end++; + } + + // Calculate target column position + int target_col = cursor_pos - line_start; + + // Move cursor to same column in next line (or end of line if shorter) + int next_line_length = next_line_end - next_line_start; + cursor_pos = next_line_start + (target_col < next_line_length ? target_col : next_line_length); + calculate_cursor_position(); + redraw_screen(); + break; + } + + case SCAN_CODE_LEFT_ARROW: + if(cursor_pos > 0) { + cursor_pos--; + calculate_cursor_position(); + redraw_screen(); + } + break; + + case SCAN_CODE_RIGHT_ARROW: + if(cursor_pos < buffer_size) { + cursor_pos++; + calculate_cursor_position(); + redraw_screen(); + } + break; + } +} + +static void load_file(const char* filename) { + File* file = fs_find_file(filename); + if (!file) { + draw_status_line("Creating new file"); + return; + } + + size_t size; + const char* content = file_get_content(file, &size); + if (!content || size > BUFFER_SIZE) { + draw_status_line("Error: Could not load file or file too large"); + return; + } + + // Copy content to buffer + for (size_t i = 0; i < size; i++) { + text_buffer[i] = content[i]; + } + buffer_size = size; + cursor_pos = 0; // Start at beginning of file + has_unsaved_changes = 0; // File is not modified after loading + draw_status_line("File loaded successfully"); +} + +void txtedit_run(const char* filename) { + print_set_color(PRINT_INDEX_7, PRINT_INDEX_0); // Set initial color to latte + + // Clear the buffers and filename + for(int i = 0; i < BUFFER_SIZE; i++) { + text_buffer[i] = 0; + } + buffer_size = 0; + cursor_pos = 0; + scroll_offset = 0; + has_unsaved_changes = 0; + current_filename[0] = '\0'; + + // If filename is provided, store it + if (filename) { + int i; + for (i = 0; filename[i] && i < FS_MAX_FILENAME - 1; i++) { + current_filename[i] = filename[i]; + } + current_filename[i] = '\0'; + load_file(filename); + } + + print_clear(); + redraw_screen(); + print_enable_cursor(); + + while(1) { + if(check_keyboard()) { + unsigned char scan_code = read_scan_code(); + + if(scan_code == ESC_KEY) { + // Only ask to save if there are unsaved changes + if(has_unsaved_changes) { + draw_status_line("Save before exit? (Y/N)"); + while(1) { + if(check_keyboard()) { + unsigned char save_scan = read_scan_code(); + char save_char = scan_code_to_ascii(save_scan); + if(save_char == 'Y' || save_char == 'y') { + // Only prompt for filename if we don't have one (started with txtedit with no args) + save_current_buffer(text_buffer, buffer_size, current_filename[0] == '\0'); + break; + } + else if(save_char == 'N' || save_char == 'n') { + break; + } + } + } + } + break; + } + else if(scan_code == ENTER_KEY) { + insert_char('\n'); + } + else if(scan_code == BACKSPACE_KEY) { + delete_char(); + } + else if(scan_code == SCAN_CODE_UP_ARROW || scan_code == SCAN_CODE_DOWN_ARROW || + scan_code == SCAN_CODE_LEFT_ARROW || scan_code == SCAN_CODE_RIGHT_ARROW) { + handle_special_key(scan_code); + } + else { + char c = scan_code_to_ascii(scan_code); + if(c) insert_char(c); + } + } + } + + +} + +#endif // TXTEDIT_H \ No newline at end of file diff --git a/src/kernel/cli_apps_old/uptime.h b/src/kernel/cli_apps_old/uptime.h new file mode 100644 index 0000000..5900012 --- /dev/null +++ b/src/kernel/cli_apps_old/uptime.h @@ -0,0 +1,97 @@ +/* + * Brew Kernel + * Copyright (C) 2024-2026 boreddevnl + * + * 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 3 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. If not, see . + */ +#ifndef UPTIME_H +#define UPTIME_H + +#include "print.h" +#include "rtc.h" + +// Global variable to store boot time +static int boot_time_hour = -1; +static int boot_time_min = -1; +static int boot_time_sec = -1; + +// Initialize boot time +void init_uptime(void) { + if (boot_time_hour == -1) { + int year, month, day; + get_datetime(&year, &month, &day, &boot_time_hour, &boot_time_min, &boot_time_sec); + } +} + +// Function to print a zero-padded integer +static void brew_int_padded_uptime(int n) { + if (n >= 0 && n < 10) { + print_char('0'); + } + brew_int(n); +} + +// Calculate time difference considering wraparound +static void calc_time_diff(int start_h, int start_m, int start_s, + int curr_h, int curr_m, int curr_s, + int *diff_h, int *diff_m, int *diff_s) { + int total_start_secs = start_h * 3600 + start_m * 60 + start_s; + int total_curr_secs = curr_h * 3600 + curr_m * 60 + curr_s; + + // Handle day wraparound + if (total_curr_secs < total_start_secs) { + total_curr_secs += 24 * 3600; // Add a full day of seconds + } + + int diff_secs = total_curr_secs - total_start_secs; + + *diff_h = diff_secs / 3600; + diff_secs %= 3600; + *diff_m = diff_secs / 60; + *diff_s = diff_secs % 60; +} + +// Display uptime command +static void display_uptime(void) { + init_uptime(); // Ensure boot time is initialized + + int curr_hour, curr_min, curr_sec; + int year, month, day; + get_datetime(&year, &month, &day, &curr_hour, &curr_min, &curr_sec); + + int up_hours, up_minutes, up_seconds; + calc_time_diff(boot_time_hour, boot_time_min, boot_time_sec, + curr_hour, curr_min, curr_sec, + &up_hours, &up_minutes, &up_seconds); + + brew_str("\nSystem uptime: "); + if (up_hours > 0) { + brew_int(up_hours); + brew_str(" hour"); + if (up_hours != 1) brew_str("s"); + brew_str(" "); + } + if (up_minutes > 0 || up_hours > 0) { + brew_int(up_minutes); + brew_str(" minute"); + if (up_minutes != 1) brew_str("s"); + brew_str(" "); + } + brew_int(up_seconds); + brew_str(" second"); + if (up_seconds != 1) brew_str("s"); + brew_str("\n"); +} + +#endif // UPTIME_H \ No newline at end of file diff --git a/src/kernel/cmd.c b/src/kernel/cmd.c new file mode 100644 index 0000000..ecd6d92 --- /dev/null +++ b/src/kernel/cmd.c @@ -0,0 +1,711 @@ +#include "cmd.h" +#include "graphics.h" +#include "wm.h" +#include "io.h" +#include "rtc.h" +#include "notepad.h" +#include "calculator.h" +#include "fat32.h" +#include "cli_apps/cli_apps.h" +#include "licensewr.h" +#include +#include "memory_manager.h" +#include +#include +#include + +#define CMD_COLS 70 +#define CMD_ROWS 25 +#define LINE_HEIGHT 10 +#define CHAR_WIDTH 8 +#define PROMPT "> " + +#define COLOR_RED 0xFFFF0000 + +#define TXT_BUFFER_SIZE 4096 +#define TXT_VISIBLE_LINES (CMD_ROWS - 2) + +#define FS_MAX_FILES 16 +#define FS_MAX_FILENAME 64 +#define FS_MAX_SIZE 4096 + +typedef struct { + char name[FS_MAX_FILENAME]; + char content[FS_MAX_SIZE]; + int size; + bool used; +} RamFile; + +static RamFile ram_fs[FS_MAX_FILES]; + +static void fs_init() { + for (int i = 0; i < FS_MAX_FILES; i++) { + ram_fs[i].used = false; + ram_fs[i].size = 0; + } +} + +static RamFile* fs_find(const char *name) { + for (int i = 0; i < FS_MAX_FILES; i++) { + if (ram_fs[i].used) { + // Simple strcmp + const char *a = ram_fs[i].name; + const char *b = name; + bool match = true; + while (*a && *b) { + if (*a != *b) { match = false; break; } + a++; b++; + } + if (match && *a == *b) return &ram_fs[i]; + } + } + return NULL; +} + +static RamFile* fs_create(const char *name) { + if (fs_find(name)) return fs_find(name); + for (int i = 0; i < FS_MAX_FILES; i++) { + if (!ram_fs[i].used) { + ram_fs[i].used = true; + int j = 0; + while (name[j] && j < FS_MAX_FILENAME - 1) { + ram_fs[i].name[j] = name[j]; + j++; + } + ram_fs[i].name[j] = 0; + ram_fs[i].size = 0; + return &ram_fs[i]; + } + } + return NULL; +} + +// --- Structs --- +typedef struct { + char c; + uint32_t color; +} CharCell; + +typedef enum { + MODE_SHELL, + MODE_PAGER +} CmdMode; + +// --- State --- +Window win_cmd; + +// Shell State +static CharCell screen_buffer[CMD_ROWS][CMD_COLS]; +static int cursor_row = 0; +static int cursor_col = 0; +static uint32_t current_color = COLOR_LTGRAY; + +// Pager State +static CmdMode current_mode = MODE_SHELL; +static char pager_wrapped_lines[2000][CMD_COLS + 1]; +static int pager_total_lines = 0; +static int pager_top_line = 0; + +// Boot time for uptime +int boot_time_init = 0; +int boot_year, boot_month, boot_day, boot_hour, boot_min, boot_sec; + +// --- Helpers --- +static void cmd_memset(void *dest, int val, size_t len) { + unsigned char *ptr = dest; + while (len-- > 0) *ptr++ = val; +} + +static size_t cmd_strlen(const char *str) { + size_t len = 0; + while (str[len]) len++; + return len; +} + +static int cmd_strcmp(const char *s1, const char *s2) { + while (*s1 && (*s1 == *s2)) { + s1++; + s2++; + } + return *(const unsigned char*)s1 - *(const unsigned char*)s2; +} + +static void cmd_strcpy(char *dest, const char *src) { + while (*src) *dest++ = *src++; + *dest = 0; +} + +static int cmd_atoi(const char *str) { + int res = 0; + int sign = 1; + if (*str == '-') { sign = -1; str++; } + while (*str >= '0' && *str <= '9') { + res = res * 10 + (*str - '0'); + str++; + } + return res * sign; +} + +static void brewing(int iterations) { + for (volatile int i = 0; i < iterations; i++) { + __asm__ __volatile__("nop"); + } +} + +static void itoa(int n, char *buf) { + if (n == 0) { + buf[0] = '0'; buf[1] = 0; return; + } + int i = 0; + int sign = n < 0; + if (sign) n = -n; + while (n > 0) { + buf[i++] = (n % 10) + '0'; + n /= 10; + } + if (sign) buf[i++] = '-'; + buf[i] = 0; + // Reverse + for (int j = 0; j < i / 2; j++) { + char t = buf[j]; + buf[j] = buf[i - 1 - j]; + buf[i - 1 - j] = t; + } +} + +// Manual and license pages are now in the individual command files + +// --- Terminal Emulation --- + +static void cmd_scroll_up() { + for (int r = 1; r < CMD_ROWS; r++) { + for (int c = 0; c < CMD_COLS; c++) { + screen_buffer[r - 1][c] = screen_buffer[r][c]; + } + } + // Clear bottom row + for (int c = 0; c < CMD_COLS; c++) { + screen_buffer[CMD_ROWS - 1][c].c = ' '; + screen_buffer[CMD_ROWS - 1][c].color = current_color; + } +} + + +// Public for CLI apps to use +void cmd_putchar(char c) { + if (c == '\n') { + cursor_col = 0; + cursor_row++; + } else if (c == '\b') { + if (cursor_col > 0) { + cursor_col--; + screen_buffer[cursor_row][cursor_col].c = ' '; + } + } else { + if (cursor_col >= CMD_COLS) { + cursor_col = 0; + cursor_row++; + } + + if (cursor_row >= CMD_ROWS) { + cmd_scroll_up(); + cursor_row = CMD_ROWS - 1; + } + + screen_buffer[cursor_row][cursor_col].c = c; + screen_buffer[cursor_row][cursor_col].color = current_color; + cursor_col++; + } + + if (cursor_row >= CMD_ROWS) { + cmd_scroll_up(); + cursor_row = CMD_ROWS - 1; + } +} + +// Public for CLI apps to use +void cmd_write(const char *str) { + while (*str) { + cmd_putchar(*str++); + } +} + +// Public for CLI apps to use +void cmd_write_int(int n) { + char buf[32]; + itoa(n, buf); + cmd_write(buf); +} + +// --- Pager Logic --- + +// Public for CLI apps to use - clear the terminal screen +void cmd_screen_clear() { + for(int r=0; r= 2000) break; + + int remaining = len - processed; + int chunk_len = remaining; + if (chunk_len > CMD_COLS) chunk_len = CMD_COLS; + + // If we are cutting a word, backtrack to last space + if (chunk_len < remaining) { // Only check if we are actually wrapping + int split_point = chunk_len; + while (split_point > 0 && line[processed + split_point] != ' ') { + split_point--; + } + + if (split_point > 0) { + chunk_len = split_point; // Cut at space + } + // If split_point == 0, the word is longer than the line, so forced split is okay. + } + + // Copy chunk + for (int k = 0; k < chunk_len; k++) { + pager_wrapped_lines[pager_total_lines][k] = line[processed + k]; + } + pager_wrapped_lines[pager_total_lines][chunk_len] = 0; + + pager_total_lines++; + processed += chunk_len; + + // Skip the space we just split on + if (processed < len && line[processed] == ' ') { + processed++; + } + } + } +} + +// Public for CLI apps to use +void pager_set_mode(void) { + current_mode = MODE_PAGER; +} + +// --- Commands (now delegated to cli_apps/) --- + +// Command dispatch table +typedef struct { + const char *name; + void (*func)(char *args); +} CommandEntry; + +static const CommandEntry commands[] = { + {"HELP", cli_cmd_help}, + {"help", cli_cmd_help}, + {"DATE", cli_cmd_date}, + {"date", cli_cmd_date}, + {"CLEAR", cli_cmd_clear}, + {"clear", cli_cmd_clear}, + {"ABOUT", cli_cmd_about}, + {"about", cli_cmd_about}, + {"MATH", cli_cmd_math}, + {"math", cli_cmd_math}, + {"MAN", cli_cmd_man}, + {"man", cli_cmd_man}, + {"LICENSE", cli_cmd_license}, + {"license", cli_cmd_license}, + {"TXTEDIT", cli_cmd_txtedit}, + {"txtedit", cli_cmd_txtedit}, + {"UPTIME", cli_cmd_uptime}, + {"uptime", cli_cmd_uptime}, + {"BEEP", cli_cmd_beep}, + {"beep", cli_cmd_beep}, + {"COWSAY", cli_cmd_cowsay}, + {"cowsay", cli_cmd_cowsay}, + {"REBOOT", cli_cmd_reboot}, + {"reboot", cli_cmd_reboot}, + {"SHUTDOWN", cli_cmd_shutdown}, + {"shutdown", cli_cmd_shutdown}, + {"IREADTHEMANUAL", cli_cmd_readtheman}, + {"ireadthemanual", cli_cmd_readtheman}, + {"BLIND", cli_cmd_blind}, + {"blind", cli_cmd_blind}, + {"EXIT", cli_cmd_exit}, + {"exit", cli_cmd_exit}, + // Filesystem Commands + {"CD", cli_cmd_cd}, + {"cd", cli_cmd_cd}, + {"PWD", cli_cmd_pwd}, + {"pwd", cli_cmd_pwd}, + {"LS", cli_cmd_ls}, + {"ls", cli_cmd_ls}, + {"MKDIR", cli_cmd_mkdir}, + {"mkdir", cli_cmd_mkdir}, + {"RM", cli_cmd_rm}, + {"rm", cli_cmd_rm}, + {"ECHO", cli_cmd_echo}, + {"echo", cli_cmd_echo}, + {"CAT", cli_cmd_cat}, + {"cat", cli_cmd_cat}, + // Memory Management Commands + {"MEMINFO", cli_cmd_meminfo}, + {"meminfo", cli_cmd_meminfo}, + {"MALLOC", cli_cmd_malloc}, + {"malloc", cli_cmd_malloc}, + {"FREEMEM", cli_cmd_free_mem}, + {"freemem", cli_cmd_free_mem}, + {"MEMBLOCK", cli_cmd_memblock}, + {"memblock", cli_cmd_memblock}, + {"MEMVALID", cli_cmd_memvalid}, + {"memvalid", cli_cmd_memvalid}, + {"MEMTEST", cli_cmd_memtest}, + {"memtest", cli_cmd_memtest}, + {NULL, NULL} +}; + +// --- Dispatcher --- + +// Buffer for capturing command output +static char pipe_buffer[4096]; +static int pipe_buffer_pos = 0; + +static void pipe_capture_write(const char *str) { + while (*str && pipe_buffer_pos < (int)sizeof(pipe_buffer) - 1) { + pipe_buffer[pipe_buffer_pos++] = *str++; + } +} + +// Execute a single command +static void cmd_exec_single(char *cmd) { + while (*cmd == ' ') cmd++; + if (!*cmd) return; + + // Split cmd and args + char *args = cmd; + while (*args && *args != ' ') args++; + if (*args) { + *args = 0; // Null terminate cmd + args++; // Point to start of args + } + + // Use command dispatch table + for (int i = 0; commands[i].name != NULL; i++) { + if (cmd_strcmp(cmd, commands[i].name) == 0) { + commands[i].func(args); + return; + } + } + + cmd_write("Unknown command: "); + cmd_write(cmd); + cmd_write("\n"); +} + +// Execute command with pipe support +static void cmd_exec(char *cmd) { + // Check for pipe operator + char *pipe_ptr = NULL; + for (int i = 0; cmd[i]; i++) { + if (cmd[i] == '|' && (i == 0 || cmd[i-1] != '>' && cmd[i+1] != '>' )) { + pipe_ptr = &cmd[i]; + break; + } + } + + if (!pipe_ptr) { + // No pipe - execute normally + cmd_exec_single(cmd); + return; + } + + // Split into two commands + *pipe_ptr = 0; + char *second_cmd = pipe_ptr + 1; + + // Execute first command with output captured + pipe_buffer_pos = 0; + cmd_memset(pipe_buffer, 0, sizeof(pipe_buffer)); + + FAT32_FileHandle *pipe_file = fat32_open("_pipe_temp.tmp", "w"); + if (!pipe_file) { + cmd_write("Error: Cannot create pipe\n"); + return; + } + + cmd_exec_single(cmd); + + fat32_close(pipe_file); + + cmd_exec_single(second_cmd); +} + + +// --- Window Functions --- + +static void cmd_paint(Window *win) { + // Draw Window Content Background + int offset_x = win->x + 4; + int offset_y = win->y + 24; + + // Fill background + draw_rect(offset_x, offset_y, win->w - 8, win->h - 28, COLOR_BLACK); + + int start_y = offset_y + 4; + int start_x = offset_x + 4; + + if (current_mode == MODE_PAGER) { + // Draw Pager Content (Wrapped) + for (int i = 0; i < CMD_ROWS && (pager_top_line + i) < pager_total_lines; i++) { + draw_string(start_x, start_y + (i * LINE_HEIGHT), pager_wrapped_lines[pager_top_line + i], COLOR_LTGRAY); + } + + // Status Bar + draw_string(start_x, start_y + (CMD_ROWS * LINE_HEIGHT), "-- Press Q to quit --", COLOR_WHITE); + + } else { + // Draw Shell Buffer + for (int r = 0; r < CMD_ROWS; r++) { + for (int c = 0; c < CMD_COLS; c++) { + char ch = screen_buffer[r][c].c; + if (ch != 0 && ch != ' ') { + draw_char(start_x + (c * CHAR_WIDTH), start_y + (r * LINE_HEIGHT), ch, screen_buffer[r][c].color); + } + } + } + + // Draw Cursor + if (win->focused) { + draw_rect(start_x + (cursor_col * CHAR_WIDTH), start_y + (cursor_row * LINE_HEIGHT) + 8, CHAR_WIDTH, 2, COLOR_WHITE); + } + } +} + +static void cmd_key(Window *target, char c) { + (void)target; + if (current_mode == MODE_PAGER) { + if (c == 'q' || c == 'Q') { + current_mode = MODE_SHELL; + } else if (c == 17) { // UP + if (pager_top_line > 0) pager_top_line--; + } else if (c == 18) { // DOWN + if (pager_top_line < pager_total_lines - CMD_ROWS) pager_top_line++; + } + return; + } + + // Shell Mode + if (c == '\n') { // Enter + char cmd_buf[CMD_COLS + 1]; + int len = 0; + int prompt_len = cmd_strlen(PROMPT); + + for (int i = prompt_len; i < CMD_COLS; i++) { + char ch = screen_buffer[cursor_row][i].c; + if (ch == 0) break; + cmd_buf[len++] = ch; + } + while (len > 0 && cmd_buf[len-1] == ' ') len--; + cmd_buf[len] = 0; + + cmd_putchar('\n'); + + cmd_exec(cmd_buf); + + cmd_write(PROMPT); + } else if (c == 17) { // UP + // History not implemented + } else if (c == 18) { // DOWN + + } else if (c == 19) { // LEFT + if (cursor_col > (int)cmd_strlen(PROMPT)) { + cursor_col--; + } + } else if (c == 20) { // RIGHT + if (cursor_col < CMD_COLS - 1) { + cursor_col++; + } + } else if (c == '\b') { // Backspace + if (cursor_col > (int)cmd_strlen(PROMPT)) { + cursor_col--; + screen_buffer[cursor_row][cursor_col].c = ' '; + } + } else { + if (c >= 32 && c <= 126) { + cmd_putchar(c); + } + } +} + +void cmd_reset(void) { + // Reset terminal to fresh state + cmd_screen_clear(); + cmd_write("BrewOS Command Prompt\n"); + cmd_write(PROMPT); +} + +static void create_test_files(void) { + fat32_mkdir("Documents"); + fat32_mkdir("Projects"); + fat32_mkdir("Documents/Important"); + + FAT32_FileHandle *fh = fat32_open("README.md", "w"); + if (fh) { + const char *content = + "BREW OS 1.01 ALPHA\n" + "==================\n\n" + "BREWKERNEL IS NOW BREWOS!\n\n" + "Brewkernel will from now on be deprecated as its core became too messy.\n" + "I have built a less bloated kernel and wrote a DE above it, which is why\n" + "it is now an OS instead of a kernel.\n\n" + "Brew Kernel is a simple x86_64 hobbyist operating system.\n" + "It features a DE (and WM), a FAT32 filesystem, customizable UI and much much more!\n" + "ramdisk-like filesystem.\n\n" + "FEATURES\n" + "--------\n" + "* Brew WM (Window Manager)\n" + "* FAT32 Filesystem\n" + "* 64-bit long mode support\n" + "* Multiboot2 compliant\n" + "* Text editor and file explorer\n" + "* IDT (Interrupt Descriptor Table)\n" + "* Ability to run on actual x86_64 hardware\n" + "* Command-line interface (CLI)\n\n" + "PREREQUISITES\n" + "-------------\n" + "To build BrewOS, you'll need the following tools installed:\n\n" + "* x86_64 ELF Toolchain (x86_64-elf-gcc, x86_64-elf-ld)\n" + "* NASM (Netwide Assembler)\n" + "* xorriso (for creating bootable ISO images)\n" + "* QEMU (optional, for testing in emulator)\n\n" + "On macOS, install via Homebrew:\n" + " brew install x86_64-elf-binutils x86_64-elf-gcc nasm xorriso qemu\n\n" + "BUILDING\n" + "--------\n" + "Simply run 'make' from the project root:\n\n" + " make\n\n" + "This will:\n" + "1. Download Limine v7.0.0 bootloader files (if not present)\n" + "2. Compile all kernel C sources and assembly files\n" + "3. Link the kernel ELF binary\n" + "4. Generate a bootable ISO image (brewos.iso)\n\n" + "Build output:\n" + "* Compiled object files: build/\n" + "* ISO root filesystem: iso_root/\n" + "* Final ISO image: brewos.iso\n\n" + "RUNNING\n" + "-------\n" + "QEMU EMULATION:\n" + "Run in QEMU with:\n" + " make run\n\n" + "Or manually:\n" + " qemu-system-x86_64 -m 2G -serial stdio -cdrom brewos.iso -boot d\n\n" + "RUNNING ON REAL HARDWARE:\n" + "WARNING: This is at YOUR OWN RISK. This software comes with ZERO warranty\n" + "and may break your system.\n\n" + "1. Create bootable USB using Balena Etcher to flash brewos.iso\n" + "2. Enable legacy (BIOS) boot in your system BIOS/UEFI settings\n" + "3. Disable Secure Boot if needed\n" + "4. Insert USB drive and select it in boot menu during startup\n\n" + "Tested Hardware:\n" + "* HP EliteDesk 705 G4 DM (AMD Ryzen 5 PRO 2400G, Radeon Vega)\n" + "* Lenovo ThinkPad A475 20KL002VMH (AMD Pro A12-8830B, Radeon R7)\n\n" + "PROJECT STRUCTURE\n" + "-----------------\n" + "* src/kernel/ - Main kernel implementation\n" + " - boot.asm - Boot assembly code\n" + " - main.c - Kernel entry point\n" + " - *.c / *.h - Core kernel modules\n" + " - cli_apps/ - Command-line applications\n" + " - wallpaper.ppm - Default desktop wallpaper\n" + "* build/ - Compiled object files (generated during build)\n" + "* iso_root/ - ISO filesystem layout (generated during build)\n" + "* limine/ - Limine bootloader files (downloaded automatically)\n" + "* linker.ld - Linker script for x86_64 ELF\n" + "* limine.cfg - Limine bootloader configuration\n" + "* Makefile - Build configuration and targets\n\n" + "LICENSE\n" + "-------\n" + "Copyright (C) 2024-2026 boreddevnl\n\n" + "This program is free software: you can redistribute it and/or modify\n" + "it under the terms of the GNU General Public License as published by\n" + "the Free Software Foundation, either version 3 of the License, or\n" + "(at your option) any later version.\n\n" + "For full license details, see the LICENSE file in the repository.\n"; + fat32_write(fh, (void *)content, cmd_strlen(content)); + fat32_close(fh); + } + + write_license_file(); + + fh = fat32_open("Documents/notes.txt", "w"); + if (fh) { + const char *content = "My Notes\n\n- First note\n- Second note\n"; + fat32_write(fh, (void *)content, 39); + fat32_close(fh); + } + + fh = fat32_open("Documents/notes.txt", "w"); + if (fh) { + const char *content = "My Notes\n\n- First note\n- Second note\n"; + fat32_write(fh, (void *)content, 39); + fat32_close(fh); + } + + fh = fat32_open("Projects/project1.txt", "w"); + if (fh) { + const char *content = "Project 1\n\nStatus: In Progress\n"; + fat32_write(fh, (void *)content, 32); + fat32_close(fh); + } +} + +void cmd_init(void) { + fs_init(); // Init RAMFS + fat32_init(); // Init FAT32 filesystem + create_test_files(); + + win_cmd.title = "Command Prompt"; + win_cmd.x = 50; + win_cmd.y = 50; + win_cmd.w = (CMD_COLS * CHAR_WIDTH) + 20; + win_cmd.h = (CMD_ROWS * LINE_HEIGHT) + 40; + + win_cmd.visible = false; + win_cmd.focused = false; + win_cmd.z_index = 0; + win_cmd.paint = cmd_paint; + win_cmd.handle_key = cmd_key; + win_cmd.handle_click = NULL; + win_cmd.handle_right_click = NULL; + + cmd_reset(); + + if (!boot_time_init) { + rtc_get_datetime(&boot_year, &boot_month, &boot_day, &boot_hour, &boot_min, &boot_sec); + boot_time_init = 1; + } +} diff --git a/src/kernel/cmd.h b/src/kernel/cmd.h new file mode 100644 index 0000000..c495792 --- /dev/null +++ b/src/kernel/cmd.h @@ -0,0 +1,11 @@ +#ifndef CMD_H +#define CMD_H + +#include "wm.h" + +extern Window win_cmd; + +void cmd_init(void); +void cmd_reset(void); + +#endif \ No newline at end of file diff --git a/src/kernel/control_panel.c b/src/kernel/control_panel.c new file mode 100644 index 0000000..eacf130 --- /dev/null +++ b/src/kernel/control_panel.c @@ -0,0 +1,329 @@ +#include "control_panel.h" +#include "graphics.h" +#include +#include "wm.h" + +Window win_control_panel; + +#define COLOR_COFFEE 0xFF6B4423 +#define COLOR_TEAL 0xFF008080 +#define COLOR_GREEN 0xFF008000 +#define COLOR_BLUE_BG 0xFF000080 +#define COLOR_PURPLE 0xFF800080 + +// Control panel state +#define VIEW_MAIN 0 +#define VIEW_WALLPAPER 1 + +static int current_view = VIEW_MAIN; +static char rgb_r[4] = ""; +static char rgb_g[4] = ""; +static char rgb_b[4] = ""; +static int focused_field = -1; // -1=none, 0=R, 1=G, 2=B +static int input_cursor = 0; + +static uint32_t parse_rgb_separate(const char *r, const char *g, const char *b) { + int rv = 0, gv = 0, bv = 0; + + // Parse R + for (int i = 0; r[i] && i < 3; i++) { + if (r[i] >= '0' && r[i] <= '9') { + rv = rv * 10 + (r[i] - '0'); + } + } + + // Parse G + for (int i = 0; g[i] && i < 3; i++) { + if (g[i] >= '0' && g[i] <= '9') { + gv = gv * 10 + (g[i] - '0'); + } + } + + // Parse B + for (int i = 0; b[i] && i < 3; i++) { + if (b[i] >= '0' && b[i] <= '9') { + bv = bv * 10 + (b[i] - '0'); + } + } + + // Clamp values + if (rv > 255) rv = 255; + if (gv > 255) gv = 255; + if (bv > 255) bv = 255; + + return 0xFF000000 | (rv << 16) | (gv << 8) | bv; +} + +static void control_panel_paint_main(Window *win) { + int offset_x = win->x + 8; + int offset_y = win->y + 30; + + // Draw wallpaper folder icon + // Folder icon + draw_rect(offset_x + 5, offset_y, 15, 6, COLOR_LTGRAY); + draw_rect(offset_x + 5, offset_y, 15, 1, COLOR_BLACK); + draw_rect(offset_x + 5, offset_y, 1, 6, COLOR_BLACK); + draw_rect(offset_x + 19, offset_y, 1, 6, COLOR_BLACK); + + draw_rect(offset_x + 5, offset_y + 6, 25, 15, COLOR_LTGRAY); + draw_rect(offset_x + 5, offset_y + 6, 25, 1, COLOR_BLACK); + draw_rect(offset_x + 5, offset_y + 6, 1, 15, COLOR_BLACK); + draw_rect(offset_x + 29, offset_y + 6, 1, 15, COLOR_BLACK); + draw_rect(offset_x + 5, offset_y + 20, 25, 1, COLOR_BLACK); + + // Label + draw_string(offset_x + 40, offset_y + 8, "Wallpaper", 0xFF000000); +} + +static void control_panel_paint_wallpaper(Window *win) { + int offset_x = win->x + 8; + int offset_y = win->y + 30; + + // Back button + draw_string(offset_x, offset_y, "< Back", 0xFF000080); + + draw_string(offset_x, offset_y + 25, "Presets:", 0xFF000000); + + // Color buttons + int button_y = offset_y + 45; + int button_x = offset_x; + + // Coffee button + draw_button(button_x, button_y, 60, 20, "Coffee", false); + draw_rect(button_x + 65, button_y + 5, 20, 10, COLOR_COFFEE); + + // Teal button + draw_button(button_x + 100, button_y, 60, 20, "Teal", false); + draw_rect(button_x + 165, button_y + 5, 20, 10, COLOR_TEAL); + + // Green button + draw_button(button_x + 200, button_y, 60, 20, "Green", false); + draw_rect(button_x + 265, button_y + 5, 20, 10, COLOR_GREEN); + + // Blue button + button_y += 30; + draw_button(button_x, button_y, 60, 20, "Blue", false); + draw_rect(button_x + 65, button_y + 5, 20, 10, COLOR_BLUE_BG); + + // Purple button + draw_button(button_x + 100, button_y, 60, 20, "Purple", false); + draw_rect(button_x + 165, button_y + 5, 20, 10, COLOR_PURPLE); + + // Custom color section + button_y += 40; + draw_string(offset_x, button_y, "Or something custom", 0xFF000000); + + button_y += 20; + + // R input box + draw_string(button_x, button_y, "R:", 0xFF000000); + draw_rect(button_x + 25, button_y, 50, 15, 0xFFFFFFFF); + draw_rect(button_x + 25, button_y, 50, 1, COLOR_BLACK); + draw_rect(button_x + 25, button_y, 1, 15, COLOR_BLACK); + draw_rect(button_x + 74, button_y, 1, 15, COLOR_BLACK); + draw_rect(button_x + 25, button_y + 14, 50, 1, COLOR_BLACK); + draw_string(button_x + 30, button_y + 3, rgb_r, (focused_field == 0) ? 0xFFFF0000 : COLOR_BLACK); + if (focused_field == 0) { + // Draw cursor + int cursor_x = button_x + 30 + input_cursor * 8; + draw_rect(cursor_x, button_y + 3, 1, 9, 0xFFFF0000); + } + + // G input box + draw_string(button_x + 90, button_y, "G:", 0xFF000000); + draw_rect(button_x + 115, button_y, 50, 15, 0xFFFFFFFF); + draw_rect(button_x + 115, button_y, 50, 1, COLOR_BLACK); + draw_rect(button_x + 115, button_y, 1, 15, COLOR_BLACK); + draw_rect(button_x + 164, button_y, 1, 15, COLOR_BLACK); + draw_rect(button_x + 115, button_y + 14, 50, 1, COLOR_BLACK); + draw_string(button_x + 120, button_y + 3, rgb_g, (focused_field == 1) ? 0xFF00AA00 : COLOR_BLACK); + if (focused_field == 1) { + // Draw cursor + int cursor_x = button_x + 120 + input_cursor * 8; + draw_rect(cursor_x, button_y + 3, 1, 9, 0xFF00AA00); + } + + // B input box + draw_string(button_x + 180, button_y, "B:", 0xFF000000); + draw_rect(button_x + 205, button_y, 50, 15, 0xFFFFFFFF); + draw_rect(button_x + 205, button_y, 50, 1, COLOR_BLACK); + draw_rect(button_x + 205, button_y, 1, 15, COLOR_BLACK); + draw_rect(button_x + 254, button_y, 1, 15, COLOR_BLACK); + draw_rect(button_x + 205, button_y + 14, 50, 1, COLOR_BLACK); + draw_string(button_x + 210, button_y + 3, rgb_b, (focused_field == 2) ? 0xFF0000FF : COLOR_BLACK); + if (focused_field == 2) { + // Draw cursor + int cursor_x = button_x + 210 + input_cursor * 8; + draw_rect(cursor_x, button_y + 3, 1, 9, 0xFF0000FF); + } + + // Apply button + draw_button(button_x, button_y + 25, 70, 20, "Apply", false); +} + +static void control_panel_paint(Window *win) { + if (current_view == VIEW_MAIN) { + control_panel_paint_main(win); + } else if (current_view == VIEW_WALLPAPER) { + control_panel_paint_wallpaper(win); + } +} + +static void control_panel_handle_click(Window *win, int x, int y) { + (void)win; // Unused parameter + + if (current_view == VIEW_MAIN) { + int offset_x = 8; + int offset_y = 30; + + // Check wallpaper folder click + if (x >= offset_x + 5 && x < offset_x + 35 && + y >= offset_y && y < offset_y + 25) { + current_view = VIEW_WALLPAPER; + } + } else if (current_view == VIEW_WALLPAPER) { + int offset_x = 8; + int offset_y = 30; + int button_y = offset_y + 45; + int button_x = offset_x; + + // Back button + if (x >= offset_x && x < offset_x + 40 && + y >= offset_y && y < offset_y + 15) { + current_view = VIEW_MAIN; + return; + } + + // Check Coffee button + if (x >= button_x && x < button_x + 60 && y >= button_y && y < button_y + 20) { + graphics_set_bg_color(COLOR_COFFEE); + return; + } + + // Check Teal button + if (x >= button_x + 100 && x < button_x + 160 && y >= button_y && y < button_y + 20) { + graphics_set_bg_color(COLOR_TEAL); + return; + } + + // Check Green button + if (x >= button_x + 200 && x < button_x + 260 && y >= button_y && y < button_y + 20) { + graphics_set_bg_color(COLOR_GREEN); + return; + } + + // Check Blue button + button_y += 30; + if (x >= button_x && x < button_x + 60 && y >= button_y && y < button_y + 20) { + graphics_set_bg_color(COLOR_BLUE_BG); + return; + } + + // Check Purple button + if (x >= button_x + 100 && x < button_x + 160 && y >= button_y && y < button_y + 20) { + graphics_set_bg_color(COLOR_PURPLE); + return; + } + + // Custom RGB section + button_y += 40; + button_y += 20; + + // Check R input box click + if (x >= button_x + 25 && x < button_x + 75 && y >= button_y && y < button_y + 15) { + if (focused_field != 0) { + rgb_r[0] = '\0'; // Clear when first focused + } + focused_field = 0; + input_cursor = 0; + return; + } + + // Check G input box click + if (x >= button_x + 115 && x < button_x + 165 && y >= button_y && y < button_y + 15) { + if (focused_field != 1) { + rgb_g[0] = '\0'; // Clear when first focused + } + focused_field = 1; + input_cursor = 0; + return; + } + + // Check B input box click + if (x >= button_x + 205 && x < button_x + 255 && y >= button_y && y < button_y + 15) { + if (focused_field != 2) { + rgb_b[0] = '\0'; // Clear when first focused + } + focused_field = 2; + input_cursor = 0; + return; + } + + // Check Apply button + if (x >= button_x && x < button_x + 70 && y >= button_y + 25 && y < button_y + 45) { + graphics_set_bg_color(parse_rgb_separate(rgb_r, rgb_g, rgb_b)); + return; + } + } +} + +static void control_panel_handle_key(Window *win, char c) { + (void)win; // Unused parameter + + if (current_view != VIEW_WALLPAPER) return; + if (focused_field < 0) return; // No field focused + + // Get the currently focused field buffer + char *focused_buffer = NULL; + int max_len = 3; // RGB values are 0-255, max 3 digits + + if (focused_field == 0) { + focused_buffer = rgb_r; + } else if (focused_field == 1) { + focused_buffer = rgb_g; + } else if (focused_field == 2) { + focused_buffer = rgb_b; + } else { + return; + } + + if (c == '\b') { // Backspace + if (input_cursor > 0) { + input_cursor--; + focused_buffer[input_cursor] = '\0'; + } + } else if (c >= '0' && c <= '9') { // Digits only + if (input_cursor < max_len) { + focused_buffer[input_cursor] = c; + input_cursor++; + focused_buffer[input_cursor] = '\0'; + } + } else if (c == '\t') { // Tab - switch to next field + focused_field = (focused_field + 1) % 3; + input_cursor = 0; + } +} + +void control_panel_init(void) { + win_control_panel.title = "Control Panel"; + win_control_panel.x = 200; + win_control_panel.y = 150; + win_control_panel.w = 350; + win_control_panel.h = 300; + win_control_panel.visible = false; + win_control_panel.focused = false; + win_control_panel.z_index = 0; + win_control_panel.paint = control_panel_paint; + win_control_panel.handle_key = control_panel_handle_key; + win_control_panel.handle_click = control_panel_handle_click; + win_control_panel.handle_right_click = NULL; + win_control_panel.buf_len = 0; + win_control_panel.cursor_pos = 0; +} + +void control_panel_reset(void) { + win_control_panel.focused = false; + current_view = VIEW_MAIN; + focused_field = -1; + input_cursor = 0; +} diff --git a/src/kernel/control_panel.h b/src/kernel/control_panel.h new file mode 100644 index 0000000..471b7e9 --- /dev/null +++ b/src/kernel/control_panel.h @@ -0,0 +1,11 @@ +#ifndef CONTROL_PANEL_H +#define CONTROL_PANEL_H + +#include "wm.h" + +extern Window win_control_panel; + +void control_panel_init(void); +void control_panel_reset(void); + +#endif diff --git a/src/kernel/editor.c b/src/kernel/editor.c new file mode 100644 index 0000000..66b05be --- /dev/null +++ b/src/kernel/editor.c @@ -0,0 +1,445 @@ +#include "editor.h" +#include "graphics.h" +#include "fat32.h" +#include "wm.h" +#include +#include + +// === Text Editor State === +Window win_editor; + +#define EDITOR_MAX_LINES 128 +#define EDITOR_MAX_LINE_LEN 256 +#define EDITOR_LINE_HEIGHT 16 +#define EDITOR_CHAR_WIDTH 8 + +typedef struct { + char content[EDITOR_MAX_LINE_LEN]; + int length; +} EditorLine; + +static EditorLine lines[EDITOR_MAX_LINES]; +static int line_count = 1; +static int cursor_line = 0; +static int cursor_col = 0; +static int scroll_top = 0; +static char open_filename[256] = ""; +static bool file_modified = false; + +// === Helper Functions === + +static size_t editor_strlen(const char *str) { + size_t len = 0; + while (str[len]) len++; + return len; +} + +static void editor_strcpy(char *dest, const char *src) { + while (*src) *dest++ = *src++; + *dest = 0; +} + +static int editor_strcmp(const char *s1, const char *s2) { + while (*s1 && (*s1 == *s2)) { + s1++; + s2++; + } + return *(const unsigned char*)s1 - *(const unsigned char*)s2; +} + +// === Editor Logic === + +// Forward declaration +static void editor_ensure_cursor_visible(void); + +static void editor_clear_all(void) { + for (int i = 0; i < EDITOR_MAX_LINES; i++) { + lines[i].content[0] = 0; + lines[i].length = 0; + } + line_count = 1; + cursor_line = 0; + cursor_col = 0; + scroll_top = 0; + open_filename[0] = 0; + file_modified = false; +} + +void editor_open_file(const char *filename) { + editor_clear_all(); + editor_strcpy(open_filename, filename); + + FAT32_FileHandle *fh = fat32_open(filename, "r"); + if (!fh) { + // New file + file_modified = false; + return; + } + + // Read file content + char buffer[16384]; + int bytes_read = fat32_read(fh, buffer, sizeof(buffer)); + fat32_close(fh); + + if (bytes_read <= 0) { + file_modified = false; + return; + } + + // Parse into lines + int line = 0; + int col = 0; + + for (int i = 0; i < bytes_read && line < EDITOR_MAX_LINES; i++) { + char ch = buffer[i]; + + if (ch == '\n') { + lines[line].content[col] = 0; + lines[line].length = col; + line++; + col = 0; + } else if (ch != '\r') { + if (col < EDITOR_MAX_LINE_LEN - 1) { + lines[line].content[col] = ch; + col++; + } + } + } + + if (col > 0) { + lines[line].content[col] = 0; + lines[line].length = col; + line++; + } + + line_count = (line > 0) ? line : 1; + file_modified = false; +} + +static void editor_save_file(void) { + if (!open_filename[0]) { + // No filename set + return; + } + + FAT32_FileHandle *fh = fat32_open(open_filename, "w"); + if (!fh) { + return; + } + + // Write lines + for (int i = 0; i < line_count; i++) { + fat32_write(fh, lines[i].content, lines[i].length); + fat32_write(fh, "\n", 1); + } + + fat32_close(fh); + file_modified = false; +} + +// Insert character at cursor position +static void editor_insert_char(char ch) { + if (cursor_line >= EDITOR_MAX_LINES) return; + + EditorLine *line = &lines[cursor_line]; + + if (ch == '\n') { + // Split line - shift all lines below down first + if (line_count >= EDITOR_MAX_LINES) return; + + // Shift all lines from cursor_line+1 onwards down by one position + for (int j = line_count; j > cursor_line; j--) { + lines[j] = lines[j - 1]; + } + line_count++; + + // Clear the new line completely (zero entire buffer) + for (int k = 0; k < EDITOR_MAX_LINE_LEN; k++) { + lines[cursor_line + 1].content[k] = 0; + } + lines[cursor_line + 1].length = 0; + + // Now split the current line at cursor position + int current_len = lines[cursor_line].length; + int new_len = current_len - cursor_col; + + // Copy the second part to the new line + for (int i = 0; i < new_len; i++) { + lines[cursor_line + 1].content[i] = lines[cursor_line].content[cursor_col + i]; + } + lines[cursor_line + 1].content[new_len] = 0; + lines[cursor_line + 1].length = new_len; + + // Truncate current line + lines[cursor_line].content[cursor_col] = 0; + lines[cursor_line].length = cursor_col; + + cursor_line++; + cursor_col = 0; + } else if (ch == '\b') { + // Backspace + if (cursor_col > 0) { + for (int i = cursor_col - 1; i < line->length; i++) { + line->content[i] = line->content[i + 1]; + } + line->length--; + cursor_col--; + } else if (cursor_line > 0) { + // Merge with previous line + EditorLine *prev = &lines[cursor_line - 1]; + int merge_point = prev->length; + + for (int i = 0; i < line->length; i++) { + if (merge_point + i < EDITOR_MAX_LINE_LEN - 1) { + prev->content[merge_point + i] = line->content[i]; + } + } + prev->content[merge_point + line->length] = 0; + prev->length = merge_point + line->length; + + // Shift lines up + for (int i = cursor_line; i < line_count - 1; i++) { + lines[i] = lines[i + 1]; + } + lines[line_count - 1].length = 0; + lines[line_count - 1].content[0] = 0; + + cursor_line--; + cursor_col = merge_point; + line_count--; + } + } else if (ch >= 32 && ch <= 126) { + // Regular character + if (cursor_col < EDITOR_MAX_LINE_LEN - 1) { + // Shift characters right + for (int i = line->length; i > cursor_col; i--) { + line->content[i] = line->content[i - 1]; + } + line->content[cursor_col] = ch; + line->length++; + cursor_col++; + } + } + + file_modified = true; + editor_ensure_cursor_visible(); +} + +// Ensure cursor is visible by adjusting scroll position +static void editor_ensure_cursor_visible(void) { + int visible_lines = 22; // Allow ~24 lines to use available window space + + // Scroll up if cursor is above visible area + if (cursor_line < scroll_top) { + scroll_top = cursor_line; + } + + // Scroll down if cursor is below visible area + if (cursor_line >= scroll_top + visible_lines) { + scroll_top = cursor_line - visible_lines + 1; + } +} + +// === Paint Function === + +static void editor_paint(Window *win) { + int offset_x = win->x + 4; + int offset_y = win->y + 24; + int content_width = win->w - 8; + int content_height = win->h - 28; + + // Draw filename and save button area at top of content + draw_rect(offset_x, offset_y, content_width, 25, COLOR_GRAY); + draw_string(offset_x + 10, offset_y + 5, "File: ", COLOR_BLACK); + draw_string(offset_x + 55, offset_y + 5, open_filename, COLOR_BLACK); + + // Draw save button + draw_button(offset_x + content_width - 80, offset_y + 3, 70, 20, "Save", false); + + // Draw modification indicator + if (file_modified) { + draw_string(offset_x + content_width - 200, offset_y + 5, "[Modified]", COLOR_RED); + } + + // Fill editor background + draw_rect(offset_x, offset_y + 30, content_width, content_height - 55, COLOR_WHITE); + + // Draw line numbers and content + int visible_lines = (content_height - 55) / EDITOR_LINE_HEIGHT; + int max_line = scroll_top + visible_lines; + if (max_line > line_count) max_line = line_count; + + for (int i = scroll_top; i < max_line; i++) { + int display_y = offset_y + 35 + (i - scroll_top) * EDITOR_LINE_HEIGHT; + + // Draw line number + char line_num_str[16]; + int temp = i + 1; + int str_len = 0; + if (temp == 0) { + line_num_str[0] = '0'; + str_len = 1; + } else { + while (temp > 0) { + line_num_str[str_len++] = (temp % 10) + '0'; + temp /= 10; + } + // Reverse + for (int j = 0; j < str_len / 2; j++) { + char t = line_num_str[j]; + line_num_str[j] = line_num_str[str_len - 1 - j]; + line_num_str[str_len - 1 - j] = t; + } + } + line_num_str[str_len] = 0; + draw_string(offset_x + 4, display_y, line_num_str, COLOR_DKGRAY); + + // Draw line content + draw_string(offset_x + 40, display_y, lines[i].content, COLOR_BLACK); + + // Draw cursor if on this line + if (i == cursor_line) { + int cursor_x = offset_x + 40 + (cursor_col * EDITOR_CHAR_WIDTH); + draw_rect(cursor_x, display_y, 2, 10, COLOR_BLACK); + } + } + + // Draw status bar at bottom + draw_rect(offset_x, offset_y + content_height - 20, content_width, 20, COLOR_GRAY); + draw_string(offset_x + 10, offset_y + content_height - 15, "Line: ", COLOR_WHITE); + + char line_str[32]; + int temp = cursor_line + 1; + int idx = 0; + while (temp > 0) { + line_str[idx++] = (temp % 10) + '0'; + temp /= 10; + } + for (int j = 0; j < idx / 2; j++) { + char t = line_str[j]; + line_str[j] = line_str[idx - 1 - j]; + line_str[idx - 1 - j] = t; + } + line_str[idx] = 0; + + draw_string(offset_x + 60, offset_y + content_height - 15, line_str, COLOR_WHITE); + draw_string(offset_x + 100, offset_y + content_height - 15, " Col: ", COLOR_WHITE); + + char col_str[32]; + temp = cursor_col + 1; + idx = 0; + while (temp > 0) { + col_str[idx++] = (temp % 10) + '0'; + temp /= 10; + } + for (int j = 0; j < idx / 2; j++) { + char t = col_str[j]; + col_str[j] = col_str[idx - 1 - j]; + col_str[idx - 1 - j] = t; + } + col_str[idx] = 0; + + draw_string(offset_x + 170, offset_y + content_height - 15, col_str, COLOR_WHITE); +} + +// === Key Handler === + +static void editor_handle_key(Window *win, char c) { + if (c == 'q' || c == 'Q') { + if (file_modified) { + } + win->visible = false; + return; + } + + // Arrow keys - UP + if (c == 17) { + if (cursor_line > 0) { + cursor_line--; + if (cursor_col > (int)lines[cursor_line].length) { + cursor_col = lines[cursor_line].length; + } + if (cursor_line < scroll_top) { + scroll_top = cursor_line; + } + } + return; + } + + // Arrow keys - DOWN + if (c == 18) { + if (cursor_line < line_count - 1) { + cursor_line++; + if (cursor_col > (int)lines[cursor_line].length) { + cursor_col = lines[cursor_line].length; + } + int visible_lines = 20; + if (cursor_line >= scroll_top + visible_lines) { + scroll_top = cursor_line - visible_lines + 1; + } + } + return; + } + + // Arrow keys - LEFT + if (c == 19) { + if (cursor_col > 0) { + cursor_col--; + } else if (cursor_line > 0) { + cursor_line--; + cursor_col = lines[cursor_line].length; + } + return; + } + + // Arrow keys - RIGHT + if (c == 20) { + if (cursor_col < (int)lines[cursor_line].length) { + cursor_col++; + } else if (cursor_line < line_count - 1) { + cursor_line++; + cursor_col = 0; + } + return; + } + + // Regular character input + editor_insert_char(c); +} + +// === Click Handler === + +static void editor_handle_click(Window *win, int x, int y) { + // x and y are relative to window origin + int content_width = win->w - 8; + + // Check save button - position is at (4 + content_width - 80, 24 + 3) = (4 + w - 8 - 80, 27) + // Button dimensions: 70 wide, 20 tall + int button_x = 4 + content_width - 80; + int button_y = 24 + 3; + + if (x >= button_x && x < button_x + 70 && + y >= button_y && y < button_y + 20) { + editor_save_file(); + return; + } +} + +// === Initialization === + +void editor_init(void) { + win_editor.title = "Text Editor"; + win_editor.x = 100; + win_editor.y = 150; + win_editor.w = 700; + win_editor.h = 450; + win_editor.visible = false; + win_editor.focused = false; + win_editor.z_index = 0; + win_editor.paint = editor_paint; + win_editor.handle_key = editor_handle_key; + win_editor.handle_click = editor_handle_click; + win_editor.handle_right_click = NULL; + + editor_clear_all(); +} diff --git a/src/kernel/editor.h b/src/kernel/editor.h new file mode 100644 index 0000000..2e4653b --- /dev/null +++ b/src/kernel/editor.h @@ -0,0 +1,11 @@ +#ifndef EDITOR_H +#define EDITOR_H + +#include "wm.h" + +extern Window win_editor; + +void editor_init(void); +void editor_open_file(const char *filename); + +#endif diff --git a/src/kernel/explorer.c b/src/kernel/explorer.c new file mode 100644 index 0000000..bcbf272 --- /dev/null +++ b/src/kernel/explorer.c @@ -0,0 +1,696 @@ +#include "explorer.h" +#include "graphics.h" +#include "fat32.h" +#include "wm.h" +#include "editor.h" +#include +#include + +// === File Explorer State === +Window win_explorer; + +#define EXPLORER_MAX_FILES 64 +#define EXPLORER_ITEM_HEIGHT 80 +#define EXPLORER_ITEM_WIDTH 120 +#define EXPLORER_COLS 3 +#define EXPLORER_ROWS 4 +#define EXPLORER_PADDING 15 + +// Dialog states +#define DIALOG_NONE 0 +#define DIALOG_CREATE_FILE 1 +#define DIALOG_CREATE_FOLDER 2 +#define DIALOG_DELETE_CONFIRM 3 +#define DIALOG_INPUT_MAX 256 + +typedef struct { + char name[256]; + bool is_directory; + uint32_t size; +} ExplorerItem; + +typedef enum { + ACTION_NONE, + ACTION_CREATE_FILE, + ACTION_CREATE_FOLDER, + ACTION_DELETE_FILE, + ACTION_DELETE_FOLDER +} ContextMenuAction; + +static ExplorerItem items[EXPLORER_MAX_FILES]; +static int item_count = 0; +static int selected_item = -1; +static char current_path[256] = "/"; +static int last_clicked_item = -1; +static uint32_t last_click_time = 0; + +// Dialog state +static int dialog_state = DIALOG_NONE; +static char dialog_input[DIALOG_INPUT_MAX] = ""; +static int dialog_input_cursor = 0; +static char dialog_target_path[256] = ""; // For delete confirmations +static bool dialog_target_is_dir = false; // For delete confirmations + +// Dropdown menu state +static bool dropdown_menu_visible = false; +static int dropdown_menu_item_height = 25; +#define DROPDOWN_MENU_WIDTH 120 +#define DROPDOWN_MENU_ITEMS 3 + +// === Helper Functions === + +static size_t explorer_strlen(const char *str); +static void explorer_strcpy(char *dest, const char *src); +static int explorer_strcmp(const char *s1, const char *s2); +static void explorer_strcat(char *dest, const char *src); +static void explorer_load_directory(const char *path); + +static size_t explorer_strlen(const char *str) { + size_t len = 0; + while (str[len]) len++; + return len; +} + +static void explorer_strcpy(char *dest, const char *src) { + while (*src) *dest++ = *src++; + *dest = 0; +} + +static int explorer_strcmp(const char *s1, const char *s2) { + while (*s1 && (*s1 == *s2)) { + s1++; + s2++; + } + return *(const unsigned char*)s1 - *(const unsigned char*)s2; +} + +static void explorer_strcat(char *dest, const char *src) { + while (*dest) dest++; + explorer_strcpy(dest, src); +} + +// === Dialog and File Operations === + +static void dialog_open_create_file(void) { + dialog_state = DIALOG_CREATE_FILE; + dialog_input[0] = 0; + dialog_input_cursor = 0; +} + +static void dialog_open_create_folder(void) { + dialog_state = DIALOG_CREATE_FOLDER; + dialog_input[0] = 0; + dialog_input_cursor = 0; +} + +static void dialog_open_delete_confirm(int item_idx) { + if (item_idx < 0 || item_idx >= item_count) return; + + dialog_state = DIALOG_DELETE_CONFIRM; + dialog_target_is_dir = items[item_idx].is_directory; + + // Build full path to target + explorer_strcpy(dialog_target_path, current_path); + if (dialog_target_path[explorer_strlen(dialog_target_path) - 1] != '/') { + explorer_strcat(dialog_target_path, "/"); + } + explorer_strcat(dialog_target_path, items[item_idx].name); +} + +static void dialog_close(void) { + dialog_state = DIALOG_NONE; + dialog_input[0] = 0; + dialog_input_cursor = 0; + dialog_target_path[0] = 0; +} + +static void dialog_confirm_create_file(void) { + if (dialog_input[0] == 0) return; + + char full_path[256]; + explorer_strcpy(full_path, current_path); + if (full_path[explorer_strlen(full_path) - 1] != '/') { + explorer_strcat(full_path, "/"); + } + explorer_strcat(full_path, dialog_input); + + // Create empty file + FAT32_FileHandle *file = fat32_open(full_path, "w"); + if (file) { + fat32_close(file); + explorer_load_directory(current_path); + } + + dialog_close(); +} + +static void dialog_confirm_create_folder(void) { + if (dialog_input[0] == 0) return; + + char full_path[256]; + explorer_strcpy(full_path, current_path); + if (full_path[explorer_strlen(full_path) - 1] != '/') { + explorer_strcat(full_path, "/"); + } + explorer_strcat(full_path, dialog_input); + + // Create directory + if (fat32_mkdir(full_path)) { + explorer_load_directory(current_path); + } + + dialog_close(); +} + +// Recursive delete for directories +static bool explorer_delete_recursive(const char *path) { + if (fat32_is_directory(path)) { + // List contents and delete recursively + FAT32_FileInfo entries[64]; + int count = fat32_list_directory(path, entries, 64); + + for (int i = 0; i < count; i++) { + char child_path[256]; + explorer_strcpy(child_path, path); + if (child_path[explorer_strlen(child_path) - 1] != '/') { + explorer_strcat(child_path, "/"); + } + explorer_strcat(child_path, entries[i].name); + + if (entries[i].is_directory) { + explorer_delete_recursive(child_path); + } else { + fat32_delete(child_path); + } + } + // Delete the directory itself + return fat32_rmdir(path); + } else { + // Regular file + return fat32_delete(path); + } +} + +static void dialog_confirm_delete(void) { + explorer_delete_recursive(dialog_target_path); + explorer_load_directory(current_path); + dialog_close(); +} + +static void dropdown_menu_toggle(void) { + dropdown_menu_visible = !dropdown_menu_visible; +} + +// === Helper Functions (continued) + +// === Explorer Logic === + +static void explorer_load_directory(const char *path) { + explorer_strcpy(current_path, path); + + FAT32_FileInfo entries[EXPLORER_MAX_FILES]; + int count = fat32_list_directory(path, entries, EXPLORER_MAX_FILES); + + item_count = 0; + for (int i = 0; i < count && i < EXPLORER_MAX_FILES; i++) { + explorer_strcpy(items[i].name, entries[i].name); + items[i].is_directory = entries[i].is_directory; + items[i].size = entries[i].size; + item_count++; + } + + selected_item = -1; +} + +static void explorer_navigate_to(const char *dirname) { + char new_path[256]; + + if (explorer_strcmp(dirname, "..") == 0) { + // Go to parent directory + int len = explorer_strlen(current_path); + int i = len - 1; + + // Skip trailing slashes + while (i > 0 && current_path[i] == '/') i--; + + // Find last slash + while (i > 0 && current_path[i] != '/') i--; + + if (i == 0) { + explorer_strcpy(new_path, "/"); + } else { + for (int j = 0; j < i; j++) { + new_path[j] = current_path[j]; + } + new_path[i] = 0; + } + } else { + // Go to subdirectory + explorer_strcpy(new_path, current_path); + if (new_path[explorer_strlen(new_path) - 1] != '/') { + explorer_strcat(new_path, "/"); + } + explorer_strcat(new_path, dirname); + } + + explorer_load_directory(new_path); +} + +// Draw a simple file icon +static void explorer_draw_file_icon(int x, int y, bool is_dir) { + if (is_dir) { + // Folder icon - larger + draw_rect(x + 10, y + 10, 30, 5, COLOR_BLUE); // Tab + draw_rect(x + 10, y + 15, 30, 25, COLOR_WHITE); // Main folder + draw_rect(x + 10, y + 15, 2, 25, COLOR_BLACK); + draw_rect(x + 10, y + 15, 30, 2, COLOR_BLACK); + draw_rect(x + 38, y + 15, 2, 25, COLOR_BLACK); + draw_rect(x + 10, y + 38, 30, 2, COLOR_BLACK); + } else { + // Document icon - larger + draw_rect(x + 12, y + 10, 20, 25, COLOR_WHITE); + draw_rect(x + 12, y + 10, 20, 2, COLOR_BLACK); + draw_rect(x + 12, y + 10, 2, 25, COLOR_BLACK); + draw_rect(x + 30, y + 10, 2, 25, COLOR_BLACK); + draw_rect(x + 12, y + 33, 20, 2, COLOR_BLACK); + // Lines on document + draw_rect(x + 15, y + 18, 14, 1, COLOR_DKGRAY); + draw_rect(x + 15, y + 23, 14, 1, COLOR_DKGRAY); + draw_rect(x + 15, y + 28, 14, 1, COLOR_DKGRAY); + } +} + +// === Paint Function === + +static void explorer_paint(Window *win) { + int offset_x = win->x + 4; + int offset_y = win->y + 24; + + // Fill background + draw_rect(offset_x, offset_y, win->w - 8, win->h - 28, COLOR_LTGRAY); + + // Draw path bar + int path_height = 30; + draw_bevel_rect(offset_x + 4, offset_y + 4, win->w - 16, path_height, true); + draw_string(offset_x + 10, offset_y + 10, "Path: ", COLOR_BLACK); + draw_string(offset_x + 50, offset_y + 10, current_path, COLOR_BLACK); + + // Draw dropdown menu button (right-aligned, before back button) + int dropdown_btn_x = win->x + win->w - 90; + draw_button(dropdown_btn_x, offset_y + 4, 35, 30, "...", false); + + // Draw back button (right-aligned) + draw_button(win->x + win->w - 40, offset_y + 4, 30, 30, "<", false); + + // Draw dropdown menu if visible + if (dropdown_menu_visible) { + int menu_x = dropdown_btn_x; + int menu_y = offset_y + 34; + + // Draw menu background + draw_rect(menu_x, menu_y, DROPDOWN_MENU_WIDTH, dropdown_menu_item_height * DROPDOWN_MENU_ITEMS, COLOR_LTGRAY); + draw_bevel_rect(menu_x, menu_y, DROPDOWN_MENU_WIDTH, dropdown_menu_item_height * DROPDOWN_MENU_ITEMS, true); + + // Draw menu items + draw_string(menu_x + 8, menu_y + 5, "New File", COLOR_BLACK); + draw_string(menu_x + 8, menu_y + dropdown_menu_item_height + 5, "New Folder", COLOR_BLACK); + draw_string(menu_x + 8, menu_y + dropdown_menu_item_height * 2 + 5, "Delete", COLOR_RED); + } + + // Draw file list + int content_start_y = offset_y + 40; + + for (int i = 0; i < item_count; i++) { + int row = i / EXPLORER_COLS; + int col = i % EXPLORER_COLS; + + int item_x = offset_x + 10 + (col * (EXPLORER_ITEM_WIDTH + EXPLORER_PADDING)); + int item_y = content_start_y + (row * (EXPLORER_ITEM_HEIGHT + EXPLORER_PADDING)); + + // Draw item background + uint32_t bg_color = (i == selected_item) ? COLOR_BLUE : COLOR_WHITE; + draw_bevel_rect(item_x, item_y, EXPLORER_ITEM_WIDTH, EXPLORER_ITEM_HEIGHT, false); + draw_rect(item_x + 2, item_y + 2, EXPLORER_ITEM_WIDTH - 4, EXPLORER_ITEM_HEIGHT - 4, bg_color); + + // Draw icon (larger area) + explorer_draw_file_icon(item_x + 5, item_y + 5, items[i].is_directory); + + // Draw name below icon + uint32_t text_color = (i == selected_item) ? COLOR_WHITE : COLOR_BLACK; + int name_len = explorer_strlen(items[i].name); + int text_x = item_x + 5; + int text_y = item_y + 50; + + // Truncate name if too long + char display_name[24]; + int copy_len = name_len > 18 ? 18 : name_len; + for (int j = 0; j < copy_len; j++) { + display_name[j] = items[i].name[j]; + } + display_name[copy_len] = 0; + + draw_string(text_x, text_y, display_name, text_color); + } + + // Draw dialogs + if (dialog_state == DIALOG_CREATE_FILE) { + int dlg_x = win->x + win->w / 2 - 150; + int dlg_y = win->y + win->h / 2 - 60; + + // Dialog background + draw_rect(dlg_x - 5, dlg_y - 5, 310, 120, COLOR_LTGRAY); + draw_bevel_rect(dlg_x, dlg_y, 300, 110, true); + + // Title + draw_string(dlg_x + 10, dlg_y + 10, "Create New File", COLOR_BLACK); + + // Input field + draw_bevel_rect(dlg_x + 10, dlg_y + 35, 280, 20, false); + draw_string(dlg_x + 15, dlg_y + 40, dialog_input, COLOR_BLACK); + draw_string(dlg_x + 15 + dialog_input_cursor * 8, dlg_y + 40, "|", COLOR_BLACK); + + // Buttons + draw_button(dlg_x + 50, dlg_y + 65, 80, 25, "Create", false); + draw_button(dlg_x + 170, dlg_y + 65, 80, 25, "Cancel", false); + } else if (dialog_state == DIALOG_CREATE_FOLDER) { + int dlg_x = win->x + win->w / 2 - 150; + int dlg_y = win->y + win->h / 2 - 60; + + // Dialog background + draw_rect(dlg_x - 5, dlg_y - 5, 310, 120, COLOR_LTGRAY); + draw_bevel_rect(dlg_x, dlg_y, 300, 110, true); + + // Title + draw_string(dlg_x + 10, dlg_y + 10, "Create New Folder", COLOR_BLACK); + + // Input field + draw_bevel_rect(dlg_x + 10, dlg_y + 35, 280, 20, false); + draw_string(dlg_x + 15, dlg_y + 40, dialog_input, COLOR_BLACK); + draw_string(dlg_x + 15 + dialog_input_cursor * 8, dlg_y + 40, "|", COLOR_BLACK); + + // Buttons + draw_button(dlg_x + 50, dlg_y + 65, 80, 25, "Create", false); + draw_button(dlg_x + 170, dlg_y + 65, 80, 25, "Cancel", false); + } else if (dialog_state == DIALOG_DELETE_CONFIRM) { + int dlg_x = win->x + win->w / 2 - 150; + int dlg_y = win->y + win->h / 2 - 60; + + // Dialog background + draw_rect(dlg_x - 5, dlg_y - 5, 310, 120, COLOR_LTGRAY); + draw_bevel_rect(dlg_x, dlg_y, 300, 110, true); + + // Title + const char *title = dialog_target_is_dir ? "Delete Folder?" : "Delete File?"; + draw_string(dlg_x + 10, dlg_y + 10, title, COLOR_BLACK); + + // Message + draw_string(dlg_x + 10, dlg_y + 35, "This action cannot be undone.", COLOR_BLACK); + + // Buttons + draw_button(dlg_x + 50, dlg_y + 65, 80, 25, "Delete", false); + draw_button(dlg_x + 170, dlg_y + 65, 80, 25, "Cancel", false); + } +} + +// === Mouse Handler === + +static void explorer_handle_click(Window *win, int x, int y) { + // Handle dialog clicks first + if (dialog_state == DIALOG_CREATE_FILE || dialog_state == DIALOG_CREATE_FOLDER) { + int dlg_x = win->w / 2 - 150; + int dlg_y = win->h / 2 - 60; + + // Create button + if (x >= dlg_x + 50 && x < dlg_x + 130 && + y >= dlg_y + 65 && y < dlg_y + 90) { + if (dialog_state == DIALOG_CREATE_FILE) { + dialog_confirm_create_file(); + } else { + dialog_confirm_create_folder(); + } + return; + } + + // Cancel button + if (x >= dlg_x + 170 && x < dlg_x + 250 && + y >= dlg_y + 65 && y < dlg_y + 90) { + dialog_close(); + return; + } + + // Input field click + if (x >= dlg_x + 10 && x < dlg_x + 290 && + y >= dlg_y + 35 && y < dlg_y + 55) { + dialog_input_cursor = (x - dlg_x - 15) / 8; + if (dialog_input_cursor > (int)explorer_strlen(dialog_input)) { + dialog_input_cursor = explorer_strlen(dialog_input); + } + return; + } + } else if (dialog_state == DIALOG_DELETE_CONFIRM) { + int dlg_x = win->w / 2 - 150; + int dlg_y = win->h / 2 - 60; + + // Delete button + if (x >= dlg_x + 50 && x < dlg_x + 130 && + y >= dlg_y + 65 && y < dlg_y + 90) { + dialog_confirm_delete(); + return; + } + + // Cancel button + if (x >= dlg_x + 170 && x < dlg_x + 250 && + y >= dlg_y + 65 && y < dlg_y + 90) { + dialog_close(); + return; + } + } + + // Handle dropdown menu clicks + if (dropdown_menu_visible) { + int dropdown_btn_x = win->w - 90; // Window-relative + int menu_y = 58; // Window-relative (offset_y + 34, where offset_y = 24) + + // New File + if (x >= dropdown_btn_x && x < dropdown_btn_x + DROPDOWN_MENU_WIDTH && + y >= menu_y && y < menu_y + dropdown_menu_item_height) { + dropdown_menu_toggle(); + dialog_open_create_file(); + return; + } + + // New Folder + if (x >= dropdown_btn_x && x < dropdown_btn_x + DROPDOWN_MENU_WIDTH && + y >= menu_y + dropdown_menu_item_height && + y < menu_y + dropdown_menu_item_height * 2) { + dropdown_menu_toggle(); + dialog_open_create_folder(); + return; + } + + // Delete + if (x >= dropdown_btn_x && x < dropdown_btn_x + DROPDOWN_MENU_WIDTH && + y >= menu_y + dropdown_menu_item_height * 2 && + y < menu_y + dropdown_menu_item_height * 3) { + dropdown_menu_toggle(); + if (selected_item >= 0) { + dialog_open_delete_confirm(selected_item); + } + return; + } + + // Click outside menu closes it + dropdown_menu_toggle(); + return; + } + + // x, y are already relative to window (0,0 is top-left of window content area) + // Check dropdown menu button + int button_y = 28; // Position from top of window title bar + if (x >= win->w - 90 && x < win->w - 55 && + y >= button_y && y < button_y + 30) { + // Dropdown menu button clicked + dropdown_menu_toggle(); + return; + } + + // Check back button (right-aligned) + if (x >= win->w - 40 && x < win->w - 10 && + y >= button_y && y < button_y + 30) { + // Back button clicked + explorer_navigate_to(".."); + return; + } + + // File items start at y=64 relative to window + int content_start_y = 64; + int offset_x = 4; + + for (int i = 0; i < item_count; i++) { + int row = i / EXPLORER_COLS; + int col = i % EXPLORER_COLS; + + int item_x = offset_x + 10 + (col * (EXPLORER_ITEM_WIDTH + EXPLORER_PADDING)); + int item_y = content_start_y + (row * (EXPLORER_ITEM_HEIGHT + EXPLORER_PADDING)); + + if (x >= item_x && x < item_x + EXPLORER_ITEM_WIDTH && + y >= item_y && y < item_y + EXPLORER_ITEM_HEIGHT) { + + // Check for double-click + if (last_clicked_item == i) { + // Double-click detected + if (items[i].is_directory) { + explorer_navigate_to(items[i].name); + } else { + // Open file in editor + char full_path[256]; + explorer_strcpy(full_path, current_path); + if (full_path[explorer_strlen(full_path) - 1] != '/') { + explorer_strcat(full_path, "/"); + } + explorer_strcat(full_path, items[i].name); + + // Open in editor and bring to front + win_editor.visible = true; + win_editor.focused = true; + int max_z = 0; + for (int j = 0; j < 5; j++) { // window_count is 5 + // Need to find max z_index - check all windows + if (win_explorer.z_index > max_z) max_z = win_explorer.z_index; + if (win_cmd.z_index > max_z) max_z = win_cmd.z_index; + if (win_notepad.z_index > max_z) max_z = win_notepad.z_index; + if (win_calculator.z_index > max_z) max_z = win_calculator.z_index; + } + win_editor.z_index = max_z + 1; + editor_open_file(full_path); + } + last_clicked_item = -1; + } else { + // Single-click - select + selected_item = i; + last_clicked_item = i; + last_click_time = 0; // Reset for next click + } + return; + } + } +} + +// === Key Handler === + +static void explorer_handle_key(Window *win, char c) { + (void)win; + + // Handle dialog input + if (dialog_state == DIALOG_CREATE_FILE || dialog_state == DIALOG_CREATE_FOLDER) { + if (c == 27) { // ESC - close dialog + dialog_close(); + return; + } else if (c == '\n') { // ENTER - confirm + if (dialog_state == DIALOG_CREATE_FILE) { + dialog_confirm_create_file(); + } else { + dialog_confirm_create_folder(); + } + return; + } else if (c == 8 || c == 127) { // BACKSPACE + if (dialog_input_cursor > 0) { + dialog_input_cursor--; + // Shift characters + for (int i = dialog_input_cursor; i < (int)explorer_strlen(dialog_input); i++) { + dialog_input[i] = dialog_input[i + 1]; + } + } + return; + } else if (c >= 32 && c < 127) { // Printable character + int len = explorer_strlen(dialog_input); + if (len < DIALOG_INPUT_MAX - 1) { + // Shift characters to make room + for (int i = len; i >= dialog_input_cursor; i--) { + dialog_input[i + 1] = dialog_input[i]; + } + dialog_input[dialog_input_cursor] = c; + dialog_input_cursor++; + } + return; + } + return; + } + + if (dialog_state == DIALOG_DELETE_CONFIRM) { + if (c == 27) { // ESC + dialog_close(); + return; + } + return; + } + + if (c == 'q' || c == 'Q') { + win->visible = false; + return; + } + + // Close dropdown menu if open with ESC + if (dropdown_menu_visible && c == 27) { + dropdown_menu_toggle(); + return; + } + + if (c == 17) { // UP + if (selected_item > 0) { + selected_item -= EXPLORER_COLS; + if (selected_item < 0) selected_item = 0; + } + } else if (c == 18) { // DOWN + if (selected_item < item_count - 1) { + selected_item += EXPLORER_COLS; + if (selected_item >= item_count) selected_item = item_count - 1; + } + } else if (c == 19) { // LEFT + if (selected_item > 0) { + selected_item--; + } + } else if (c == 20) { // RIGHT + if (selected_item < item_count - 1) { + selected_item++; + } + } else if (c == '\n') { // ENTER + if (selected_item >= 0 && selected_item < item_count) { + if (items[selected_item].is_directory) { + explorer_navigate_to(items[selected_item].name); + } + } + } else if (c == 'd' || c == 'D') { // Delete key + if (selected_item >= 0) { + dialog_open_delete_confirm(selected_item); + } + } else if (c == 'n' || c == 'N') { // New file + dialog_open_create_file(); + } else if (c == 'f' || c == 'F') { // New folder + dialog_open_create_folder(); + } +} + +// === Initialization === + +void explorer_init(void) { + win_explorer.title = "File Explorer"; + win_explorer.x = 300; + win_explorer.y = 100; + win_explorer.w = 600; + win_explorer.h = 400; + win_explorer.visible = false; + win_explorer.focused = false; + win_explorer.z_index = 0; + win_explorer.paint = explorer_paint; + win_explorer.handle_key = explorer_handle_key; + win_explorer.handle_click = explorer_handle_click; + win_explorer.handle_right_click = NULL; + + explorer_load_directory("/"); +} +void explorer_reset(void) { + // Reset explorer to root directory on close/reopen + explorer_load_directory("/"); + win_explorer.focused = false; +} \ No newline at end of file diff --git a/src/kernel/explorer.h b/src/kernel/explorer.h new file mode 100644 index 0000000..fcbff52 --- /dev/null +++ b/src/kernel/explorer.h @@ -0,0 +1,15 @@ +#ifndef EXPLORER_H +#define EXPLORER_H + +#include "wm.h" + +extern Window win_explorer; +extern Window win_editor; +extern Window win_cmd; +extern Window win_notepad; +extern Window win_calculator; + +void explorer_init(void); +void explorer_reset(void); + +#endif diff --git a/src/kernel/fat32.c b/src/kernel/fat32.c new file mode 100644 index 0000000..f5b158a --- /dev/null +++ b/src/kernel/fat32.c @@ -0,0 +1,480 @@ +#include "fat32.h" +#include +#include + +// === Memory-based FAT32 Implementation === +// This is a simplified FAT32 for the OS kernel +// It allocates everything in RAM instead of using a real disk + +#define MAX_FILES 256 +#define MAX_CLUSTERS 1024 +#define MAX_OPEN_HANDLES 32 + +// In-memory FAT table +static uint32_t fat_table[MAX_CLUSTERS]; +static uint8_t cluster_data[MAX_CLUSTERS][FAT32_CLUSTER_SIZE]; + +// File/Directory tracking +typedef struct { + char full_path[FAT32_MAX_PATH]; + char filename[FAT32_MAX_FILENAME]; + uint32_t start_cluster; + uint32_t size; + uint32_t attributes; + bool used; + char parent_path[FAT32_MAX_PATH]; +} FileEntry; + +static FileEntry files[MAX_FILES]; +static uint32_t next_cluster = 3; // Start after reserved clusters 0, 1, 2 +static FAT32_FileHandle open_handles[MAX_OPEN_HANDLES]; +static char current_dir[FAT32_MAX_PATH] = "/"; + +// === Helper Functions === + +static size_t fs_strlen(const char *str) { + size_t len = 0; + while (str[len]) len++; + return len; +} + +static void fs_strcpy(char *dest, const char *src) { + while (*src) *dest++ = *src++; + *dest = 0; +} + +static int fs_strcmp(const char *s1, const char *s2) { + while (*s1 && (*s1 == *s2)) { + s1++; + s2++; + } + return *(const unsigned char*)s1 - *(const unsigned char*)s2; +} + +static void fs_strcat(char *dest, const char *src) { + while (*dest) dest++; + fs_strcpy(dest, src); +} + +static bool fs_ends_with(const char *str, const char *suffix) { + int str_len = fs_strlen(str); + int suffix_len = fs_strlen(suffix); + if (suffix_len > str_len) return false; + return fs_strcmp(str + str_len - suffix_len, suffix) == 0; +} + +// Extract filename from path +static void extract_filename(const char *path, char *filename) { + int len = fs_strlen(path); + int i = len - 1; + + // Skip trailing slashes + while (i > 0 && path[i] == '/') i--; + + // Find last slash + int start = i; + while (start >= 0 && path[start] != '/') start--; + start++; + + // Copy filename + int j = 0; + for (int k = start; k <= i; k++) { + filename[j++] = path[k]; + } + filename[j] = 0; +} + +// Extract parent path +static void extract_parent_path(const char *path, char *parent) { + int len = fs_strlen(path); + int i = len - 1; + + // Skip trailing slashes + while (i > 0 && path[i] == '/') i--; + + // Find last slash + while (i > 0 && path[i] != '/') i--; + + if (i == 0) { + parent[0] = '/'; + parent[1] = 0; + } else { + for (int j = 0; j < i; j++) { + parent[j] = path[j]; + } + parent[i] = 0; + } +} + +// Normalize path (remove .., ., etc) +void fat32_normalize_path(const char *path, char *normalized) { + if (path[0] == '/') { + // Absolute path + fs_strcpy(normalized, path); + } else { + // Relative path - prepend current directory + if (fs_strcmp(current_dir, "/") == 0) { + normalized[0] = '/'; + fs_strcpy(normalized + 1, path); + } else { + fs_strcpy(normalized, current_dir); + if (normalized[fs_strlen(normalized) - 1] != '/') { + fs_strcat(normalized, "/"); + } + fs_strcat(normalized, path); + } + } + + // Remove trailing slashes (except for root) + int len = fs_strlen(normalized); + while (len > 1 && normalized[len - 1] == '/') { + normalized[--len] = 0; + } +} + +// Find file entry by path +static FileEntry* find_file(const char *path) { + char normalized[FAT32_MAX_PATH]; + fat32_normalize_path(path, normalized); + + for (int i = 0; i < MAX_FILES; i++) { + if (files[i].used && fs_strcmp(files[i].full_path, normalized) == 0) { + return &files[i]; + } + } + return NULL; +} + +// Find first unused file entry +static FileEntry* find_free_entry(void) { + for (int i = 0; i < MAX_FILES; i++) { + if (!files[i].used) { + return &files[i]; + } + } + return NULL; +} + +// Find free handle +static FAT32_FileHandle* find_free_handle(void) { + for (int i = 0; i < MAX_OPEN_HANDLES; i++) { + if (!open_handles[i].valid) { + return &open_handles[i]; + } + } + return NULL; +} + +// Allocate cluster +static uint32_t allocate_cluster(void) { + if (next_cluster >= MAX_CLUSTERS) return 0; + uint32_t cluster = next_cluster++; + fat_table[cluster] = 0xFFFFFFFF; // End of chain + return cluster; +} + +// === Public API === + +void fat32_init(void) { + // Initialize FAT table + for (int i = 0; i < MAX_CLUSTERS; i++) { + fat_table[i] = 0; + } + fat_table[0] = 0xFFFFFFF8; // Media descriptor + fat_table[1] = 0xFFFFFFFF; // Bad sector marker + + // Create root directory entry + FileEntry *root = find_free_entry(); + if (root) { + root->used = true; + root->filename[0] = 0; + fs_strcpy(root->full_path, "/"); + root->start_cluster = 2; // Root cluster + root->size = 0; + root->attributes = ATTR_DIRECTORY; + fat_table[2] = 0xFFFFFFFF; // Root is EOF + } + + next_cluster = 3; + current_dir[0] = '/'; + current_dir[1] = 0; +} + +FAT32_FileHandle* fat32_open(const char *path, const char *mode) { + char normalized[FAT32_MAX_PATH]; + fat32_normalize_path(path, normalized); + + FileEntry *entry = find_file(normalized); + + if (mode[0] == 'r') { + // Read mode + if (!entry || (entry->attributes & ATTR_DIRECTORY)) { + return NULL; // File not found or is directory + } + } else if (mode[0] == 'w' || (mode[0] == 'a')) { + // Write/append mode - create if not exists + if (!entry) { + entry = find_free_entry(); + if (!entry) return NULL; + + entry->used = true; + fs_strcpy(entry->full_path, normalized); + extract_filename(normalized, entry->filename); + extract_parent_path(normalized, entry->parent_path); + entry->start_cluster = allocate_cluster(); + if (!entry->start_cluster) return NULL; + entry->size = 0; + entry->attributes = 0; // Regular file + } + + if (mode[0] == 'w') { + entry->size = 0; // Truncate + } + } + + // Find free handle + FAT32_FileHandle *handle = find_free_handle(); + if (!handle) return NULL; + + handle->valid = true; + handle->cluster = entry->start_cluster; + handle->position = 0; + handle->size = entry->size; + + if (mode[0] == 'r') { + handle->mode = 0; + } else if (mode[0] == 'w') { + handle->mode = 1; + } else { + handle->mode = 2; // append + handle->position = entry->size; + } + + return handle; +} + +void fat32_close(FAT32_FileHandle *handle) { + if (handle) { + handle->valid = false; + } +} + +int fat32_read(FAT32_FileHandle *handle, void *buffer, int size) { + if (!handle || !handle->valid || handle->mode != 0) { + return -1; + } + + int bytes_read = 0; + uint8_t *buf = (uint8_t *)buffer; + + while (bytes_read < size && handle->position < handle->size) { + uint32_t offset_in_cluster = handle->position % FAT32_CLUSTER_SIZE; + int to_read = size - bytes_read; + int available = handle->size - handle->position; + + if (to_read > available) { + to_read = available; + } + if (to_read > FAT32_CLUSTER_SIZE - offset_in_cluster) { + to_read = FAT32_CLUSTER_SIZE - offset_in_cluster; + } + + if (handle->cluster >= MAX_CLUSTERS) { + break; + } + + uint8_t *src = cluster_data[handle->cluster] + offset_in_cluster; + for (int i = 0; i < to_read; i++) { + buf[bytes_read + i] = src[i]; + } + + bytes_read += to_read; + handle->position += to_read; + + // Move to next cluster if needed + if (handle->position % FAT32_CLUSTER_SIZE == 0 && handle->position < handle->size) { + handle->cluster = fat_table[handle->cluster]; + } + } + + return bytes_read; +} + +int fat32_write(FAT32_FileHandle *handle, const void *buffer, int size) { + if (!handle || !handle->valid || (handle->mode != 1 && handle->mode != 2)) { + return -1; + } + + int bytes_written = 0; + const uint8_t *buf = (const uint8_t *)buffer; + uint32_t initial_cluster = handle->cluster; + + while (bytes_written < size) { + uint32_t offset_in_cluster = handle->position % FAT32_CLUSTER_SIZE; + int to_write = size - bytes_written; + + if (to_write > FAT32_CLUSTER_SIZE - offset_in_cluster) { + to_write = FAT32_CLUSTER_SIZE - offset_in_cluster; + } + + if (handle->cluster >= MAX_CLUSTERS) { + break; + } + + uint8_t *dest = cluster_data[handle->cluster] + offset_in_cluster; + for (int i = 0; i < to_write; i++) { + dest[i] = buf[bytes_written + i]; + } + + bytes_written += to_write; + handle->position += to_write; + + if (handle->position > handle->size) { + handle->size = handle->position; + } + + // Move to next cluster if needed + if (offset_in_cluster + to_write >= FAT32_CLUSTER_SIZE && bytes_written < size) { + uint32_t next = allocate_cluster(); + if (!next) break; + fat_table[handle->cluster] = next; + handle->cluster = next; + } + } + + // Update file entry + for (int i = 0; i < MAX_FILES; i++) { + if (files[i].used && files[i].start_cluster == initial_cluster) { + files[i].size = handle->size; + break; + } + } + + return bytes_written; +} + +int fat32_seek(FAT32_FileHandle *handle, int offset, int whence) { + if (!handle || !handle->valid) { + return -1; + } + + uint32_t new_position = handle->position; + + if (whence == 0) { // SEEK_SET + new_position = offset; + } else if (whence == 1) { // SEEK_CUR + new_position += offset; + } else if (whence == 2) { // SEEK_END + new_position = handle->size + offset; + } + + if (new_position > handle->size) { + new_position = handle->size; + } + + handle->position = new_position; + return new_position; +} + +bool fat32_mkdir(const char *path) { + char normalized[FAT32_MAX_PATH]; + fat32_normalize_path(path, normalized); + + if (find_file(normalized)) { + return false; // Already exists + } + + FileEntry *entry = find_free_entry(); + if (!entry) return false; + + entry->used = true; + fs_strcpy(entry->full_path, normalized); + extract_filename(normalized, entry->filename); + extract_parent_path(normalized, entry->parent_path); + entry->start_cluster = allocate_cluster(); + entry->size = 0; + entry->attributes = ATTR_DIRECTORY; + + return true; +} + +bool fat32_rmdir(const char *path) { + char normalized[FAT32_MAX_PATH]; + fat32_normalize_path(path, normalized); + + FileEntry *entry = find_file(normalized); + if (!entry || !(entry->attributes & ATTR_DIRECTORY)) { + return false; + } + + entry->used = false; + return true; +} + +bool fat32_delete(const char *path) { + char normalized[FAT32_MAX_PATH]; + fat32_normalize_path(path, normalized); + + FileEntry *entry = find_file(normalized); + if (!entry || (entry->attributes & ATTR_DIRECTORY)) { + return false; + } + + entry->used = false; + return true; +} + +bool fat32_exists(const char *path) { + return find_file(path) != NULL; +} + +bool fat32_is_directory(const char *path) { + FileEntry *entry = find_file(path); + return entry && (entry->attributes & ATTR_DIRECTORY); +} + +int fat32_list_directory(const char *path, FAT32_FileInfo *entries, int max_entries) { + char normalized[FAT32_MAX_PATH]; + fat32_normalize_path(path, normalized); + + FileEntry *dir = find_file(normalized); + if (!dir || !(dir->attributes & ATTR_DIRECTORY)) { + return 0; // Not a directory + } + + int count = 0; + for (int i = 0; i < MAX_FILES && count < max_entries; i++) { + if (files[i].used && fs_strcmp(files[i].parent_path, normalized) == 0) { + fs_strcpy(entries[count].name, files[i].filename); + entries[count].size = files[i].size; + entries[count].is_directory = (files[i].attributes & ATTR_DIRECTORY) != 0; + entries[count].start_cluster = files[i].start_cluster; + count++; + } + } + + return count; +} + +bool fat32_chdir(const char *path) { + char normalized[FAT32_MAX_PATH]; + fat32_normalize_path(path, normalized); + + FileEntry *entry = find_file(normalized); + if (!entry || !(entry->attributes & ATTR_DIRECTORY)) { + return false; + } + + fs_strcpy(current_dir, normalized); + return true; +} + +void fat32_get_current_dir(char *buffer, int size) { + int len = fs_strlen(current_dir); + if (len >= size) len = size - 1; + + for (int i = 0; i < len; i++) { + buffer[i] = current_dir[i]; + } + buffer[len] = 0; +} diff --git a/src/kernel/fat32.h b/src/kernel/fat32.h new file mode 100644 index 0000000..c3124b1 --- /dev/null +++ b/src/kernel/fat32.h @@ -0,0 +1,126 @@ +#ifndef FAT32_H +#define FAT32_H + +#include +#include + +// === FAT32 Structures === + +// Boot Sector (512 bytes) +typedef struct { + uint8_t jmp[3]; // Jump instruction + uint8_t oem[8]; // OEM identifier + uint16_t bytes_per_sector; // Bytes per sector (usually 512) + uint8_t sectors_per_cluster; // Sectors per cluster + uint16_t reserved_sectors; // Reserved sectors (usually 1) + uint8_t num_fats; // Number of FATs (usually 2) + uint16_t root_entries; // Root directory entries (0 for FAT32) + uint16_t total_sectors_16; // Total sectors 16-bit (0 for FAT32) + uint8_t media_descriptor; // Media descriptor + uint16_t sectors_per_fat_16; // Sectors per FAT 16-bit (0 for FAT32) + uint16_t sectors_per_track; // Sectors per track + uint16_t num_heads; // Number of heads + uint32_t hidden_sectors; // Hidden sectors + uint32_t total_sectors_32; // Total sectors 32-bit + + // FAT32 Specific + uint32_t sectors_per_fat_32; // Sectors per FAT 32-bit + uint16_t flags; // Flags + uint16_t version; // Version + uint32_t root_cluster; // Root directory cluster + uint16_t fsinfo_sector; // FSInfo sector number + uint16_t backup_boot_sector; // Backup boot sector number + uint8_t reserved[12]; // Reserved + uint8_t drive_number; // Drive number + uint8_t reserved2; // Reserved + uint8_t boot_signature; // Boot signature + uint32_t serial_number; // Volume serial number + uint8_t volume_label[11]; // Volume label + uint8_t fs_type[8]; // Filesystem type ("FAT32 ") + uint8_t boot_code[420]; // Boot code + uint16_t boot_signature_value; // Boot signature value (0xAA55) +} __attribute__((packed)) FAT32_BootSector; + +// Directory Entry (32 bytes) +typedef struct { + uint8_t filename[8]; // Filename (8 bytes) + uint8_t extension[3]; // Extension (3 bytes) + uint8_t attributes; // File attributes + uint8_t reserved; // Reserved + uint8_t creation_time_tenths; // Creation time (tenths of second) + uint16_t creation_time; // Creation time (HH:MM:SS) + uint16_t creation_date; // Creation date (YYYY:MM:DD) + uint16_t last_access_date; // Last access date + uint16_t start_cluster_high; // Start cluster (high word) + uint16_t write_time; // Write time + uint16_t write_date; // Write date + uint16_t start_cluster_low; // Start cluster (low word) + uint32_t file_size; // File size +} __attribute__((packed)) FAT32_DirEntry; + +// File Attributes +#define ATTR_READ_ONLY 0x01 +#define ATTR_HIDDEN 0x02 +#define ATTR_SYSTEM 0x04 +#define ATTR_VOLUME_ID 0x08 +#define ATTR_DIRECTORY 0x10 +#define ATTR_ARCHIVE 0x20 +#define ATTR_DEVICE 0x40 +#define ATTR_RESERVED 0x80 + +// FAT32 Constants +#define FAT32_SECTOR_SIZE 512 +#define FAT32_CLUSTER_SIZE 4096 // 8 sectors per cluster +#define FAT32_MAX_FILENAME 256 +#define FAT32_MAX_PATH 1024 +#define FAT32_ROOT_CLUSTER 2 + +// File Handle +typedef struct { + uint32_t cluster; // Current cluster + uint32_t position; // Current position in file + uint32_t size; // File size + uint32_t mode; // 0=read, 1=write, 2=append + bool valid; // Is this handle valid? +} FAT32_FileHandle; + +// Directory Entry Info (for listing) +typedef struct { + char name[FAT32_MAX_FILENAME]; + uint32_t size; + bool is_directory; + uint32_t start_cluster; + uint16_t write_date; + uint16_t write_time; +} FAT32_FileInfo; + +// === Function Declarations === + +// Initialization +void fat32_init(void); + +// File Operations +FAT32_FileHandle* fat32_open(const char *path, const char *mode); +void fat32_close(FAT32_FileHandle *handle); +int fat32_read(FAT32_FileHandle *handle, void *buffer, int size); +int fat32_write(FAT32_FileHandle *handle, const void *buffer, int size); +int fat32_seek(FAT32_FileHandle *handle, int offset, int whence); + +// Directory Operations +bool fat32_mkdir(const char *path); +bool fat32_rmdir(const char *path); +bool fat32_delete(const char *path); +bool fat32_exists(const char *path); +bool fat32_is_directory(const char *path); + +// Listing +int fat32_list_directory(const char *path, FAT32_FileInfo *entries, int max_entries); + +// Working Directory +bool fat32_chdir(const char *path); +void fat32_get_current_dir(char *buffer, int size); + +// Utilities +void fat32_normalize_path(const char *path, char *normalized); + +#endif diff --git a/src/kernel/font.h b/src/kernel/font.h new file mode 100644 index 0000000..c595c58 --- /dev/null +++ b/src/kernel/font.h @@ -0,0 +1,206 @@ +#ifndef FONT_H +#define FONT_H + +#include + +// Minimal 8x8 font for ASCII 32-127 +// Derived from standard VGA font +static const uint8_t font8x8_basic[128][8] = { + // 0-31 Control chars (empty) + {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, + {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, {0}, + // 32 Space + {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}, + // 33 ! + {0x18, 0x3C, 0x3C, 0x18, 0x18, 0x00, 0x18, 0x00}, + // 34 " + {0x66, 0x66, 0x22, 0x00, 0x00, 0x00, 0x00, 0x00}, + // 35 # + {0x6C, 0x6C, 0xFE, 0x6C, 0xFE, 0x6C, 0x6C, 0x00}, + // 36 $ + {0x18, 0x3E, 0x60, 0x3C, 0x06, 0x7C, 0x18, 0x00}, + // 37 % + {0x00, 0xC6, 0xCC, 0x18, 0x30, 0x66, 0xC6, 0x00}, + // 38 & + {0x38, 0x6C, 0x38, 0x76, 0xDC, 0xCC, 0x76, 0x00}, + // 39 ' + {0x18, 0x18, 0x30, 0x00, 0x00, 0x00, 0x00, 0x00}, + // 40 ( + {0x0C, 0x18, 0x30, 0x30, 0x30, 0x18, 0x0C, 0x00}, + // 41 ) + {0x30, 0x18, 0x0C, 0x0C, 0x0C, 0x18, 0x30, 0x00}, + // 42 * + {0x00, 0x66, 0x3C, 0xFF, 0x3C, 0x66, 0x00, 0x00}, + // 43 + + {0x00, 0x18, 0x18, 0x7E, 0x18, 0x18, 0x00, 0x00}, + // 44 , + {0x00, 0x00, 0x00, 0x00, 0x00, 0x18, 0x18, 0x30}, + // 45 - + {0x00, 0x00, 0x00, 0x7E, 0x00, 0x00, 0x00, 0x00}, + // 46 . + {0x00, 0x00, 0x00, 0x00, 0x00, 0x18, 0x18, 0x00}, + // 47 / + {0x00, 0x06, 0x0C, 0x18, 0x30, 0x60, 0xC0, 0x00}, + // 48 0 + {0x3C, 0x66, 0x6E, 0x76, 0x66, 0x66, 0x3C, 0x00}, + // 49 1 + {0x18, 0x18, 0x38, 0x18, 0x18, 0x18, 0x7E, 0x00}, + // 50 2 + {0x3C, 0x66, 0x06, 0x0C, 0x30, 0x60, 0x7E, 0x00}, + // 51 3 + {0x3C, 0x66, 0x06, 0x1C, 0x06, 0x66, 0x3C, 0x00}, + // 52 4 + {0x06, 0x0E, 0x1E, 0x36, 0x66, 0x7F, 0x06, 0x00}, + // 53 5 + {0x7E, 0x60, 0x7C, 0x06, 0x06, 0x66, 0x3C, 0x00}, + // 54 6 + {0x3C, 0x66, 0x60, 0x7C, 0x66, 0x66, 0x3C, 0x00}, + // 55 7 + {0x7E, 0x06, 0x0C, 0x18, 0x30, 0x30, 0x30, 0x00}, + // 56 8 + {0x3C, 0x66, 0x66, 0x3C, 0x66, 0x66, 0x3C, 0x00}, + // 57 9 + {0x3C, 0x66, 0x66, 0x3E, 0x06, 0x66, 0x3C, 0x00}, + // 58 : + {0x00, 0x18, 0x18, 0x00, 0x00, 0x18, 0x18, 0x00}, + // 59 ; + {0x00, 0x18, 0x18, 0x00, 0x00, 0x18, 0x18, 0x30}, + // 60 < + {0x0C, 0x18, 0x30, 0x60, 0x30, 0x18, 0x0C, 0x00}, + // 61 = + {0x00, 0x00, 0x7E, 0x00, 0x7E, 0x00, 0x00, 0x00}, + // 62 > + {0x30, 0x18, 0x0C, 0x06, 0x0C, 0x18, 0x30, 0x00}, + // 63 ? + {0x3C, 0x66, 0x06, 0x0C, 0x18, 0x00, 0x18, 0x00}, + // 64 @ + {0x3C, 0x66, 0x6E, 0x6A, 0x68, 0x60, 0x3C, 0x00}, + // 65 A + {0x18, 0x3C, 0x66, 0x66, 0x7E, 0x66, 0x66, 0x00}, + // 66 B + {0x7C, 0x66, 0x66, 0x7C, 0x66, 0x66, 0x7C, 0x00}, + // 67 C + {0x3C, 0x66, 0x60, 0x60, 0x60, 0x66, 0x3C, 0x00}, + // 68 D + {0x78, 0x6C, 0x66, 0x66, 0x66, 0x6C, 0x78, 0x00}, + // 69 E + {0x7E, 0x60, 0x60, 0x7C, 0x60, 0x60, 0x7E, 0x00}, + // 70 F + {0x7E, 0x60, 0x60, 0x7C, 0x60, 0x60, 0x60, 0x00}, + // 71 G + {0x3C, 0x66, 0x60, 0x6E, 0x66, 0x66, 0x3C, 0x00}, + // 72 H + {0x66, 0x66, 0x66, 0x7E, 0x66, 0x66, 0x66, 0x00}, + // 73 I + {0x3C, 0x18, 0x18, 0x18, 0x18, 0x18, 0x3C, 0x00}, + // 74 J + {0x06, 0x06, 0x06, 0x06, 0x06, 0x66, 0x3C, 0x00}, + // 75 K + {0x66, 0x6C, 0x78, 0x70, 0x78, 0x6C, 0x66, 0x00}, + // 76 L + {0x60, 0x60, 0x60, 0x60, 0x60, 0x60, 0x7E, 0x00}, + // 77 M + {0x63, 0x77, 0x7F, 0x6B, 0x63, 0x63, 0x63, 0x00}, + // 78 N + {0x66, 0x76, 0x7E, 0x7E, 0x6E, 0x66, 0x66, 0x00}, + // 79 O + {0x3C, 0x66, 0x66, 0x66, 0x66, 0x66, 0x3C, 0x00}, + // 80 P + {0x7C, 0x66, 0x66, 0x7C, 0x60, 0x60, 0x60, 0x00}, + // 81 Q + {0x3C, 0x66, 0x66, 0x66, 0x6A, 0x3C, 0x0E, 0x00}, + // 82 R + {0x7C, 0x66, 0x66, 0x7C, 0x78, 0x6C, 0x66, 0x00}, + // 83 S + {0x3C, 0x66, 0x60, 0x3C, 0x06, 0x66, 0x3C, 0x00}, + // 84 T + {0x7E, 0x18, 0x18, 0x18, 0x18, 0x18, 0x18, 0x00}, + // 85 U + {0x66, 0x66, 0x66, 0x66, 0x66, 0x66, 0x3C, 0x00}, + // 86 V + {0x66, 0x66, 0x66, 0x66, 0x66, 0x3C, 0x18, 0x00}, + // 87 W + {0x63, 0x63, 0x63, 0x6B, 0x7F, 0x77, 0x63, 0x00}, + // 88 X + {0x66, 0x66, 0x3C, 0x18, 0x3C, 0x66, 0x66, 0x00}, + // 89 Y + {0x66, 0x66, 0x66, 0x3C, 0x18, 0x18, 0x18, 0x00}, + // 90 Z + {0x7E, 0x06, 0x0C, 0x18, 0x30, 0x60, 0x7E, 0x00}, + // 91 [ + {0x1E, 0x18, 0x18, 0x18, 0x18, 0x18, 0x1E, 0x00}, + // 92 backslash + {0x00, 0xC0, 0x60, 0x30, 0x18, 0x0C, 0x06, 0x00}, + // 93 ] + {0x78, 0x18, 0x18, 0x18, 0x18, 0x18, 0x78, 0x00}, + // 94 ^ + {0x08, 0x1C, 0x36, 0x63, 0x00, 0x00, 0x00, 0x00}, + // 95 _ + {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF}, + // 96 ` + {0x18, 0x0C, 0x06, 0x00, 0x00, 0x00, 0x00, 0x00}, + // 97 a + {0x00, 0x00, 0x3C, 0x06, 0x3E, 0x66, 0x3E, 0x00}, + // 98 b + {0x60, 0x60, 0x7C, 0x66, 0x66, 0x66, 0x7C, 0x00}, + // 99 c + {0x00, 0x00, 0x3C, 0x66, 0x60, 0x66, 0x3C, 0x00}, + // 100 d + {0x06, 0x06, 0x3E, 0x66, 0x66, 0x66, 0x3E, 0x00}, + // 101 e + {0x00, 0x00, 0x3C, 0x66, 0x7E, 0x60, 0x3C, 0x00}, + // 102 f + {0x0C, 0x18, 0x18, 0x3E, 0x18, 0x18, 0x18, 0x00}, + // 103 g + {0x00, 0x00, 0x3E, 0x66, 0x66, 0x3E, 0x06, 0x3C}, + // 104 h + {0x60, 0x60, 0x7C, 0x66, 0x66, 0x66, 0x66, 0x00}, + // 105 i + {0x18, 0x00, 0x38, 0x18, 0x18, 0x18, 0x3C, 0x00}, + // 106 j + {0x06, 0x00, 0x06, 0x06, 0x06, 0x06, 0x3C, 0x00}, + // 107 k + {0x60, 0x60, 0x66, 0x6C, 0x78, 0x6C, 0x66, 0x00}, + // 108 l + {0x38, 0x18, 0x18, 0x18, 0x18, 0x18, 0x3C, 0x00}, + // 109 m + {0x00, 0x00, 0x76, 0x7F, 0x6B, 0x6B, 0x63, 0x00}, + // 110 n + {0x00, 0x00, 0x7C, 0x66, 0x66, 0x66, 0x66, 0x00}, + // 111 o + {0x00, 0x00, 0x3C, 0x66, 0x66, 0x66, 0x3C, 0x00}, + // 112 p + {0x00, 0x00, 0x7C, 0x66, 0x66, 0x7C, 0x60, 0x60}, + // 113 q + {0x00, 0x00, 0x3E, 0x66, 0x66, 0x3E, 0x06, 0x06}, + // 114 r + {0x00, 0x00, 0x5C, 0x62, 0x60, 0x60, 0x60, 0x00}, + // 115 s + {0x00, 0x00, 0x3C, 0x60, 0x3C, 0x06, 0x3C, 0x00}, + // 116 t + {0x10, 0x10, 0x3E, 0x10, 0x10, 0x10, 0x0C, 0x00}, + // 117 u + {0x00, 0x00, 0x66, 0x66, 0x66, 0x66, 0x3E, 0x00}, + // 118 v + {0x00, 0x00, 0x66, 0x66, 0x66, 0x3C, 0x18, 0x00}, + // 119 w + {0x00, 0x00, 0x63, 0x6B, 0x7F, 0x3E, 0x36, 0x00}, + // 120 x + {0x00, 0x00, 0x66, 0x3C, 0x18, 0x3C, 0x66, 0x00}, + // 121 y + {0x00, 0x00, 0x66, 0x66, 0x66, 0x3E, 0x06, 0x3C}, + // 122 z + {0x00, 0x00, 0x7E, 0x0C, 0x18, 0x30, 0x7E, 0x00}, + // 123 { + {0x0E, 0x18, 0x18, 0x70, 0x18, 0x18, 0x0E, 0x00}, + // 124 | + {0x18, 0x18, 0x18, 0x00, 0x18, 0x18, 0x18, 0x00}, + // 125 } + {0x70, 0x18, 0x18, 0x0E, 0x18, 0x18, 0x70, 0x00}, + // 126 ~ + {0x76, 0xDC, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}, + // 127 DEL + {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00} +}; + +#endif diff --git a/src/kernel/graphics.c b/src/kernel/graphics.c new file mode 100644 index 0000000..50a3a37 --- /dev/null +++ b/src/kernel/graphics.c @@ -0,0 +1,180 @@ +#include +#include "graphics.h" +#include "font.h" + +static struct limine_framebuffer *g_fb = NULL; +static uint32_t g_bg_color = 0xFF6B4423; // Coffee color by default + +// Dirty rectangle tracking +static DirtyRect g_dirty = {0, 0, 0, 0, false}; + +// Double buffering - allocate a back buffer +// Max screen size: 2048x2048 @ 32bpp = 16MB, but we'll allocate for common sizes +// Using a simple approach: allocate max size buffer +#define MAX_FB_WIDTH 2048 +#define MAX_FB_HEIGHT 2048 +static uint32_t g_back_buffer[MAX_FB_WIDTH * MAX_FB_HEIGHT] __attribute__((aligned(4096))); + +void graphics_init(struct limine_framebuffer *fb) { + g_fb = fb; + g_dirty.active = false; + // Initialize back buffer to black + for (int i = 0; i < MAX_FB_WIDTH * MAX_FB_HEIGHT; i++) { + g_back_buffer[i] = 0; + } +} + +int get_screen_width(void) { + return g_fb ? g_fb->width : 0; +} + +int get_screen_height(void) { + return g_fb ? g_fb->height : 0; +} + +// Merge new dirty rect with existing one +static void merge_dirty_rect(int x, int y, int w, int h) { + if (!g_dirty.active) { + g_dirty.x = x; + g_dirty.y = y; + g_dirty.w = w; + g_dirty.h = h; + g_dirty.active = true; + } else { + // Calculate union of two rectangles + int x1 = g_dirty.x; + int y1 = g_dirty.y; + int x2 = g_dirty.x + g_dirty.w; + int y2 = g_dirty.y + g_dirty.h; + + int new_x1 = x; + int new_y1 = y; + int new_x2 = x + w; + int new_y2 = y + h; + + g_dirty.x = new_x1 < x1 ? new_x1 : x1; + g_dirty.y = new_y1 < y1 ? new_y1 : y1; + g_dirty.w = (new_x2 > x2 ? new_x2 : x2) - g_dirty.x; + g_dirty.h = (new_y2 > y2 ? new_y2 : y2) - g_dirty.y; + } +} + +void graphics_mark_dirty(int x, int y, int w, int h) { + // Clamp to screen bounds + if (x < 0) { + w += x; + x = 0; + } + if (y < 0) { + h += y; + y = 0; + } + if (x + w > get_screen_width()) { + w = get_screen_width() - x; + } + if (y + h > get_screen_height()) { + h = get_screen_height() - y; + } + + if (w <= 0 || h <= 0) return; + + merge_dirty_rect(x, y, w, h); +} + +void graphics_mark_screen_dirty(void) { + g_dirty.x = 0; + g_dirty.y = 0; + g_dirty.w = get_screen_width(); + g_dirty.h = get_screen_height(); + g_dirty.active = true; +} + +DirtyRect graphics_get_dirty_rect(void) { + return g_dirty; +} + +void graphics_clear_dirty(void) { + g_dirty.active = false; +} + +void put_pixel(int x, int y, uint32_t color) { + if (!g_fb) return; + if (x < 0 || x >= (int)g_fb->width || y < 0 || y >= (int)g_fb->height) return; + + // Draw to back buffer + uint32_t pixel_offset = y * g_fb->width + x; + g_back_buffer[pixel_offset] = color; +} + +void draw_rect(int x, int y, int w, int h, uint32_t color) { + for (int i = 0; i < h; i++) { + for (int j = 0; j < w; j++) { + put_pixel(x + j, y + i, color); + } + } +} + +void draw_char(int x, int y, char c, uint32_t color) { + unsigned char uc = (unsigned char)c; + if (uc > 127) return; + const uint8_t *glyph = font8x8_basic[uc]; + + for (int row = 0; row < 8; row++) { + for (int col = 0; col < 8; col++) { + if ((glyph[row] >> (7 - col)) & 1) { + put_pixel(x + col, y + row, color); + } + } + } +} + +void draw_string(int x, int y, const char *s, uint32_t color) { + int cur_x = x; + int cur_y = y; + while (*s) { + if (*s == '\n') { + cur_x = x; + cur_y += 10; + } else { + draw_char(cur_x, cur_y, *s, color); + cur_x += 8; + } + s++; + } +} + +void draw_desktop_background(void) { + if (!g_fb) return; + draw_rect(0, 0, g_fb->width, g_fb->height, g_bg_color); +} + +void graphics_set_bg_color(uint32_t color) { + g_bg_color = color; +} + +// Double buffering functions +void graphics_clear_back_buffer(uint32_t color) { + if (!g_fb) return; + uint32_t *buf = g_back_buffer; + for (int i = 0; i < (int)g_fb->width * (int)g_fb->height; i++) { + *buf++ = color; + } +} + +void graphics_flip_buffer(void) { + if (!g_fb) return; + + // Copy back buffer to framebuffer + uint32_t *src = g_back_buffer; + uint8_t *dst = (uint8_t *)g_fb->address; + + for (int y = 0; y < (int)g_fb->height; y++) { + // Copy one scanline + uint32_t *dst_row = (uint32_t *)dst; + for (int x = 0; x < (int)g_fb->width; x++) { + dst_row[x] = src[x]; + } + src += g_fb->width; + dst += g_fb->pitch; + } +} diff --git a/src/kernel/graphics.h b/src/kernel/graphics.h new file mode 100644 index 0000000..c802943 --- /dev/null +++ b/src/kernel/graphics.h @@ -0,0 +1,36 @@ +#ifndef GRAPHICS_H +#define GRAPHICS_H + +#include +#include +#include "limine.h" + +// Dirty rectangle structure +typedef struct { + int x, y, w, h; + bool active; +} DirtyRect; + +void graphics_init(struct limine_framebuffer *fb); +void put_pixel(int x, int y, uint32_t color); +void draw_rect(int x, int y, int w, int h, uint32_t color); +void draw_char(int x, int y, char c, uint32_t color); +void draw_string(int x, int y, const char *s, uint32_t color); +void draw_desktop_background(void); +void graphics_set_bg_color(uint32_t color); + +// Get screen dimensions +int get_screen_width(void); +int get_screen_height(void); + +// Dirty rectangle management +void graphics_mark_dirty(int x, int y, int w, int h); +void graphics_mark_screen_dirty(void); +DirtyRect graphics_get_dirty_rect(void); +void graphics_clear_dirty(void); + +// Double buffering +void graphics_flip_buffer(void); +void graphics_clear_back_buffer(uint32_t color); + +#endif diff --git a/src/kernel/idt.c b/src/kernel/idt.c new file mode 100644 index 0000000..665e3ec --- /dev/null +++ b/src/kernel/idt.c @@ -0,0 +1,103 @@ +#include "idt.h" +#include "io.h" + +#define IDT_ENTRIES 256 + +struct idt_entry { + uint16_t isr_low; + uint16_t kernel_cs; + uint8_t ist; + uint8_t attributes; + uint16_t isr_mid; + uint32_t isr_high; + uint32_t reserved; +} __attribute__((packed)); + +struct idt_ptr { + uint16_t limit; + uint64_t base; +} __attribute__((packed)); + +static struct idt_entry idt[IDT_ENTRIES]; +static struct idt_ptr idtr; + +void idt_set_gate(uint8_t vector, void *isr, uint16_t cs, uint8_t flags) { + uint64_t addr = (uint64_t)isr; + idt[vector].isr_low = addr & 0xFFFF; + idt[vector].kernel_cs = cs; + idt[vector].ist = 0; + idt[vector].attributes = flags; + idt[vector].isr_mid = (addr >> 16) & 0xFFFF; + idt[vector].isr_high = (addr >> 32) & 0xFFFFFFFF; + idt[vector].reserved = 0; +} + +// Remap PIC +static void pic_remap(void) { + uint8_t a1, a2; + a1 = inb(0x21); + a2 = inb(0xA1); + + outb(0x20, 0x11); io_wait(); + outb(0xA0, 0x11); io_wait(); + outb(0x21, 0x20); io_wait(); // Master offset 0x20 (32) + outb(0xA1, 0x28); io_wait(); // Slave offset 0x28 (40) + outb(0x21, 0x04); io_wait(); + outb(0xA1, 0x02); io_wait(); + outb(0x21, 0x01); io_wait(); + outb(0xA1, 0x01); io_wait(); + + // Restore masks (but verify we don't mask IRQ 1 and 12) + // Actually, simple OSs often just mask everything except what they want. + // Let's unmask IRQ 1 (Keyboard) and IRQ 12 (Mouse) explicitly and mask others. + // 0xFD = 1111 1101 (IRQ 1 unmasked) + // 0xEF = 1110 1111 (IRQ 12 (4 on slave) unmasked) + + outb(0x21, 0xF9); // Unmask Keyboard (IRQ1) and Cascade (IRQ2) + outb(0xA1, 0xEF); // Unmask Mouse (IRQ12) +} + +// Set up PIT (Programmable Interval Timer) for ~60Hz (16.67ms intervals) +static void pit_setup(void) { + uint16_t divisor = 1193182 / 60; // ~60Hz + + // Send command byte + outb(0x43, 0x36); // Channel 0, lobyte/hibyte, mode 3 (square wave), binary + + // Send divisor + outb(0x40, divisor & 0xFF); + outb(0x40, (divisor >> 8) & 0xFF); +} + +void idt_init(void) { + uint16_t cs; + asm volatile ("mov %%cs, %0" : "=r"(cs)); + + for (int i = 0; i < IDT_ENTRIES; i++) { + idt[i] = (struct idt_entry){0}; + } + + pic_remap(); + + // Unmask IRQ 0 (Timer) in addition to IRQ 1 and 12 + outb(0x21, 0xF8); // Unmask Timer (IRQ0), Keyboard (IRQ1) and Cascade (IRQ2) + outb(0xA1, 0xEF); // Unmask Mouse (IRQ12) + + pit_setup(); +} + +void idt_register_interrupts(void) { + uint16_t cs; + asm volatile ("mov %%cs, %0" : "=r"(cs)); + + idt_set_gate(32, isr0_wrapper, cs, 0x8E); // Timer (IRQ 0) + idt_set_gate(33, isr1_wrapper, cs, 0x8E); // Keyboard (IRQ 1) + idt_set_gate(44, isr12_wrapper, cs, 0x8E); // Mouse (IRQ 12) +} + +void idt_load(void) { + idtr.base = (uint64_t)&idt; + idtr.limit = sizeof(struct idt_entry) * IDT_ENTRIES - 1; + asm volatile ("lidt %0" : : "m"(idtr)); + asm volatile ("sti"); +} diff --git a/src/kernel/idt.h b/src/kernel/idt.h new file mode 100644 index 0000000..c1aa2c5 --- /dev/null +++ b/src/kernel/idt.h @@ -0,0 +1,16 @@ +#ifndef IDT_H +#define IDT_H + +#include + +void idt_init(void); +void idt_set_gate(uint8_t vector, void *isr, uint16_t cs, uint8_t flags); +void idt_register_interrupts(void); +void idt_load(void); + +// ISR wrappers defined in assembly +extern void isr0_wrapper(void); // Timer +extern void isr1_wrapper(void); // Keyboard +extern void isr12_wrapper(void); // Mouse + +#endif diff --git a/src/kernel/interrupts.asm b/src/kernel/interrupts.asm new file mode 100644 index 0000000..2c8b1cb --- /dev/null +++ b/src/kernel/interrupts.asm @@ -0,0 +1,65 @@ +section .text +global isr0_wrapper +global isr1_wrapper +global isr12_wrapper +extern timer_handler +extern keyboard_handler +extern mouse_handler + +; Helper to send EOI (End of Interrupt) to PIC +send_eoi: + push rax + mov al, 0x20 + out 0x20, al ; Master PIC + ; If IRQ > 7, send to Slave too (Mouse is IRQ 12) + ; We'll handle this in the specific wrappers or C code. + ; Actually, simpler to do EOI in C or here. + ; Let's just do it in C. + pop rax + ret + +%macro ISR_NOERRCODE 1 + push rdi + push rsi + push rdx + push rcx + push r8 + push r9 + push rax + push rbx + push rbp + push r10 + push r11 + push r12 + push r13 + push r14 + push r15 + + call %1 + + pop r15 + pop r14 + pop r13 + pop r12 + pop r11 + pop r10 + pop rbp + pop rbx + pop rax + pop r9 + pop r8 + pop rcx + pop rdx + pop rsi + pop rdi + iretq +%endmacro + +isr0_wrapper: + ISR_NOERRCODE timer_handler + +isr1_wrapper: + ISR_NOERRCODE keyboard_handler + +isr12_wrapper: + ISR_NOERRCODE mouse_handler diff --git a/src/kernel/io.h b/src/kernel/io.h new file mode 100644 index 0000000..98edf6e --- /dev/null +++ b/src/kernel/io.h @@ -0,0 +1,24 @@ +#ifndef IO_H +#define IO_H + +#include + +static inline void outb(uint16_t port, uint8_t val) { + asm volatile ("outb %0, %1" : : "a"(val), "Nd"(port)); +} + +static inline void outw(uint16_t port, uint16_t val) { + asm volatile ("outw %0, %1" : : "a"(val), "Nd"(port)); +} + +static inline uint8_t inb(uint16_t port) { + uint8_t ret; + asm volatile ("inb %1, %0" : "=a"(ret) : "Nd"(port)); + return ret; +} + +static inline void io_wait(void) { + outb(0x80, 0); +} + +#endif diff --git a/src/kernel/licensewr.c b/src/kernel/licensewr.c new file mode 100644 index 0000000..f153f47 --- /dev/null +++ b/src/kernel/licensewr.c @@ -0,0 +1,573 @@ +#include "fat32.h" +#include + +// Helper function to calculate string length +static size_t lic_strlen(const char *str) { + size_t len = 0; + while (str[len]) len++; + return len; +} + +void write_license_file(void) { + FAT32_FileHandle *fh = fat32_open("LICENSE", "w"); + if (fh) { + const char *content = + " GNU GENERAL PUBLIC LICENSE\n" + " Version 3, 29 June 2007\n\n" + " Copyright(C) Chris (boreddevnl) 2024-2026\n\n" + " Copyright (C) 2007 Free Software Foundation, Inc. \n" + " Everyone is permitted to copy and distribute verbatim copies\n" + " of this license document, but changing it is not allowed.\n\n" + " Preamble\n\n" + " The GNU General Public License is a free, copyleft license for\n" + "software and other kinds of works.\n\n" + " The licenses for most software and other practical works are designed\n" + "to take away your freedom to share and change the works. By contrast,\n" + "the GNU General Public License is intended to guarantee your freedom to\n" + "share and change all versions of a program--to make sure it remains free\n" + "software for all its users. We, the Free Software Foundation, use the\n" + "GNU General Public License for most of our software; it applies also to\n" + "any other work released this way by its authors. You can apply it to\n" + "your programs, too.\n\n" + " When we speak of free software, we are referring to freedom, not\n" + "price. Our General Public Licenses are designed to make sure that you\n" + "have the freedom to distribute copies of free software (and charge for\n" + "them if you wish), that you receive source code or can get it if you\n" + "want it, that you can change the software or use pieces of it in new\n" + "free programs, and that you know you can do these things.\n\n" + " To protect your rights, we need to prevent others from denying you\n" + "these rights or asking you to surrender the rights. Therefore, you have\n" + "certain responsibilities if you distribute copies of the software, or if\n" + "you modify it: responsibilities to respect the freedom of others.\n\n" + " For example, if you distribute copies of such a program, whether\n" + "gratis or for a fee, you must pass on to the recipients the same\n" + "freedoms that you received. You must make sure that they, too, receive\n" + "or can get the source code. And you must show them these terms so they\n" + "know their rights.\n\n" + " Developers that use the GNU GPL protect your rights with two steps:\n" + "(1) assert copyright on the software, and (2) offer you this License\n" + "giving you legal permission to copy, distribute and/or modify it.\n\n" + " For the developers' and authors' protection, the GPL clearly explains\n" + "that there is no warranty for this free software. For both users' and\n" + "authors' sake, the GPL requires that modified versions be marked as\n" + "changed, so that their problems will not be attributed erroneously to\n" + "authors of previous versions.\n\n" + " Some devices are designed to deny users access to install or run\n" + "modified versions of the software inside them, although the manufacturer\n" + "can do so. This is fundamentally incompatible with the aim of\n" + "protecting users' freedom to change the software. The systematic\n" + "pattern of such abuse occurs in the area of products for individuals to\n" + "use, which is precisely where it is most unacceptable. Therefore, we\n" + "have designed this version of the GPL to prohibit the practice for those\n" + "products. If such problems arise substantially in other domains, we\n" + "stand ready to extend this provision to those domains in future versions\n" + "of the GPL, as needed to protect the freedom of users.\n\n" + " Finally, every program is threatened constantly by software patents.\n" + "States should not allow patents to restrict development and use of\n" + "software on general-purpose computers, but in those that do, we wish to\n" + "avoid the special danger that patents applied to a free program could\n" + "make it effectively proprietary. To prevent this, the GPL assures that\n" + "patents cannot be used to render the program non-free.\n\n" + " The precise terms and conditions for copying, distribution and\n" + "modification follow.\n\n" + " TERMS AND CONDITIONS\n\n" + " 0. Definitions.\n\n" + " \"This License\" refers to version 3 of the GNU General Public License.\n\n" + " \"Copyright\" also means copyright-like laws that apply to other kinds of\n" + "works, such as semiconductor masks.\n\n" + " \"The Program\" refers to any copyrightable work licensed under this\n" + "License. Each licensee is addressed as \"you\". \"Licensees\" and\n" + "\"recipients\" may be individuals or organizations.\n\n" + " To \"modify\" a work means to copy from or adapt all or part of the work\n" + "in a fashion requiring copyright permission, other than the making of an\n" + "exact copy. The resulting work is called a \"modified version\" of the\n" + "earlier work or a work \"based on\" the earlier work.\n\n" + " A \"covered work\" means either the unmodified Program or a work based\n" + "on the Program.\n\n" + " To \"propagate\" a work means to do anything with it that, without\n" + "permission, would make you directly or secondarily liable for\n" + "infringement under applicable copyright law, except executing it on a\n" + "computer or modifying a private copy. Propagation includes copying,\n" + "distribution (with or without modification), making available to the\n" + "public, and in some countries other activities as well.\n\n" + " To \"convey\" a work means any kind of propagation that enables other\n" + "parties to make or receive copies. Mere interaction with a user through\n" + "a computer network, with no transfer of a copy, is not conveying.\n\n" + " An interactive user interface displays \"Appropriate Legal Notices\"\n" + "to the extent that it includes a convenient and prominently visible\n" + "feature that (1) displays an appropriate copyright notice, and (2)\n" + "tells the user that there is no warranty for the work (except to the\n" + "extent that warranties are provided), that licensees may convey the\n" + "work under this License, and how to view a copy of this License. If\n" + "the interface presents a list of user commands or options, such as a\n" + "menu, a prominent item in the list meets this criterion.\n\n" + " 1. Source Code.\n\n" + " The \"source code\" for a work means the preferred form of the work\n" + "for making modifications to it. \"Object code\" means any non-source\n" + "form of a work.\n\n" + " A \"Standard Interface\" means an interface that either is an official\n" + "standard defined by a recognized standards body, or, in the case of\n" + "interfaces specified for a particular programming language, one that\n" + "is widely used among developers working in that language.\n\n" + " The \"System Libraries\" of an executable work include anything, other\n" + "than the work as a whole, that (a) is included in the normal form of\n" + "packaging a Major Component, but which is not part of that Major\n" + "Component, and (b) serves only to enable use of the work with that\n" + "Major Component, or to implement a Standard Interface for which an\n" + "implementation is available to the public in source code form. A\n" + "\"Major Component\", in this context, means a major essential component\n" + "(kernel, window system, and so on) of the specific operating system\n" + "(if any) on which the executable work runs, or a compiler used to\n" + "produce the work, or an object code interpreter used to run it.\n\n" + " The \"Corresponding Source\" for a work in object code form means all\n" + "the source code needed to generate, install, and (for an executable\n" + "work) run the object code and to modify the work, including scripts to\n" + "control those activities. However, it does not include the work's\n" + "System Libraries, or general-purpose tools or generally available free\n" + "programs which are used unmodified in performing those activities but\n" + "which are not part of the work. For example, Corresponding Source\n" + "includes interface definition files associated with source files for\n" + "the work, and the source code for shared libraries and dynamically\n" + "linked subprograms that the work is specifically designed to require,\n" + "such as by intimate data communication or control flow between those\n" + "subprograms and other parts of the work.\n\n" + " The Corresponding Source need not include anything that users\n" + "can regenerate automatically from other parts of the Corresponding\n" + "Source.\n\n" + " The Corresponding Source for a work in source code form is that\n" + "same work.\n\n" + " 2. Basic Permissions.\n\n" + " All rights granted under this License are granted for the term of\n" + "copyright on the Program, and are irrevocable provided the stated\n" + "conditions are met. This License explicitly affirms your unlimited\n" + "permission to run the unmodified Program. The output from running a\n" + "covered work is covered by this License only if the output, given its\n" + "content, constitutes a covered work. This License acknowledges your\n" + "rights of fair use or other equivalent, as provided by copyright law.\n\n" + " You may make, run and propagate covered works that you do not\n" + "convey, without conditions so long as your license otherwise remains\n" + "in force. You may convey covered works to others for the sole purpose\n" + "of having them make modifications exclusively for you, or provide you\n" + "with facilities for running those works, provided that you comply with\n" + "the terms of this License in conveying all material for which you do\n" + "not control copyright. Those thus making or running the covered works\n" + "for you must do so exclusively on your behalf, under your direction\n" + "and control, on terms that prohibit them from making any copies of\n" + "your copyrighted material outside their relationship with you.\n\n" + " Conveying under any other circumstances is permitted solely under\n" + "the conditions stated below. Sublicensing is not allowed; section 10\n" + "makes it unnecessary.\n\n" + " 3. Protecting Users' Legal Rights From Anti-Circumvention Law.\n\n" + " No covered work shall be deemed part of an effective technological\n" + "measure under any applicable law fulfilling obligations under article\n" + "11 of the WIPO copyright treaty adopted on 20 December 1996, or\n" + "similar laws prohibiting or restricting circumvention of such\n" + "measures.\n\n" + " When you convey a covered work, you waive any legal power to forbid\n" + "circumvention of technological measures to the extent such circumvention\n" + "is effected by exercising rights under this License with respect to\n" + "the covered work, and you disclaim any intention to limit operation or\n" + "modification of the work as a means of enforcing, against the work's\n" + "users, your or third parties' legal rights to forbid circumvention of\n" + "technological measures.\n\n" + " 4. Conveying Verbatim Copies.\n\n" + " You may convey verbatim copies of the Program's source code as you\n" + "receive it, in any medium, provided that you conspicuously and\n" + "appropriately publish on each copy an appropriate copyright notice;\n" + "keep intact all notices stating that this License and any\n" + "non-permissive terms added in accord with section 7 apply to the code;\n" + "keep intact all notices of the absence of any warranty; and give all\n" + "recipients a copy of this License along with the Program.\n\n" + " You may charge any price or no price for each copy that you convey,\n" + "and you may offer support or warranty protection for a fee.\n\n" + " 5. Conveying Modified Source Versions.\n\n" + " You may convey a work based on the Program, or the modifications to\n" + "produce it from the Program, in the form of source code under the\n" + "terms of section 4, provided that you also meet all of these conditions:\n\n" + " a) The work must carry prominent notices stating that you modified\n" + " it, and giving a relevant date.\n\n" + " b) The work must carry prominent notices stating that it is\n" + " released under this License and any conditions added under section\n" + " 7. This requirement modifies the requirement in section 4 to\n" + " \"keep intact all notices\".\n\n" + " c) You must license the entire work, as a whole, under this\n" + " License to anyone who comes into possession of a copy. This\n" + " License will therefore apply, along with any applicable section 7\n" + " additional terms, to the whole of the work, and all its parts,\n" + " regardless of how they are packaged. This License gives no\n" + " permission to license the work in any other way, but it does not\n" + " invalidate such permission if you have separately received it.\n\n" + " d) If the work has interactive user interfaces, each must display\n" + " Appropriate Legal Notices; however, if the Program has interactive\n" + " interfaces that do not display Appropriate Legal Notices, your\n" + " work need not make them do so.\n\n" + " A compilation of a covered work with other separate and independent\n" + "works, which are not by their nature extensions of the covered work,\n" + "and which are not combined with it such as to form a larger program,\n" + "in or on a volume of a storage or distribution medium, is called an\n" + "\"aggregate\" if the compilation and its resulting copyright are not\n" + "used to limit the access or legal rights of the compilation's users\n" + "beyond what the individual works permit. Inclusion of a covered work\n" + "in an aggregate does not cause this License to apply to the other\n" + "parts of the aggregate.\n\n" + " 6. Conveying Non-Source Forms.\n\n" + " You may convey a covered work in object code form under the terms\n" + "of sections 4 and 5, provided that you also convey the\n" + "machine-readable Corresponding Source under the terms of this License,\n" + "in one of these ways:\n\n" + " a) Convey the object code in, or embodied in, a physical product\n" + " (including a physical distribution medium), accompanied by the\n" + " Corresponding Source fixed on a durable physical medium\n" + " customarily used for software interchange.\n\n" + " b) Convey the object code in, or embodied in, a physical product\n" + " (including a physical distribution medium), accompanied by a\n" + " written offer, valid for at least three years and valid for as\n" + " long as you offer spare parts or customer support for that product\n" + " model, to give anyone who possesses the object code either (1) a\n" + " copy of the Corresponding Source for all the software in the\n" + " product that is covered by this License, on a durable physical\n" + " medium customarily used for software interchange, for a price no\n" + " more than your reasonable cost of physically performing this\n" + " conveying of source, or (2) access to copy the\n" + " Corresponding Source from a network server at no charge.\n\n" + " c) Convey individual copies of the object code with a copy of the\n" + " written offer to provide the Corresponding Source. This\n" + " alternative is allowed only occasionally and noncommercially, and\n" + " only if you received the object code with such an offer, in accord\n" + " with subsection 6b.\n\n" + " d) Convey the object code by offering access from a designated\n" + " place (gratis or for a charge), and offer equivalent access to the\n" + " Corresponding Source in the same way through the same place at no\n" + " further charge. You need not require recipients to copy the\n" + " Corresponding Source along with the object code. If the place to\n" + " copy the object code is a network server, the Corresponding Source\n" + " may be on a different server (operated by you or a third party)\n" + " that supports equivalent copying facilities, provided you maintain\n" + " clear directions next to the object code saying where to find the\n" + " Corresponding Source. Regardless of what server hosts the\n" + " Corresponding Source, you remain obligated to ensure that it is\n" + " available for as long as needed to satisfy these requirements.\n\n" + " e) Convey the object code using peer-to-peer transmission, provided\n" + " you inform other peers where the object code and Corresponding\n" + " Source of the work are being offered to the general public at no\n" + " charge under subsection 6d.\n\n" + " A separable portion of the object code, whose source code is excluded\n" + "from the Corresponding Source as a System Library, need not be\n" + "included in conveying the object code work.\n\n" + " A \"User Product\" is either (1) a \"consumer product\", which means any\n" + "tangible personal property which is normally used for personal, family,\n" + "or household purposes, or (2) anything designed or sold for incorporation\n" + "into a dwelling. In determining whether a product is a consumer product,\n" + "doubtful cases shall be resolved in favor of coverage. For a particular\n" + "product received by a particular user, \"normally used\" refers to a\n" + "typical or common use of that class of product, regardless of the status\n" + "of the particular user or of the way in which the particular user\n" + "actually uses, or expects or is expected to use, the product. A product\n" + "is a consumer product regardless of whether the product has substantial\n" + "commercial, industrial or non-consumer uses, unless such uses represent\n" + "the only significant mode of use of the product.\n\n" + " \"Installation Information\" for a User Product means any methods,\n" + "procedures, authorization keys, or other information required to install\n" + "and execute modified versions of a covered work in that User Product from\n" + "a modified version of its Corresponding Source. The information must\n" + "suffice to ensure that the continued functioning of the modified object\n" + "code is in no case prevented or interfered with solely because\n" + "modification has been made.\n\n" + " If you convey an object code work under this section in, or with, or\n" + "specifically for use in, a User Product, and the conveying occurs as\n" + "part of a transaction in which the right of possession and use of the\n" + "User Product is transferred to the recipient in perpetuity or for a\n" + "fixed term (regardless of how the transaction is characterized), the\n" + "Corresponding Source conveyed under this section must be accompanied\n" + "by the Installation Information. But this requirement does not apply\n" + "if neither you nor any third party retains the ability to install\n" + "modified object code on the User Product (for example, the work has\n" + "been installed in ROM).\n\n" + " The requirement to provide Installation Information does not include a\n" + "requirement to continue to provide support service, warranty, or updates\n" + "for a work that has been modified or installed by the recipient, or for\n" + "the User Product in which it has been modified or installed. Access to a\n" + "network may be denied when the modification itself materially and\n" + "adversely affects the operation of the network or violates the rules and\n" + "protocols for communication across the network.\n\n" + " Corresponding Source conveyed, and Installation Information provided,\n" + "in accord with this section must be in a format that is publicly\n" + "documented (and with an implementation available to the public in\n" + "source code form), and must require no special password or key for\n" + "unpacking, reading or copying.\n\n" + " 7. Additional Terms.\n\n" + " \"Additional permissions\" are terms that supplement the terms of this\n" + "License by making exceptions from one or more of its conditions.\n" + "Additional permissions that are applicable to the entire Program shall\n" + "be treated as though they were included in this License, to the extent\n" + "that they are valid under applicable law. If additional permissions\n" + "apply only to part of the Program, that part may be used separately\n" + "under those permissions, but the entire Program remains governed by\n" + "this License without regard to the additional permissions.\n\n" + " When you convey a copy of a covered work, you may at your option\n" + "remove any additional permissions from that copy, or from any part of\n" + "it. (Additional permissions may be written to require their own\n" + "removal in certain cases when you modify the work.) You may place\n" + "additional permissions on material, added by you to a covered work,\n" + "for which you have or can give appropriate copyright permission.\n\n" + " Notwithstanding any other provision of this License, for material you\n" + "add to a covered work, you may (if authorized by the copyright holders of\n" + "that material) supplement the terms of this License with terms:\n\n" + " a) Disclaiming warranty or limiting liability differently from the\n" + " terms of sections 15 and 16 of this License; or\n\n" + " b) Requiring preservation of specified reasonable legal notices or\n" + " author attributions in that material or in the Appropriate Legal\n" + " Notices displayed by works containing it; or\n\n" + " c) Prohibiting misrepresentation of the origin of that material, or\n" + " requiring that modified versions of such material be marked in\n" + " reasonable ways as different from the original version; or\n\n" + " d) Limiting the use for publicity purposes of names of licensors or\n" + " authors of the material; or\n\n" + " e) Declining to grant rights under trademark law for use of some\n" + " trade names, trademarks, or service marks; or\n\n" + " f) Requiring indemnification of licensors and authors of that\n" + " material by anyone who conveys the material (or modified versions of\n" + " it) with contractual assumptions of liability to the recipient, for\n" + " any liability that these contractual assumptions directly impose on\n" + " those licensors and authors.\n\n" + " All other non-permissive additional terms are considered \"further\n" + "restrictions\" within the meaning of section 10. If the Program as you\n" + "received it, or any part of it, contains a notice stating that it is\n" + "governed by this License along with a term that is a further\n" + "restriction, you may remove that term. If a license document contains\n" + "a further restriction but permits relicensing or conveying under this\n" + "License, you may add to a covered work material governed by the terms\n" + "of that license document, provided that the further restriction does\n" + "not survive such relicensing or conveying.\n\n" + " If you add terms to a covered work in accord with this section, you\n" + "must place, in the relevant source files, a statement of the\n" + "additional terms that apply to those files, or a notice indicating\n" + "where to find the applicable terms.\n\n" + " Additional terms, permissive or non-permissive, may be stated in the\n" + "form of a separately written license, or stated as exceptions;\n" + "the above requirements apply either way.\n\n" + " 8. Termination.\n\n" + " You may not propagate or modify a covered work except as expressly\n" + "provided under this License. Any attempt otherwise to propagate or\n" + "modify it is void, and will automatically terminate your rights under\n" + "this License (including any patent licenses granted under the third\n" + "paragraph of section 11).\n\n" + " However, if you cease all violation of this License, then your\n" + "license from a particular copyright holder is reinstated (a)\n" + "provisionally, unless and until the copyright holder explicitly and\n" + "finally terminates your license, and (b) permanently, if the copyright\n" + "holder fails to notify you of the violation by some reasonable means\n" + "prior to 60 days after the cessation.\n\n" + " Moreover, your license from a particular copyright holder is\n" + "reinstated permanently if the copyright holder notifies you of the\n" + "violation by some reasonable means, this is the first time you have\n" + "received notice of violation of this License (for any work) from that\n" + "copyright holder, and you cure the violation prior to 30 days after\n" + "your receipt of the notice.\n\n" + " Termination of your rights under this section does not terminate the\n" + "licenses of parties who have received copies or rights from you under\n" + "this License. If your rights have been terminated and not permanently\n" + "reinstated, you do not qualify to receive new licenses for the same\n" + "material under section 10.\n\n" + " 9. Acceptance Not Required for Having Copies.\n\n" + " You are not required to accept this License in order to receive or\n" + "run a copy of the Program. Ancillary propagation of a covered work\n" + "occurring solely as a consequence of using peer-to-peer transmission\n" + "to receive a copy likewise does not require acceptance. However,\n" + "nothing other than this License grants you permission to propagate or\n" + "modify any covered work. These actions infringe copyright if you do\n" + "not accept this License. Therefore, by modifying or propagating a\n" + "covered work, you indicate your acceptance of this License to do so.\n\n" + " 10. Automatic Licensing of Downstream Recipients.\n\n" + " Each time you convey a covered work, the recipient automatically\n" + "receives a license from the original licensors, to run, modify and\n" + "propagate that work, subject to this License. You are not responsible\n" + "for enforcing compliance by third parties with this License.\n\n" + " An \"entity transaction\" is a transaction transferring control of an\n" + "organization, or substantially all assets of one, or subdividing an\n" + "organization, or merging organizations. If propagation of a covered\n" + "work results from an entity transaction, each party to that\n" + "transaction who receives a copy of the work also receives whatever\n" + "licenses to the work the party's predecessor in interest had or could\n" + "give under the previous paragraph, plus a right to possession of the\n" + "Corresponding Source of the work from the predecessor in interest, if\n" + "the predecessor has it or can get it with reasonable efforts.\n\n" + " You may not impose any further restrictions on the exercise of the\n" + "rights granted or affirmed under this License. For example, you may\n" + "not impose a license fee, royalty, or other charge for exercise of\n" + "rights granted under this License, and you may not initiate litigation\n" + "(including a cross-claim or counterclaim in a lawsuit) alleging that\n" + "any patent claim is infringed by making, using, selling, offering for\n" + "sale, or importing the Program or any portion of it.\n\n" + " 11. Patents.\n\n" + " A \"contributor\" is a copyright holder who authorizes use under this\n" + "License of the Program or a work on which the Program is based. The\n" + "work thus licensed is called the contributor's \"contributor version\".\n\n" + " A contributor's \"essential patent claims\" are all patent claims\n" + "owned or controlled by the contributor, whether already acquired or\n" + "hereafter acquired, that would be infringed by some manner, permitted\n" + "by this License, of making, using, or selling its contributor version,\n" + "but do not include claims that would be infringed only as a\n" + "consequence of further modification of the contributor version. For\n" + "purposes of this definition, \"control\" includes the right to grant\n" + "patent sublicenses in a manner consistent with the requirements of\n" + "this License.\n\n" + " Each contributor grants you a non-exclusive, worldwide, royalty-free\n" + "patent license under the contributor's essential patent claims, to\n" + "make, use, sell, offer for sale, import and otherwise run, modify and\n" + "propagate the contents of its contributor version.\n\n" + " In the following three paragraphs, a \"patent license\" is any express\n" + "agreement or commitment, however denominated, not to enforce a patent\n" + "(such as an express permission to practice a patent or covenant not to\n" + "sue for patent infringement). To \"grant\" such a patent license to a\n" + "party means to make such an agreement or commitment not to enforce a\n" + "patent against the party.\n\n" + " If you convey a covered work, knowingly relying on a patent license,\n" + "and the Corresponding Source of the work is not available for anyone\n" + "to copy, free of charge and under the terms of this License, through a\n" + "publicly available network server or other readily accessible means,\n" + "then you must either (1) cause the Corresponding Source to be so\n" + "available, or (2) arrange to deprive yourself of the benefit of the\n" + "patent license for this particular work, or (3) arrange, in a manner\n" + "consistent with the requirements of this License, to extend the patent\n" + "license to downstream recipients. \"Knowingly relying\" means you have\n" + "actual knowledge that, but for the patent license, your conveying the\n" + "covered work in a country, or your recipient's use of the covered work\n" + "in a country, would infringe one or more identifiable patents in that\n" + "country that you have reason to believe are valid.\n\n" + " If, pursuant to or in connection with a single transaction or\n" + "arrangement, you convey, or propagate by procuring conveyance of, a\n" + "covered work, and grant a patent license to some of the parties\n" + "receiving the covered work authorizing them to use, propagate, modify\n" + "or convey a specific copy of the covered work, then the patent license\n" + "you grant is automatically extended to all recipients of the covered\n" + "work and works based on it.\n\n" + " A patent license is \"discriminatory\" if it does not include within\n" + "the scope of its coverage, prohibits the exercise of, or is\n" + "conditioned on the non-exercise of one or more of the rights that are\n" + "specifically granted under this License. You may not convey a covered\n" + "work if you are a party to an arrangement with a third party that is\n" + "in the business of distributing software, under which you make payment\n" + "to the third party based on the extent of your activity of conveying\n" + "the work, and under which the third party grants, to any of the\n" + "parties who would receive the covered work from you, a discriminatory\n" + "patent license (a) in connection with copies of the covered work\n" + "conveyed by you (or copies made from those copies), or (b) primarily\n" + "for and in connection with specific products or compilations that\n" + "contain the covered work, unless you entered into that arrangement,\n" + "or that patent license was granted, prior to 28 March 2007.\n\n" + " Nothing in this License shall be construed as excluding or limiting\n" + "any implied license or other defenses to infringement that may\n" + "otherwise be available to you under applicable patent law.\n\n" + " 12. No Surrender of Others' Freedom.\n\n" + " If conditions are imposed on you (whether by court order, agreement or\n" + "otherwise) that contradict the conditions of this License, they do not\n" + "excuse you from the conditions of this License. If you cannot convey a\n" + "covered work so as to satisfy simultaneously your obligations under this\n" + "License and any other pertinent obligations, then as a consequence you may\n" + "not convey it at all. For example, if you agree to terms that obligate you\n" + "to collect a royalty for further conveying from those to whom you convey\n" + "the Program, the only way you could satisfy both those terms and this\n" + "License would be to refrain entirely from conveying the Program.\n\n" + " 13. Use with the GNU Affero General Public License.\n\n" + " Notwithstanding any other provision of this License, you have\n" + "permission to link or combine any covered work with a work licensed\n" + "under version 3 of the GNU Affero General Public License into a single\n" + "combined work, and to convey the resulting work. The terms of this\n" + "License will continue to apply to the part which is the covered work,\n" + "but the special requirements of the GNU Affero General Public License,\n" + "section 13, concerning interaction through a network will apply to the\n" + "combination as such.\n\n" + " 14. Revised Versions of this License.\n\n" + " The Free Software Foundation may publish revised and/or new versions of\n" + "the GNU General Public License from time to time. Such new versions will\n" + "be similar in spirit to the present version, but may differ in detail to\n" + "address new problems or concerns.\n\n" + " Each version is given a distinguishing version number. If the\n" + "Program specifies that a certain numbered version of the GNU General\n" + "Public License \"or any later version\" applies to it, you have the\n" + "option of following the terms and conditions either of that numbered\n" + "version or of any later version published by the Free Software\n" + "Foundation. If the Program does not specify a version number of the\n" + "GNU General Public License, you may choose any version ever published\n" + "by the Free Software Foundation.\n\n" + " If the Program specifies that a proxy can decide which future\n" + "versions of the GNU General Public License can be used, that proxy's\n" + "public statement of acceptance of a version permanently authorizes you\n" + "to choose that version for the Program.\n\n" + " Later license versions may give you additional or different\n" + "permissions. However, no additional obligations are imposed on any\n" + "author or copyright holder as a result of your choosing to follow a\n" + "later version.\n\n" + " 15. Disclaimer of Warranty.\n\n" + " THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY\n" + "APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT\n" + "HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM \"AS IS\" WITHOUT WARRANTY\n" + "OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,\n" + "THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR\n" + "PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM\n" + "IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF\n" + "ALL NECESSARY SERVICING, REPAIR OR CORRECTION.\n\n" + " 16. Limitation of Liability.\n\n" + " IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING\n" + "WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS\n" + "THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY\n" + "GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE\n" + "USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF\n" + "DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD\n" + "PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),\n" + "EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF\n" + "SUCH DAMAGES.\n\n" + " 17. Interpretation of Sections 15 and 16.\n\n" + " If the disclaimer of warranty and limitation of liability provided\n" + "above cannot be given local legal effect according to their terms,\n" + "reviewing courts shall apply local law that most closely approximates\n" + "an absolute waiver of all civil liability in connection with the\n" + "Program, unless a warranty or assumption of liability accompanies a\n" + "copy of the Program in return for a fee.\n\n" + " END OF TERMS AND CONDITIONS\n\n" + " How to Apply These Terms to Your New Programs\n\n" + " If you develop a new program, and you want it to be of the greatest\n" + "possible use to the public, the best way to achieve this is to make it\n" + "free software which everyone can redistribute and change under these terms.\n\n" + " To do so, attach the following notices to the program. It is safest\n" + "to attach them to the start of each source file to most effectively\n" + "state the exclusion of warranty; and each file should have at least\n" + "the \"copyright\" line and a pointer to where the full notice is found.\n\n" + " \n" + " Copyright (C) \n\n" + " This program is free software: you can redistribute it and/or modify\n" + " it under the terms of the GNU General Public License as published by\n" + " the Free Software Foundation, either version 3 of the License, or\n" + " (at your option) any later version.\n\n" + " This program is distributed in the hope that it will be useful,\n" + " but WITHOUT ANY WARRANTY; without even the implied warranty of\n" + " MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\n" + " GNU General Public License for more details.\n\n" + " You should have received a copy of the GNU General Public License\n" + " along with this program. If not, see .\n\n" + "Also add information on how to contact you by electronic and paper mail.\n\n" + " If the program does terminal interaction, make it output a short\n" + "notice like this when it starts in an interactive mode:\n\n" + " Copyright (C) \n" + " This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.\n" + " This is free software, and you are welcome to redistribute it\n" + " under certain conditions; type `show c' for details.\n\n" + "The hypothetical commands `show w' and `show c' should show the appropriate\n" + "parts of the General Public License. Of course, your program's commands\n" + "might be different; for a GUI interface, you would use an \"about box\".\n\n" + " You should also get your employer (if you work as a programmer) or school,\n" + "if any, to sign a \"copyright disclaimer\" for the program, if necessary.\n" + "For more information on this, and how to apply and follow the GNU GPL, see\n" + ".\n\n" + " The GNU General Public License does not permit incorporating your program\n" + "into proprietary programs. If your program is a subroutine library, you\n" + "may consider it more useful to permit linking proprietary applications with\n" + "the library. If this is what you want to do, use the GNU Lesser General\n" + "Public License instead of this License. But first, please read\n" + ".\n"; + + fat32_write(fh, (void *)content, lic_strlen(content)); + fat32_close(fh); + } +} diff --git a/src/kernel/licensewr.h b/src/kernel/licensewr.h new file mode 100644 index 0000000..e6eb9d7 --- /dev/null +++ b/src/kernel/licensewr.h @@ -0,0 +1,6 @@ +#ifndef LICENSEWR_H +#define LICENSEWR_H + +void write_license_file(void); + +#endif diff --git a/src/kernel/limine.h b/src/kernel/limine.h new file mode 100644 index 0000000..c331a6c --- /dev/null +++ b/src/kernel/limine.h @@ -0,0 +1,579 @@ +/* BSD Zero Clause License */ + +/* Copyright (C) 2022-2023 mintsuki and contributors. + * + * Permission to use, copy, modify, and/or distribute this software for any + * purpose with or without fee is hereby granted. + * + * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES + * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF + * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY + * SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES + * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION + * OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN + * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. + */ + +#ifndef _LIMINE_H +#define _LIMINE_H 1 + +#ifdef __cplusplus +extern "C" { +#endif + +#include + +/* Misc */ + +#ifdef LIMINE_NO_POINTERS +# define LIMINE_PTR(TYPE) uint64_t +#else +# define LIMINE_PTR(TYPE) TYPE +#endif + +#ifdef __GNUC__ +# define LIMINE_DEPRECATED __attribute__((__deprecated__)) +# define LIMINE_DEPRECATED_IGNORE_START \ + _Pragma("GCC diagnostic push") \ + _Pragma("GCC diagnostic ignored \"-Wdeprecated-declarations\"") +# define LIMINE_DEPRECATED_IGNORE_END \ + _Pragma("GCC diagnostic pop") +#else +# define LIMINE_DEPRECATED +# define LIMINE_DEPRECATED_IGNORE_START +# define LIMINE_DEPRECATED_IGNORE_END +#endif + +#define LIMINE_BASE_REVISION(N) \ + uint64_t limine_base_revision[3] = { 0xf9562b2d5c95a6c8, 0x6a7b384944536bdc, (N) }; + +#define LIMINE_BASE_REVISION_SUPPORTED (limine_base_revision[2] == 0) + +#define LIMINE_COMMON_MAGIC 0xc7b1dd30df4c8b88, 0x0a82e883a194f07b + +struct limine_uuid { + uint32_t a; + uint16_t b; + uint16_t c; + uint8_t d[8]; +}; + +#define LIMINE_MEDIA_TYPE_GENERIC 0 +#define LIMINE_MEDIA_TYPE_OPTICAL 1 +#define LIMINE_MEDIA_TYPE_TFTP 2 + +struct limine_file { + uint64_t revision; + LIMINE_PTR(void *) address; + uint64_t size; + LIMINE_PTR(char *) path; + LIMINE_PTR(char *) cmdline; + uint32_t media_type; + uint32_t unused; + uint32_t tftp_ip; + uint32_t tftp_port; + uint32_t partition_index; + uint32_t mbr_disk_id; + struct limine_uuid gpt_disk_uuid; + struct limine_uuid gpt_part_uuid; + struct limine_uuid part_uuid; +}; + +/* Boot info */ + +#define LIMINE_BOOTLOADER_INFO_REQUEST { LIMINE_COMMON_MAGIC, 0xf55038d8e2a1202f, 0x279426fcf5f59740 } + +struct limine_bootloader_info_response { + uint64_t revision; + LIMINE_PTR(char *) name; + LIMINE_PTR(char *) version; +}; + +struct limine_bootloader_info_request { + uint64_t id[4]; + uint64_t revision; + LIMINE_PTR(struct limine_bootloader_info_response *) response; +}; + +/* Stack size */ + +#define LIMINE_STACK_SIZE_REQUEST { LIMINE_COMMON_MAGIC, 0x224ef0460a8e8926, 0xe1cb0fc25f46ea3d } + +struct limine_stack_size_response { + uint64_t revision; +}; + +struct limine_stack_size_request { + uint64_t id[4]; + uint64_t revision; + LIMINE_PTR(struct limine_stack_size_response *) response; + uint64_t stack_size; +}; + +/* HHDM */ + +#define LIMINE_HHDM_REQUEST { LIMINE_COMMON_MAGIC, 0x48dcf1cb8ad2b852, 0x63984e959a98244b } + +struct limine_hhdm_response { + uint64_t revision; + uint64_t offset; +}; + +struct limine_hhdm_request { + uint64_t id[4]; + uint64_t revision; + LIMINE_PTR(struct limine_hhdm_response *) response; +}; + +/* Framebuffer */ + +#define LIMINE_FRAMEBUFFER_REQUEST { LIMINE_COMMON_MAGIC, 0x9d5827dcd881dd75, 0xa3148604f6fab11b } + +#define LIMINE_FRAMEBUFFER_RGB 1 + +struct limine_video_mode { + uint64_t pitch; + uint64_t width; + uint64_t height; + uint16_t bpp; + uint8_t memory_model; + uint8_t red_mask_size; + uint8_t red_mask_shift; + uint8_t green_mask_size; + uint8_t green_mask_shift; + uint8_t blue_mask_size; + uint8_t blue_mask_shift; +}; + +struct limine_framebuffer { + LIMINE_PTR(void *) address; + uint64_t width; + uint64_t height; + uint64_t pitch; + uint16_t bpp; + uint8_t memory_model; + uint8_t red_mask_size; + uint8_t red_mask_shift; + uint8_t green_mask_size; + uint8_t green_mask_shift; + uint8_t blue_mask_size; + uint8_t blue_mask_shift; + uint8_t unused[7]; + uint64_t edid_size; + LIMINE_PTR(void *) edid; + /* Response revision 1 */ + uint64_t mode_count; + LIMINE_PTR(struct limine_video_mode **) modes; +}; + +struct limine_framebuffer_response { + uint64_t revision; + uint64_t framebuffer_count; + LIMINE_PTR(struct limine_framebuffer **) framebuffers; +}; + +struct limine_framebuffer_request { + uint64_t id[4]; + uint64_t revision; + LIMINE_PTR(struct limine_framebuffer_response *) response; +}; + +/* Terminal */ + +#define LIMINE_TERMINAL_REQUEST { LIMINE_COMMON_MAGIC, 0xc8ac59310c2b0844, 0xa68d0c7265d38878 } + +#define LIMINE_TERMINAL_CB_DEC 10 +#define LIMINE_TERMINAL_CB_BELL 20 +#define LIMINE_TERMINAL_CB_PRIVATE_ID 30 +#define LIMINE_TERMINAL_CB_STATUS_REPORT 40 +#define LIMINE_TERMINAL_CB_POS_REPORT 50 +#define LIMINE_TERMINAL_CB_KBD_LEDS 60 +#define LIMINE_TERMINAL_CB_MODE 70 +#define LIMINE_TERMINAL_CB_LINUX 80 + +#define LIMINE_TERMINAL_CTX_SIZE ((uint64_t)(-1)) +#define LIMINE_TERMINAL_CTX_SAVE ((uint64_t)(-2)) +#define LIMINE_TERMINAL_CTX_RESTORE ((uint64_t)(-3)) +#define LIMINE_TERMINAL_FULL_REFRESH ((uint64_t)(-4)) + +/* Response revision 1 */ +#define LIMINE_TERMINAL_OOB_OUTPUT_GET ((uint64_t)(-10)) +#define LIMINE_TERMINAL_OOB_OUTPUT_SET ((uint64_t)(-11)) + +#define LIMINE_TERMINAL_OOB_OUTPUT_OCRNL (1 << 0) +#define LIMINE_TERMINAL_OOB_OUTPUT_OFDEL (1 << 1) +#define LIMINE_TERMINAL_OOB_OUTPUT_OFILL (1 << 2) +#define LIMINE_TERMINAL_OOB_OUTPUT_OLCUC (1 << 3) +#define LIMINE_TERMINAL_OOB_OUTPUT_ONLCR (1 << 4) +#define LIMINE_TERMINAL_OOB_OUTPUT_ONLRET (1 << 5) +#define LIMINE_TERMINAL_OOB_OUTPUT_ONOCR (1 << 6) +#define LIMINE_TERMINAL_OOB_OUTPUT_OPOST (1 << 7) + +LIMINE_DEPRECATED_IGNORE_START + +struct LIMINE_DEPRECATED limine_terminal; + +typedef void (*limine_terminal_write)(struct limine_terminal *, const char *, uint64_t); +typedef void (*limine_terminal_callback)(struct limine_terminal *, uint64_t, uint64_t, uint64_t, uint64_t); + +struct LIMINE_DEPRECATED limine_terminal { + uint64_t columns; + uint64_t rows; + LIMINE_PTR(struct limine_framebuffer *) framebuffer; +}; + +struct LIMINE_DEPRECATED limine_terminal_response { + uint64_t revision; + uint64_t terminal_count; + LIMINE_PTR(struct limine_terminal **) terminals; + LIMINE_PTR(limine_terminal_write) write; +}; + +struct LIMINE_DEPRECATED limine_terminal_request { + uint64_t id[4]; + uint64_t revision; + LIMINE_PTR(struct limine_terminal_response *) response; + LIMINE_PTR(limine_terminal_callback) callback; +}; + +LIMINE_DEPRECATED_IGNORE_END + +/* Paging mode */ + +#define LIMINE_PAGING_MODE_REQUEST { LIMINE_COMMON_MAGIC, 0x95c1a0edab0944cb, 0xa4e5cb3842f7488a } + +#if defined (__x86_64__) || defined (__i386__) +#define LIMINE_PAGING_MODE_X86_64_4LVL 0 +#define LIMINE_PAGING_MODE_X86_64_5LVL 1 +#define LIMINE_PAGING_MODE_MAX LIMINE_PAGING_MODE_X86_64_5LVL +#define LIMINE_PAGING_MODE_DEFAULT LIMINE_PAGING_MODE_X86_64_4LVL +#elif defined (__aarch64__) +#define LIMINE_PAGING_MODE_AARCH64_4LVL 0 +#define LIMINE_PAGING_MODE_AARCH64_5LVL 1 +#define LIMINE_PAGING_MODE_MAX LIMINE_PAGING_MODE_AARCH64_5LVL +#define LIMINE_PAGING_MODE_DEFAULT LIMINE_PAGING_MODE_AARCH64_4LVL +#elif defined (__riscv) && (__riscv_xlen == 64) +#define LIMINE_PAGING_MODE_RISCV_SV39 0 +#define LIMINE_PAGING_MODE_RISCV_SV48 1 +#define LIMINE_PAGING_MODE_RISCV_SV57 2 +#define LIMINE_PAGING_MODE_MAX LIMINE_PAGING_MODE_RISCV_SV57 +#define LIMINE_PAGING_MODE_DEFAULT LIMINE_PAGING_MODE_RISCV_SV48 +#else +#error Unknown architecture +#endif + +struct limine_paging_mode_response { + uint64_t revision; + uint64_t mode; + uint64_t flags; +}; + +struct limine_paging_mode_request { + uint64_t id[4]; + uint64_t revision; + LIMINE_PTR(struct limine_paging_mode_response *) response; + uint64_t mode; + uint64_t flags; +}; + +/* 5-level paging */ + +#define LIMINE_5_LEVEL_PAGING_REQUEST { LIMINE_COMMON_MAGIC, 0x94469551da9b3192, 0xebe5e86db7382888 } + +LIMINE_DEPRECATED_IGNORE_START + +struct LIMINE_DEPRECATED limine_5_level_paging_response { + uint64_t revision; +}; + +struct LIMINE_DEPRECATED limine_5_level_paging_request { + uint64_t id[4]; + uint64_t revision; + LIMINE_PTR(struct limine_5_level_paging_response *) response; +}; + +LIMINE_DEPRECATED_IGNORE_END + +/* SMP */ + +#define LIMINE_SMP_REQUEST { LIMINE_COMMON_MAGIC, 0x95a67b819a1b857e, 0xa0b61b723b6a73e0 } + +struct limine_smp_info; + +typedef void (*limine_goto_address)(struct limine_smp_info *); + +#if defined (__x86_64__) || defined (__i386__) + +#define LIMINE_SMP_X2APIC (1 << 0) + +struct limine_smp_info { + uint32_t processor_id; + uint32_t lapic_id; + uint64_t reserved; + LIMINE_PTR(limine_goto_address) goto_address; + uint64_t extra_argument; +}; + +struct limine_smp_response { + uint64_t revision; + uint32_t flags; + uint32_t bsp_lapic_id; + uint64_t cpu_count; + LIMINE_PTR(struct limine_smp_info **) cpus; +}; + +#elif defined (__aarch64__) + +struct limine_smp_info { + uint32_t processor_id; + uint32_t gic_iface_no; + uint64_t mpidr; + uint64_t reserved; + LIMINE_PTR(limine_goto_address) goto_address; + uint64_t extra_argument; +}; + +struct limine_smp_response { + uint64_t revision; + uint64_t flags; + uint64_t bsp_mpidr; + uint64_t cpu_count; + LIMINE_PTR(struct limine_smp_info **) cpus; +}; + +#elif defined (__riscv) && (__riscv_xlen == 64) + +struct limine_smp_info { + uint64_t processor_id; + uint64_t hartid; + uint64_t reserved; + LIMINE_PTR(limine_goto_address) goto_address; + uint64_t extra_argument; +}; + +struct limine_smp_response { + uint64_t revision; + uint64_t flags; + uint64_t bsp_hartid; + uint64_t cpu_count; + LIMINE_PTR(struct limine_smp_info **) cpus; +}; + +#else +#error Unknown architecture +#endif + +struct limine_smp_request { + uint64_t id[4]; + uint64_t revision; + LIMINE_PTR(struct limine_smp_response *) response; + uint64_t flags; +}; + +/* Memory map */ + +#define LIMINE_MEMMAP_REQUEST { LIMINE_COMMON_MAGIC, 0x67cf3d9d378a806f, 0xe304acdfc50c3c62 } + +#define LIMINE_MEMMAP_USABLE 0 +#define LIMINE_MEMMAP_RESERVED 1 +#define LIMINE_MEMMAP_ACPI_RECLAIMABLE 2 +#define LIMINE_MEMMAP_ACPI_NVS 3 +#define LIMINE_MEMMAP_BAD_MEMORY 4 +#define LIMINE_MEMMAP_BOOTLOADER_RECLAIMABLE 5 +#define LIMINE_MEMMAP_KERNEL_AND_MODULES 6 +#define LIMINE_MEMMAP_FRAMEBUFFER 7 + +struct limine_memmap_entry { + uint64_t base; + uint64_t length; + uint64_t type; +}; + +struct limine_memmap_response { + uint64_t revision; + uint64_t entry_count; + LIMINE_PTR(struct limine_memmap_entry **) entries; +}; + +struct limine_memmap_request { + uint64_t id[4]; + uint64_t revision; + LIMINE_PTR(struct limine_memmap_response *) response; +}; + +/* Entry point */ + +#define LIMINE_ENTRY_POINT_REQUEST { LIMINE_COMMON_MAGIC, 0x13d86c035a1cd3e1, 0x2b0caa89d8f3026a } + +typedef void (*limine_entry_point)(void); + +struct limine_entry_point_response { + uint64_t revision; +}; + +struct limine_entry_point_request { + uint64_t id[4]; + uint64_t revision; + LIMINE_PTR(struct limine_entry_point_response *) response; + LIMINE_PTR(limine_entry_point) entry; +}; + +/* Kernel File */ + +#define LIMINE_KERNEL_FILE_REQUEST { LIMINE_COMMON_MAGIC, 0xad97e90e83f1ed67, 0x31eb5d1c5ff23b69 } + +struct limine_kernel_file_response { + uint64_t revision; + LIMINE_PTR(struct limine_file *) kernel_file; +}; + +struct limine_kernel_file_request { + uint64_t id[4]; + uint64_t revision; + LIMINE_PTR(struct limine_kernel_file_response *) response; +}; + +/* Module */ + +#define LIMINE_MODULE_REQUEST { LIMINE_COMMON_MAGIC, 0x3e7e279702be32af, 0xca1c4f3bd1280cee } + +#define LIMINE_INTERNAL_MODULE_REQUIRED (1 << 0) +#define LIMINE_INTERNAL_MODULE_COMPRESSED (1 << 1) + +struct limine_internal_module { + LIMINE_PTR(const char *) path; + LIMINE_PTR(const char *) cmdline; + uint64_t flags; +}; + +struct limine_module_response { + uint64_t revision; + uint64_t module_count; + LIMINE_PTR(struct limine_file **) modules; +}; + +struct limine_module_request { + uint64_t id[4]; + uint64_t revision; + LIMINE_PTR(struct limine_module_response *) response; + + /* Request revision 1 */ + uint64_t internal_module_count; + LIMINE_PTR(struct limine_internal_module **) internal_modules; +}; + +/* RSDP */ + +#define LIMINE_RSDP_REQUEST { LIMINE_COMMON_MAGIC, 0xc5e77b6b397e7b43, 0x27637845accdcf3c } + +struct limine_rsdp_response { + uint64_t revision; + LIMINE_PTR(void *) address; +}; + +struct limine_rsdp_request { + uint64_t id[4]; + uint64_t revision; + LIMINE_PTR(struct limine_rsdp_response *) response; +}; + +/* SMBIOS */ + +#define LIMINE_SMBIOS_REQUEST { LIMINE_COMMON_MAGIC, 0x9e9046f11e095391, 0xaa4a520fefbde5ee } + +struct limine_smbios_response { + uint64_t revision; + LIMINE_PTR(void *) entry_32; + LIMINE_PTR(void *) entry_64; +}; + +struct limine_smbios_request { + uint64_t id[4]; + uint64_t revision; + LIMINE_PTR(struct limine_smbios_response *) response; +}; + +/* EFI system table */ + +#define LIMINE_EFI_SYSTEM_TABLE_REQUEST { LIMINE_COMMON_MAGIC, 0x5ceba5163eaaf6d6, 0x0a6981610cf65fcc } + +struct limine_efi_system_table_response { + uint64_t revision; + LIMINE_PTR(void *) address; +}; + +struct limine_efi_system_table_request { + uint64_t id[4]; + uint64_t revision; + LIMINE_PTR(struct limine_efi_system_table_response *) response; +}; + +/* EFI memory map */ + +#define LIMINE_EFI_MEMMAP_REQUEST { LIMINE_COMMON_MAGIC, 0x7df62a431d6872d5, 0xa4fcdfb3e57306c8 } + +struct limine_efi_memmap_response { + uint64_t revision; + LIMINE_PTR(void *) memmap; + uint64_t memmap_size; + uint64_t desc_size; + uint64_t desc_version; +}; + +struct limine_efi_memmap_request { + uint64_t id[4]; + uint64_t revision; + LIMINE_PTR(struct limine_efi_memmap_response *) response; +}; + +/* Boot time */ + +#define LIMINE_BOOT_TIME_REQUEST { LIMINE_COMMON_MAGIC, 0x502746e184c088aa, 0xfbc5ec83e6327893 } + +struct limine_boot_time_response { + uint64_t revision; + int64_t boot_time; +}; + +struct limine_boot_time_request { + uint64_t id[4]; + uint64_t revision; + LIMINE_PTR(struct limine_boot_time_response *) response; +}; + +/* Kernel address */ + +#define LIMINE_KERNEL_ADDRESS_REQUEST { LIMINE_COMMON_MAGIC, 0x71ba76863cc55f63, 0xb2644a48c516a487 } + +struct limine_kernel_address_response { + uint64_t revision; + uint64_t physical_base; + uint64_t virtual_base; +}; + +struct limine_kernel_address_request { + uint64_t id[4]; + uint64_t revision; + LIMINE_PTR(struct limine_kernel_address_response *) response; +}; + +/* Device Tree Blob */ + +#define LIMINE_DTB_REQUEST { LIMINE_COMMON_MAGIC, 0xb40ddb48fb54bac7, 0x545081493f81ffb7 } + +struct limine_dtb_response { + uint64_t revision; + LIMINE_PTR(void *) dtb_ptr; +}; + +struct limine_dtb_request { + uint64_t id[4]; + uint64_t revision; + LIMINE_PTR(struct limine_dtb_response *) response; +}; + +#ifdef __cplusplus +} +#endif + +#endif diff --git a/src/kernel/main.c b/src/kernel/main.c new file mode 100644 index 0000000..004a686 --- /dev/null +++ b/src/kernel/main.c @@ -0,0 +1,79 @@ +#include +#include +#include +#include "limine.h" +#include "graphics.h" +#include "idt.h" +#include "ps2.h" +#include "wm.h" +#include "io.h" +#include "memory_manager.h" + +// --- Limine Requests --- +__attribute__((used, section(".requests"))) +static volatile LIMINE_BASE_REVISION(2); + +__attribute__((used, section(".requests"))) +static volatile struct limine_framebuffer_request framebuffer_request = { + .id = LIMINE_FRAMEBUFFER_REQUEST, + .revision = 1 +}; + +__attribute__((used, section(".requests_start"))) +static volatile struct limine_request *const requests_start_marker[] = { + (struct limine_request *)&framebuffer_request, + NULL +}; + +__attribute__((used, section(".requests_end"))) +static volatile struct limine_request *const requests_end_marker[] = { + NULL +}; + +static void hcf(void) { + asm("cli"); + for (;;) { + asm("hlt"); + } +} + +// Kernel Entry Point +void kmain(void) { + // 1. Graphics Init + if (LIMINE_BASE_REVISION_SUPPORTED == false) { + // Warning + } + + if (framebuffer_request.response == NULL || framebuffer_request.response->framebuffer_count < 1) { + hcf(); + } + + struct limine_framebuffer *fb = framebuffer_request.response->framebuffers[0]; + graphics_init(fb); + + // 2. Interrupts Init + idt_init(); + + // Register ISRs with correct CS + idt_register_interrupts(); + + // Load IDT and Enable Interrupts + idt_load(); + + // 2.5 Memory Manager Init + memory_manager_init(); + + // 3. PS/2 Init (Mouse/Keyboard) + asm("cli"); + ps2_init(); + asm("sti"); + + // 4. Window Manager Init (Draws initial desktop) + wm_init(); + + // 5. Main loop - just wait for interrupts + // Timer interrupt will drive the redraw system + while (1) { + asm("hlt"); + } +} \ No newline at end of file diff --git a/src/kernel/memory_manager.c b/src/kernel/memory_manager.c new file mode 100644 index 0000000..7bbc8b4 --- /dev/null +++ b/src/kernel/memory_manager.c @@ -0,0 +1,434 @@ +#include "memory_manager.h" +#include "io.h" +#include + +// --- Internal State --- +static uint8_t memory_pool[MEMORY_POOL_SIZE] __attribute__((aligned(4096))); +static MemBlock block_list[MAX_ALLOCATIONS]; +static int block_count = 0; +static size_t total_allocated = 0; +static size_t peak_allocated = 0; +static uint32_t allocation_counter = 0; +static bool initialized = false; + +// --- Helper Functions --- + +// Simple memset for internal use +static void mem_memset(void *dest, int val, size_t len) { + uint8_t *ptr = (uint8_t *)dest; + while (len-- > 0) { + *ptr++ = (uint8_t)val; + } +} + +// Simple memmove +static void mem_memmove(void *dest, const void *src, size_t len) { + uint8_t *d = (uint8_t *)dest; + const uint8_t *s = (const uint8_t *)src; + + if (d < s) { + while (len--) *d++ = *s++; + } else { + d += len; + s += len; + while (len--) *(--d) = *(--s); + } +} + +// Get current time in ticks (simple counter) +static uint32_t get_timestamp(void) { + static uint32_t tick = 0; + return tick++; +} + +// Find free space in memory pool +static void* find_free_space(size_t size) { + size_t offset = 0; + + while (offset + size <= MEMORY_POOL_SIZE) { + bool space_free = true; + + // Check if this range is free + for (int i = 0; i < block_count; i++) { + if (!block_list[i].allocated) continue; + + void *block_start = block_list[i].address; + void *block_end = (uint8_t *)block_start + block_list[i].size; + void *check_start = (uint8_t *)memory_pool + offset; + void *check_end = (uint8_t *)check_start + size; + + // Check for overlap + if (check_start < block_end && check_end > block_start) { + space_free = false; + + // Skip past this block + size_t block_offset = (uintptr_t)block_end - (uintptr_t)memory_pool; + if (block_offset > offset) { + offset = block_offset; + } + break; + } + } + + if (space_free) { + return (uint8_t *)memory_pool + offset; + } + } + + return NULL; +} + +// Calculate fragmentation +static size_t calculate_fragmentation(void) { + if (total_allocated == 0) return 0; + + // Sort blocks by address + for (int i = 0; i < block_count - 1; i++) { + for (int j = i + 1; j < block_count; j++) { + if ((uintptr_t)block_list[i].address > (uintptr_t)block_list[j].address) { + MemBlock tmp = block_list[i]; + block_list[i] = block_list[j]; + block_list[j] = tmp; + } + } + } + + // Count gaps between allocated blocks + size_t total_gaps = 0; + void *pool_end = (uint8_t *)memory_pool + MEMORY_POOL_SIZE; + + void *current_end = memory_pool; + + for (int i = 0; i < block_count; i++) { + if (!block_list[i].allocated) continue; + + if (block_list[i].address > current_end) { + total_gaps += (uintptr_t)block_list[i].address - (uintptr_t)current_end; + } + + current_end = (uint8_t *)block_list[i].address + block_list[i].size; + } + + if (total_allocated == 0) return 0; + return (total_gaps * 100) / total_allocated; +} + +// --- Public API --- + +void memory_manager_init(void) { + if (initialized) return; + + // Clear metadata + mem_memset(block_list, 0, sizeof(block_list)); + block_count = 0; + total_allocated = 0; + peak_allocated = 0; + allocation_counter = 0; + + // Create initial free block representing entire pool + block_list[0].address = memory_pool; + block_list[0].size = MEMORY_POOL_SIZE; + block_list[0].allocated = false; + block_list[0].allocation_id = 0; + block_count = 1; + + initialized = true; +} + +void* kmalloc(size_t size) { + if (!initialized) { + memory_manager_init(); + } + + if (size == 0 || size > MEMORY_POOL_SIZE) { + return NULL; + } + + // Check if we can allocate + if (total_allocated + size > MEMORY_POOL_SIZE) { + return NULL; + } + + // Find free space + void *ptr = find_free_space(size); + if (ptr == NULL) { + return NULL; + } + + // Add block entry + if (block_count >= MAX_ALLOCATIONS) { + return NULL; + } + + allocation_counter++; + int idx = block_count++; + + block_list[idx].address = ptr; + block_list[idx].size = size; + block_list[idx].allocated = true; + block_list[idx].allocation_id = allocation_counter; + block_list[idx].timestamp = get_timestamp(); + + total_allocated += size; + if (total_allocated > peak_allocated) { + peak_allocated = total_allocated; + } + + // Clear memory + mem_memset(ptr, 0, size); + + return ptr; +} + +void kfree(void *ptr) { + if (ptr == NULL || !initialized) { + return; + } + + // Find and free the block + for (int i = 0; i < block_count; i++) { + if (block_list[i].allocated && block_list[i].address == ptr) { + total_allocated -= block_list[i].size; + block_list[i].allocated = false; + + // Compact: remove freed entry and shift remaining + for (int j = i; j < block_count - 1; j++) { + block_list[j] = block_list[j + 1]; + } + block_count--; + return; + } + } +} + +void* krealloc(void *ptr, size_t new_size) { + if (!initialized) { + memory_manager_init(); + } + + if (new_size == 0) { + kfree(ptr); + return NULL; + } + + if (ptr == NULL) { + return kmalloc(new_size); + } + + // Find the block + for (int i = 0; i < block_count; i++) { + if (block_list[i].allocated && block_list[i].address == ptr) { + if (block_list[i].size >= new_size) { + // Allocation is large enough + return ptr; + } + + // Need to allocate new space + void *new_ptr = kmalloc(new_size); + if (new_ptr == NULL) { + return NULL; + } + + // Copy data + mem_memmove(new_ptr, ptr, block_list[i].size); + + // Free old pointer + kfree(ptr); + + return new_ptr; + } + } + + return NULL; +} + +MemStats memory_get_stats(void) { + MemStats stats; + + stats.total_memory = MEMORY_POOL_SIZE; + stats.used_memory = total_allocated; + stats.available_memory = MEMORY_POOL_SIZE - total_allocated; + stats.allocated_blocks = 0; + stats.free_blocks = 0; + stats.largest_free_block = 0; + stats.smallest_free_block = MEMORY_POOL_SIZE; + stats.peak_memory_used = peak_allocated; + + // Count and analyze blocks + for (int i = 0; i < block_count; i++) { + if (block_list[i].allocated) { + stats.allocated_blocks++; + } else { + stats.free_blocks++; + if (block_list[i].size > stats.largest_free_block) { + stats.largest_free_block = block_list[i].size; + } + if (block_list[i].size < stats.smallest_free_block) { + stats.smallest_free_block = block_list[i].size; + } + } + } + + if (stats.free_blocks == 0) { + stats.smallest_free_block = 0; + } + + stats.fragmentation_percent = calculate_fragmentation(); + + return stats; +} + +void memory_print_stats(void) { + MemStats stats = memory_get_stats(); + + // We need to use the CLI write functions - declare them as extern + extern void cmd_write(const char *str); + extern void cmd_write_int(int n); + extern void cmd_putchar(char c); + + cmd_write("\n=== MEMORY STATISTICS ===\n"); + cmd_write("Total Memory: "); + cmd_write_int(stats.total_memory / 1024); + cmd_write(" KB\n"); + + cmd_write("Used Memory: "); + cmd_write_int(stats.used_memory / 1024); + cmd_write(" KB\n"); + + cmd_write("Available Memory: "); + cmd_write_int(stats.available_memory / 1024); + cmd_write(" KB\n"); + + cmd_write("Allocated Blocks: "); + cmd_write_int(stats.allocated_blocks); + cmd_write("\n"); + + cmd_write("Free Blocks: "); + cmd_write_int(stats.free_blocks); + cmd_write("\n"); + + cmd_write("Largest Free: "); + cmd_write_int(stats.largest_free_block / 1024); + cmd_write(" KB\n"); + + cmd_write("Peak Usage: "); + cmd_write_int(stats.peak_memory_used / 1024); + cmd_write(" KB\n"); + + cmd_write("Fragmentation: "); + cmd_write_int(stats.fragmentation_percent); + cmd_write("%\n"); + + cmd_write("Usage: "); + int usage_percent = (stats.used_memory * 100) / stats.total_memory; + cmd_write_int(usage_percent); + cmd_write("%\n"); + + cmd_write("========================\n\n"); +} + +void memory_print_detailed(void) { + extern void cmd_write(const char *str); + extern void cmd_write_int(int n); + extern void cmd_putchar(char c); + + cmd_write("\n=== DETAILED MEMORY BLOCKS ===\n"); + cmd_write("ID Address Size Status\n"); + cmd_write("------ -------- -------- --------\n"); + + for (int i = 0; i < block_count; i++) { + if (block_list[i].size == 0) continue; + + // ID + cmd_write_int(block_list[i].allocation_id); + cmd_write(" "); + + // Address (simplified hex output) + cmd_write("0x"); + cmd_write_int((uintptr_t)block_list[i].address / 1024); + cmd_write(" "); + + // Size + cmd_write_int(block_list[i].size / 1024); + cmd_write("KB "); + + // Status + if (block_list[i].allocated) { + cmd_write("ALLOC\n"); + } else { + cmd_write("FREE\n"); + } + } + + cmd_write("==============================\n\n"); +} + +void memory_validate(void) { + extern void cmd_write(const char *str); + extern void cmd_write_int(int n); + + int errors = 0; + + // Check for overlapping blocks + for (int i = 0; i < block_count; i++) { + for (int j = i + 1; j < block_count; j++) { + void *i_start = block_list[i].address; + void *i_end = (uint8_t *)i_start + block_list[i].size; + void *j_start = block_list[j].address; + void *j_end = (uint8_t *)j_start + block_list[j].size; + + if (i_start < j_end && i_end > j_start) { + errors++; + cmd_write("ERROR: Overlapping blocks detected!\n"); + } + } + } + + if (errors == 0) { + cmd_write("Memory validation: OK\n"); + } else { + cmd_write("Memory validation failed with "); + cmd_write_int(errors); + cmd_write(" errors\n"); + } +} + +void memory_dump_blocks(void) { + extern void cmd_write(const char *str); + extern void cmd_write_int(int n); + + cmd_write("\nMemory block dump:\n"); + cmd_write("Total blocks: "); + cmd_write_int(block_count); + cmd_write("\n"); + + memory_print_detailed(); +} + +size_t memory_get_peak_usage(void) { + return peak_allocated; +} + +void memory_reset_peak(void) { + peak_allocated = total_allocated; +} + +bool memory_is_valid_ptr(void *ptr) { + if (ptr == NULL) return false; + + void *pool_start = memory_pool; + void *pool_end = (uint8_t *)memory_pool + MEMORY_POOL_SIZE; + + if (ptr < pool_start || ptr >= pool_end) { + return false; + } + + // Check if it's an allocated block + for (int i = 0; i < block_count; i++) { + if (block_list[i].allocated && block_list[i].address == ptr) { + return true; + } + } + + return false; +} diff --git a/src/kernel/memory_manager.h b/src/kernel/memory_manager.h new file mode 100644 index 0000000..c32379b --- /dev/null +++ b/src/kernel/memory_manager.h @@ -0,0 +1,56 @@ +#ifndef MEMORY_MANAGER_H +#define MEMORY_MANAGER_H + +#include +#include +#include + +// Memory Manager Configuration +#define MEMORY_POOL_SIZE (4 * 1024 * 1024) // 4MB pool +#define MAX_ALLOCATIONS 1024 +#define MAX_FRAGMENTATION_SLOTS 2048 + +// Allocation block metadata +typedef struct { + void *address; + size_t size; + bool allocated; + uint32_t allocation_id; + uint32_t timestamp; +} MemBlock; + +// Memory statistics +typedef struct { + size_t total_memory; + size_t used_memory; + size_t available_memory; + size_t allocated_blocks; + size_t free_blocks; + size_t largest_free_block; + size_t smallest_free_block; + size_t fragmentation_percent; + size_t peak_memory_used; +} MemStats; + +// Public API +void memory_manager_init(void); + +// Allocation/Deallocation +void* kmalloc(size_t size); +void kfree(void *ptr); +void* krealloc(void *ptr, size_t new_size); + +// Statistics and Information +MemStats memory_get_stats(void); +void memory_print_stats(void); +void memory_print_detailed(void); + +void memory_validate(void); +void memory_dump_blocks(void); + +// Internal utilities +size_t memory_get_peak_usage(void); +void memory_reset_peak(void); +bool memory_is_valid_ptr(void *ptr); + +#endif // MEMORY_MANAGER_H diff --git a/src/kernel/minesweeper.c b/src/kernel/minesweeper.c new file mode 100644 index 0000000..78f9ff3 --- /dev/null +++ b/src/kernel/minesweeper.c @@ -0,0 +1,248 @@ +#include "minesweeper.h" +#include "graphics.h" +#include "wm.h" +#include +#include + +Window win_minesweeper; + +// Game constants +#define GRID_WIDTH 10 +#define GRID_HEIGHT 10 +#define MINE_COUNT 10 +#define CELL_SIZE 20 + +// Game state +static int grid[GRID_HEIGHT][GRID_WIDTH]; // -1 = mine, 0-8 = adjacent mine count +static bool revealed[GRID_HEIGHT][GRID_WIDTH]; +static bool flagged[GRID_HEIGHT][GRID_WIDTH]; +static bool game_over = false; +static bool game_won = false; +static int revealed_count = 0; + +// Helper: Random number generator (simple LCG) +static uint32_t random_seed = 12345; +static uint32_t random_next(void) { + random_seed = random_seed * 1103515245 + 12345; + return (random_seed / 65536) % 32768; +} + +static void init_game(void) { + // Clear arrays + for (int y = 0; y < GRID_HEIGHT; y++) { + for (int x = 0; x < GRID_WIDTH; x++) { + grid[y][x] = 0; + revealed[y][x] = false; + flagged[y][x] = false; + } + } + + // Place mines randomly + int mines_placed = 0; + while (mines_placed < MINE_COUNT) { + int x = random_next() % GRID_WIDTH; + int y = random_next() % GRID_HEIGHT; + + if (grid[y][x] != -1) { + grid[y][x] = -1; + mines_placed++; + } + } + + // Calculate adjacent mine counts + for (int y = 0; y < GRID_HEIGHT; y++) { + for (int x = 0; x < GRID_WIDTH; x++) { + if (grid[y][x] != -1) { + int count = 0; + for (int dy = -1; dy <= 1; dy++) { + for (int dx = -1; dx <= 1; dx++) { + int ny = y + dy; + int nx = x + dx; + if (ny >= 0 && ny < GRID_HEIGHT && nx >= 0 && nx < GRID_WIDTH) { + if (grid[ny][nx] == -1) count++; + } + } + } + grid[y][x] = count; + } + } + } + + game_over = false; + game_won = false; + revealed_count = 0; +} + +static void reveal_cell(int x, int y); + +// Flood fill for empty cells +static void flood_fill(int x, int y) { + if (x < 0 || x >= GRID_WIDTH || y < 0 || y >= GRID_HEIGHT) return; + if (revealed[y][x] || flagged[y][x]) return; + if (grid[y][x] == -1) return; + + revealed[y][x] = true; + revealed_count++; + + // If cell is empty, reveal adjacent cells + if (grid[y][x] == 0) { + for (int dy = -1; dy <= 1; dy++) { + for (int dx = -1; dx <= 1; dx++) { + flood_fill(x + dx, y + dy); + } + } + } +} + +static void reveal_cell(int x, int y) { + if (x < 0 || x >= GRID_WIDTH || y < 0 || y >= GRID_HEIGHT) return; + if (revealed[y][x] || flagged[y][x]) return; + + if (grid[y][x] == -1) { + // Hit a mine - game over + game_over = true; + // Reveal all mines + for (int yy = 0; yy < GRID_HEIGHT; yy++) { + for (int xx = 0; xx < GRID_WIDTH; xx++) { + if (grid[yy][xx] == -1) { + revealed[yy][xx] = true; + } + } + } + } else if (grid[y][x] == 0) { + // Empty cell - flood fill + flood_fill(x, y); + } else { + // Numbered cell + revealed[y][x] = true; + revealed_count++; + } + + // Check win condition + if (revealed_count == (GRID_WIDTH * GRID_HEIGHT - MINE_COUNT)) { + game_won = true; + } +} + +static void flag_cell(int x, int y) { + if (x < 0 || x >= GRID_WIDTH || y < 0 || y >= GRID_HEIGHT) return; + if (revealed[y][x]) return; + + flagged[y][x] = !flagged[y][x]; +} + +static void minesweeper_right_click(Window *win, int x, int y) { + // x and y are relative to window content (0,0 is top-left of window) + int grid_start_x = 10; + int grid_start_y = 50; + + // Check grid cells + if (x >= grid_start_x && x < grid_start_x + GRID_WIDTH * CELL_SIZE && + y >= grid_start_y && y < grid_start_y + GRID_HEIGHT * CELL_SIZE) { + + if (game_over || game_won) return; + + int cell_x = (x - grid_start_x) / CELL_SIZE; + int cell_y = (y - grid_start_y) / CELL_SIZE; + + flag_cell(cell_x, cell_y); + wm_paint(); + } +} + +static void minesweeper_paint(Window *win) { + // Background + draw_rect(win->x + 4, win->y + 24, win->w - 8, win->h - 28, COLOR_LTGRAY); + + // Game status + if (game_over) { + draw_string(win->x + 10, win->y + 30, "Game Over!", COLOR_RED); + } else if (game_won) { + draw_string(win->x + 10, win->y + 30, "You Won!", COLOR_BLUE); + } else { + draw_string(win->x + 10, win->y + 30, "Minesweeper", COLOR_BLACK); + } + + // Draw grid + int grid_start_x = win->x + 10; + int grid_start_y = win->y + 50; + + for (int y = 0; y < GRID_HEIGHT; y++) { + for (int x = 0; x < GRID_WIDTH; x++) { + int px = grid_start_x + x * CELL_SIZE; + int py = grid_start_y + y * CELL_SIZE; + + if (revealed[y][x]) { + // Revealed cell - sunken + draw_bevel_rect(px, py, CELL_SIZE, CELL_SIZE, true); + + if (grid[y][x] == -1) { + // Mine + draw_string(px + 8, py + 6, "*", COLOR_RED); + } else if (grid[y][x] > 0) { + // Number + char num[2] = { '0' + grid[y][x], 0 }; + draw_string(px + 8, py + 6, num, COLOR_BLACK); + } + // 0 = empty, nothing to draw + } else { + // Unrevealed cell - raised + draw_bevel_rect(px, py, CELL_SIZE, CELL_SIZE, false); + + if (flagged[y][x]) { + draw_string(px + 7, py + 6, "F", COLOR_RED); + } + } + } + } + + // Draw new game button + int btn_y = grid_start_y + GRID_HEIGHT * CELL_SIZE + 10; + draw_button(grid_start_x, btn_y, 90, 24, "New Game", false); +} + +static void minesweeper_click(Window *win, int x, int y) { + // x and y are relative to window content (0,0 is top-left of window) + int grid_start_x = 10; + int grid_start_y = 50; + int btn_y = grid_start_y + GRID_HEIGHT * CELL_SIZE + 10; + + // Check "New Game" button + if (x >= grid_start_x && x < grid_start_x + 90 && + y >= btn_y && y < btn_y + 24) { + init_game(); + wm_paint(); + return; + } + + // Check grid cells + if (x >= grid_start_x && x < grid_start_x + GRID_WIDTH * CELL_SIZE && + y >= grid_start_y && y < grid_start_y + GRID_HEIGHT * CELL_SIZE) { + + if (game_over || game_won) return; + + int cell_x = (x - grid_start_x) / CELL_SIZE; + int cell_y = (y - grid_start_y) / CELL_SIZE; + + reveal_cell(cell_x, cell_y); + + wm_paint(); + } +} + +void minesweeper_init(void) { + win_minesweeper.title = "Minesweeper"; + win_minesweeper.x = 250; + win_minesweeper.y = 100; + win_minesweeper.w = 240; + win_minesweeper.h = 340; + win_minesweeper.visible = false; + win_minesweeper.focused = false; + win_minesweeper.z_index = 0; + win_minesweeper.paint = minesweeper_paint; + win_minesweeper.handle_click = minesweeper_click; + win_minesweeper.handle_right_click = minesweeper_right_click; + + // Initialize game + init_game(); +} diff --git a/src/kernel/minesweeper.h b/src/kernel/minesweeper.h new file mode 100644 index 0000000..120c35f --- /dev/null +++ b/src/kernel/minesweeper.h @@ -0,0 +1,10 @@ +#ifndef MINESWEEPER_H +#define MINESWEEPER_H + +#include "wm.h" + +extern Window win_minesweeper; + +void minesweeper_init(void); + +#endif diff --git a/src/kernel/notepad.c b/src/kernel/notepad.c new file mode 100644 index 0000000..f0e5ccd --- /dev/null +++ b/src/kernel/notepad.c @@ -0,0 +1,212 @@ +#include "notepad.h" +#include "graphics.h" +#include + +Window win_notepad; +static int notepad_scroll_line = 0; // Track which line is at top of visible area + +static void notepad_ensure_cursor_visible(Window *win) { + // Calculate how many lines fit in the notepad window (allow 3-4 extra lines) + int visible_lines = (win->h - 40) / 10 + 3; // Content area height / line height + buffer + if (visible_lines < 1) visible_lines = 1; + + // Count which line the cursor is on + int cursor_line = 0; + for (int i = 0; i < win->cursor_pos && i < win->buf_len; i++) { + if (win->buffer[i] == '\n') cursor_line++; + } + + // Scroll up if cursor is above visible area + if (cursor_line < notepad_scroll_line) { + notepad_scroll_line = cursor_line; + } + + // Scroll down if cursor is below visible area + if (cursor_line >= notepad_scroll_line + visible_lines) { + notepad_scroll_line = cursor_line - visible_lines + 1; + } +} + +static void notepad_paint(Window *win) { + // Draw text starting from scroll position + int display_line = 0; + int current_line = 0; + int current_x = win->x + 8; + int current_y = win->y + 30; + + for (int i = 0; i < win->buf_len; i++) { + // Skip lines before scroll position + if (current_line < notepad_scroll_line) { + if (win->buffer[i] == '\n') { + current_line++; + } + continue; + } + + // Stop if we've gone past visible area + if (display_line >= (win->h - 40) / 10) { + break; + } + + if (win->buffer[i] == '\n') { + current_x = win->x + 8; + current_y += 10; + current_line++; + display_line++; + } else { + // Draw single character + char ch[2] = {win->buffer[i], 0}; + draw_string(current_x, current_y, ch, COLOR_BLACK); + current_x += 8; + } + } + + // Cursor + if (win->focused) { + int cx = win->x + 8; + int cy = win->y + 30; + int current_line = 0; + + for (int i = 0; i < win->cursor_pos; i++) { + if (win->buffer[i] == '\n') { + cx = win->x + 8; + cy += 10; + current_line++; + } else { + cx += 8; + } + } + + // Only draw cursor if it's in visible area + if (current_line >= notepad_scroll_line && + current_line < notepad_scroll_line + (win->h - 40) / 10) { + draw_rect(cx, cy, 2, 8, COLOR_BLACK); + } + } +} + +static void notepad_key(Window *target, char c) { + if (c == 17) { // UP + if (target->cursor_pos > 0) { + int curr = target->cursor_pos; + int line_start = curr; + while (line_start > 0 && target->buffer[line_start - 1] != '\n') { + line_start--; + } + int col = curr - line_start; + + if (line_start > 0) { + int prev_line_end = line_start - 1; + int prev_line_start = prev_line_end; + while (prev_line_start > 0 && target->buffer[prev_line_start - 1] != '\n') { + prev_line_start--; + } + int prev_line_len = prev_line_end - prev_line_start; + if (col > prev_line_len) col = prev_line_len; + target->cursor_pos = prev_line_start + col; + } + } + notepad_ensure_cursor_visible(target); + } else if (c == 18) { // DOWN + int len = target->buf_len; + if (target->cursor_pos < len) { + int curr = target->cursor_pos; + int line_start = curr; + while (line_start > 0 && target->buffer[line_start - 1] != '\n') { + line_start--; + } + int col = curr - line_start; + + int next_line_start = curr; + while (next_line_start < len && target->buffer[next_line_start] != '\n') { + next_line_start++; + } + + if (next_line_start < len) { + next_line_start++; // Skip newline + int next_line_end = next_line_start; + while (next_line_end < len && target->buffer[next_line_end] != '\n') { + next_line_end++; + } + int next_line_len = next_line_end - next_line_start; + if (col > next_line_len) col = next_line_len; + target->cursor_pos = next_line_start + col; + } else { + target->cursor_pos = len; + } + } + notepad_ensure_cursor_visible(target); + } else if (c == 19) { // LEFT + if (target->cursor_pos > 0) target->cursor_pos--; + notepad_ensure_cursor_visible(target); + } else if (c == 20) { // RIGHT + if (target->cursor_pos < target->buf_len) target->cursor_pos++; + notepad_ensure_cursor_visible(target); + } else if (c == '\b') { // Backspace + if (target->cursor_pos > 0) { + // Shift left + for (int i = target->cursor_pos; i < target->buf_len; i++) { + target->buffer[i - 1] = target->buffer[i]; + } + target->buf_len--; + target->cursor_pos--; + target->buffer[target->buf_len] = 0; + notepad_ensure_cursor_visible(target); + } + } else if (c == '\n') { // Enter + if (target->buf_len < 1023) { + // Shift right + for (int i = target->buf_len; i > target->cursor_pos; i--) { + target->buffer[i] = target->buffer[i - 1]; + } + target->buffer[target->cursor_pos] = c; + target->buf_len++; + target->cursor_pos++; + target->buffer[target->buf_len] = 0; + notepad_ensure_cursor_visible(target); + } + } else { + // Printable char + if (target->buf_len < 1023) { + for (int i = target->buf_len; i > target->cursor_pos; i--) { + target->buffer[i] = target->buffer[i - 1]; + } + target->buffer[target->cursor_pos] = c; + target->buf_len++; + target->cursor_pos++; + target->buffer[target->buf_len] = 0; + notepad_ensure_cursor_visible(target); + } + } +} + +void notepad_init(void) { + win_notepad.title = "Notepad"; + win_notepad.x = 100; + win_notepad.y = 100; + win_notepad.w = 400; + win_notepad.h = 300; + win_notepad.visible = false; + win_notepad.buf_len = 0; + win_notepad.cursor_pos = 0; + win_notepad.focused = false; + win_notepad.z_index = 0; + win_notepad.paint = notepad_paint; + win_notepad.handle_key = notepad_key; + win_notepad.handle_click = NULL; + win_notepad.handle_right_click = NULL; + + notepad_scroll_line = 0; + + for(int i=0; i<1024; i++) win_notepad.buffer[i] = 0; +} + +void notepad_reset(void) { + // Clear notepad buffer and reset cursor on close/reopen + win_notepad.buf_len = 0; + win_notepad.cursor_pos = 0; + win_notepad.focused = false; + notepad_scroll_line = 0; + + for(int i=0; i<1024; i++) win_notepad.buffer[i] = 0; +} \ No newline at end of file diff --git a/src/kernel/notepad.h b/src/kernel/notepad.h new file mode 100644 index 0000000..f530c4d --- /dev/null +++ b/src/kernel/notepad.h @@ -0,0 +1,11 @@ +#ifndef NOTEPAD_H +#define NOTEPAD_H + +#include "wm.h" + +extern Window win_notepad; + +void notepad_init(void); +void notepad_reset(void); + +#endif \ No newline at end of file diff --git a/src/kernel/ps2.c b/src/kernel/ps2.c new file mode 100644 index 0000000..cbf47cd --- /dev/null +++ b/src/kernel/ps2.c @@ -0,0 +1,167 @@ +#include "ps2.h" +#include "io.h" +#include "wm.h" +#include + +extern void serial_print(const char *s); +extern void serial_print_hex(uint64_t n); + +// --- Timer Handler --- +void timer_handler(void) { + wm_timer_tick(); + outb(0x20, 0x20); // EOI to Master PIC +} + +// --- Keyboard --- +static bool shift_pressed = false; +static bool extended_scancode = false; + +// Simple US QWERTY Scan Code Set 1 Map +static char scancode_map[128] = { + 0, 27, '1', '2', '3', '4', '5', '6', '7', '8', '9', '0', '-', '=', '\b', + '\t', 'q', 'w', 'e', 'r', 't', 'y', 'u', 'i', 'o', 'p', '[', ']', '\n', + 0, 'a', 's', 'd', 'f', 'g', 'h', 'j', 'k', 'l', ';', '\'', '`', 0, + '\\', 'z', 'x', 'c', 'v', 'b', 'n', 'm', ',', '.', '/', 0, '*', + 0, ' ', 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, +}; + +static char scancode_map_shift[128] = { + 0, 27, '!', '@', '#', '$', '%', '^', '&', '*', '(', ')', '_', '+', '\b', + '\t', 'Q', 'W', 'E', 'R', 'T', 'Y', 'U', 'I', 'O', 'P', '{', '}', '\n', + 0, 'A', 'S', 'D', 'F', 'G', 'H', 'J', 'K', 'L', ':', '"', '~', 0, + '|', 'Z', 'X', 'C', 'V', 'B', 'N', 'M', '<', '>', '?', 0, '*', + 0, ' ', 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, +}; + +void keyboard_handler(void) { + uint8_t scancode = inb(0x60); + + if (scancode == 0xE0) { + extended_scancode = true; + outb(0x20, 0x20); + return; + } + + if (scancode == 0x2A || scancode == 0x36) { // Shift Down + shift_pressed = true; + } else if (scancode == 0xAA || scancode == 0xB6) { // Shift Up + shift_pressed = false; + } else if (!(scancode & 0x80)) { // Key Press (not release) + if (extended_scancode) { + // Extended scancode - arrow keys and special keys + extended_scancode = false; + switch (scancode) { + case 0x48: wm_handle_key(17); break; // Up arrow + case 0x50: wm_handle_key(18); break; // Down arrow + case 0x4B: wm_handle_key(19); break; // Left arrow + case 0x4D: wm_handle_key(20); break; // Right arrow + } + } else { + // Regular scancode + char c = shift_pressed ? scancode_map_shift[scancode] : scancode_map[scancode]; + if (c) { + wm_handle_key(c); + } + } + } else if (scancode & 0x80) { + // Key release + extended_scancode = false; + } + + outb(0x20, 0x20); // EOI +} + +// --- Mouse --- +static uint8_t mouse_cycle = 0; +static int8_t mouse_byte[3]; + +void mouse_wait(uint8_t type) { + uint32_t timeout = 100000; + if (type == 0) { // Write + while (timeout--) { + if ((inb(0x64) & 2) == 0) return; + } + } else { // Read + while (timeout--) { + if ((inb(0x64) & 1) == 1) return; + } + } +} + +void mouse_write(uint8_t write) { + mouse_wait(0); + outb(0x64, 0xD4); + mouse_wait(0); + outb(0x60, write); +} + +uint8_t mouse_read(void) { + mouse_wait(1); + return inb(0x60); +} + +void mouse_init(void) { + uint8_t status; + + // Enable Aux Device + mouse_wait(0); + outb(0x64, 0xA8); + + // Enable Interrupts + mouse_wait(0); + outb(0x64, 0x20); + mouse_wait(1); + status = inb(0x60) | 2; + mouse_wait(0); + outb(0x64, 0x60); + mouse_wait(0); + outb(0x60, status); + + // Set Defaults + mouse_write(0xF6); + mouse_read(); + + // Enable Streaming + mouse_write(0xF4); + mouse_read(); +} + +void mouse_handler(void) { + uint8_t status = inb(0x64); + if (!(status & 0x20)) { + outb(0x20, 0x20); + outb(0xA0, 0x20); + return; + } + + uint8_t b = inb(0x60); + + if (mouse_cycle == 0) { + if ((b & 0x08) == 0) { // Sync check + // Skip + } else { + mouse_byte[0] = b; + mouse_cycle++; + } + } else if (mouse_cycle == 1) { + mouse_byte[1] = b; + mouse_cycle++; + } else { + mouse_byte[2] = b; + mouse_cycle = 0; + + // Packet Full + int8_t dx = mouse_byte[1]; + int8_t dy = mouse_byte[2]; + + // Send to WM + wm_handle_mouse(dx, -dy, mouse_byte[0] & 0x07); + } + + outb(0x20, 0x20); + outb(0xA0, 0x20); // Slave EOI +} + +void ps2_init(void) { + mouse_init(); +} \ No newline at end of file diff --git a/src/kernel/ps2.h b/src/kernel/ps2.h new file mode 100644 index 0000000..ca00d4f --- /dev/null +++ b/src/kernel/ps2.h @@ -0,0 +1,8 @@ +#ifndef PS2_H +#define PS2_H + +void ps2_init(void); +void keyboard_handler(void); +void mouse_handler(void); + +#endif diff --git a/src/kernel/rtc.c b/src/kernel/rtc.c new file mode 100644 index 0000000..ec99793 --- /dev/null +++ b/src/kernel/rtc.c @@ -0,0 +1,76 @@ +#include "rtc.h" +#include "io.h" + +#define CMOS_ADDRESS 0x70 +#define CMOS_DATA 0x71 + +static int updating_rtc() { + outb(CMOS_ADDRESS, 0x0A); + return (inb(CMOS_DATA) & 0x80); +} + +static uint8_t get_rtc_register(int reg) { + outb(CMOS_ADDRESS, reg); + return inb(CMOS_DATA); +} + +void rtc_get_datetime(int *year, int *month, int *day, int *hour, int *minute, int *second) { + uint8_t century; + uint8_t last_second; + uint8_t last_minute; + uint8_t last_hour; + uint8_t last_day; + uint8_t last_month; + uint8_t last_year; + uint8_t last_century; + uint8_t registerB; + + while (updating_rtc()); + *second = get_rtc_register(0x00); + *minute = get_rtc_register(0x02); + *hour = get_rtc_register(0x04); + *day = get_rtc_register(0x07); + *month = get_rtc_register(0x08); + *year = get_rtc_register(0x09); + + // Note: Century register is not standard, but often 0x32 if ACPI table says so. + // For now, assume 20xx + + do { + last_second = *second; + last_minute = *minute; + last_hour = *hour; + last_day = *day; + last_month = *month; + last_year = *year; + + while (updating_rtc()); + *second = get_rtc_register(0x00); + *minute = get_rtc_register(0x02); + *hour = get_rtc_register(0x04); + *day = get_rtc_register(0x07); + *month = get_rtc_register(0x08); + *year = get_rtc_register(0x09); + } while( (last_second != *second) || (last_minute != *minute) || (last_hour != *hour) || + (last_day != *day) || (last_month != *month) || (last_year != *year) ); + + registerB = get_rtc_register(0x0B); + + // Convert BCD to binary values if necessary + if (!(registerB & 0x04)) { + *second = (*second & 0x0F) + ((*second / 16) * 10); + *minute = (*minute & 0x0F) + ((*minute / 16) * 10); + *hour = ( (*hour & 0x0F) + (((*hour & 0x70) / 16) * 10) ) | (*hour & 0x80); + *day = (*day & 0x0F) + ((*day / 16) * 10); + *month = (*month & 0x0F) + ((*month / 16) * 10); + *year = (*year & 0x0F) + ((*year / 16) * 10); + } + + // Convert 12 hour clock to 24 hour clock if necessary + if (!(registerB & 0x02) && (*hour & 0x80)) { + *hour = ((*hour & 0x7F) + 12) % 24; + } + + // Calculate full year + *year += 2000; +} diff --git a/src/kernel/rtc.h b/src/kernel/rtc.h new file mode 100644 index 0000000..87f6003 --- /dev/null +++ b/src/kernel/rtc.h @@ -0,0 +1,8 @@ +#ifndef RTC_H +#define RTC_H + +#include + +void rtc_get_datetime(int *year, int *month, int *day, int *hour, int *minute, int *second); + +#endif diff --git a/src/kernel/wm.c b/src/kernel/wm.c new file mode 100644 index 0000000..42902b1 --- /dev/null +++ b/src/kernel/wm.c @@ -0,0 +1,746 @@ +#include "wm.h" +#include "graphics.h" +#include "io.h" +#include "cmd.h" +#include "calculator.h" +#include "cli_apps/cli_utils.h" +#include "explorer.h" +#include "editor.h" +#include +#include +#include "notepad.h" +#include "control_panel.h" +#include "about.h" +#include "minesweeper.h" + +// --- State --- +static int mx = 400, my = 300; // Mouse Pos +static int prev_mx = 400, prev_my = 300; // Previous mouse position +static bool start_menu_open = false; + +// Dragging State +static bool is_dragging = false; +static Window *drag_window = NULL; +static int drag_offset_x = 0; +static int drag_offset_y = 0; + +// Windows array for z-order management +static Window *all_windows[8]; +static int window_count = 0; + +// Redraw system +static bool force_redraw = true; // Force full redraw on next tick +static uint32_t timer_ticks = 0; + +// Cursor state +static bool cursor_visible = true; +static int last_cursor_x = 400; +static int last_cursor_y = 300; + +// --- Drawing Helpers --- + +// Draw a bevelled box (Win 3.1 style) +void draw_bevel_rect(int x, int y, int w, int h, bool sunken) { + draw_rect(x, y, w, h, COLOR_GRAY); + + uint32_t top_left = sunken ? COLOR_DKGRAY : COLOR_WHITE; + uint32_t bot_right = sunken ? COLOR_WHITE : COLOR_DKGRAY; + + // Top + draw_rect(x, y, w, 1, top_left); + // Left + draw_rect(x, y, 1, h, top_left); + // Bottom + draw_rect(x, y + h - 1, w, 1, bot_right); + // Right + draw_rect(x + w - 1, y, 1, h, bot_right); +} + +void draw_button(int x, int y, int w, int h, const char *text, bool pressed) { + draw_bevel_rect(x, y, w, h, pressed); + // Center Text + int len = 0; while(text[len]) len++; + int tx = x + (w - (len * 8)) / 2; + int ty = y + (h - 8) / 2; + if (pressed) { tx++; ty++; } + draw_string(tx, ty, text, COLOR_BLACK); +} + +void draw_icon(int x, int y, const char *label) { + // Simple "File" Icon + draw_rect(x + 10, y, 20, 25, COLOR_WHITE); + draw_rect(x + 10, y, 20, 1, COLOR_BLACK); + draw_rect(x + 10, y, 1, 25, COLOR_BLACK); + draw_rect(x + 30, y, 1, 25, COLOR_BLACK); + draw_rect(x + 10, y + 25, 21, 1, COLOR_BLACK); + + // Label + draw_string(x, y + 30, label, COLOR_WHITE); +} + +void draw_folder_icon(int x, int y, const char *label) { + // Folder icon (yellow folder) + // Folder tab + draw_rect(x + 5, y, 15, 6, COLOR_LTGRAY); + draw_rect(x + 5, y, 15, 1, COLOR_BLACK); + draw_rect(x + 5, y, 1, 6, COLOR_BLACK); + draw_rect(x + 19, y, 1, 6, COLOR_BLACK); + + // Folder body + draw_rect(x + 5, y + 6, 25, 15, COLOR_LTGRAY); + draw_rect(x + 5, y + 6, 25, 1, COLOR_BLACK); + draw_rect(x + 5, y + 6, 1, 15, COLOR_BLACK); + draw_rect(x + 29, y + 6, 1, 15, COLOR_BLACK); + draw_rect(x + 5, y + 20, 25, 1, COLOR_BLACK); + + // Label + draw_string(x, y + 30, label, COLOR_WHITE); +} + +void draw_document_icon(int x, int y, const char *label) { + // Document icon (white paper with lines) + draw_rect(x + 10, y, 20, 25, COLOR_WHITE); + draw_rect(x + 10, y, 20, 1, COLOR_BLACK); + draw_rect(x + 10, y, 1, 25, COLOR_BLACK); + draw_rect(x + 30, y, 1, 25, COLOR_BLACK); + draw_rect(x + 10, y + 25, 21, 1, COLOR_BLACK); + + // Lines on document + draw_rect(x + 14, y + 8, 12, 1, COLOR_BLACK); + draw_rect(x + 14, y + 12, 12, 1, COLOR_BLACK); + draw_rect(x + 14, y + 16, 12, 1, COLOR_BLACK); + + // Label + draw_string(x, y + 30, label, COLOR_WHITE); +} + +void draw_window(Window *win) { + if (!win->visible) return; + + // Main Body + draw_bevel_rect(win->x, win->y, win->w, win->h, false); + + // Title Bar + uint32_t title_color = win->focused ? COLOR_BLUE : COLOR_DKGRAY; + draw_rect(win->x + 3, win->y + 3, win->w - 6, 18, title_color); + draw_string(win->x + 8, win->y + 8, win->title, COLOR_WHITE); + + // Close Button (X) + draw_button(win->x + win->w - 20, win->y + 5, 14, 14, "X", false); + + // Client Area + draw_rect(win->x + 4, win->y + 24, win->w - 8, win->h - 28, COLOR_WHITE); + + if (win->paint) { + win->paint(win); + } +} + +// Draw Mouse Cursor (Simple Arrow) +void draw_cursor(int x, int y) { + // 0 = Transparent (skip), 1 = Black, 2 = White + static const uint8_t cursor_bitmap[10][10] = { + {1,1,0,0,0,0,0,0,0,0}, + {1,2,1,0,0,0,0,0,0,0}, + {1,2,2,1,0,0,0,0,0,0}, + {1,2,2,2,1,0,0,0,0,0}, + {1,2,2,2,2,1,0,0,0,0}, + {1,2,2,2,2,2,1,0,0,0}, + {1,2,2,1,1,1,1,0,0,0}, + {1,1,1,0,1,2,1,0,0,0}, + {0,0,0,0,0,1,2,1,0,0}, + {0,0,0,0,0,0,1,0,0,0} + }; + + for (int r = 0; r < 10; r++) { + for (int c = 0; c < 10; c++) { + uint8_t p = cursor_bitmap[r][c]; + if (p == 1) put_pixel(x + c, y + r, COLOR_BLACK); + else if (p == 2) put_pixel(x + c, y + r, COLOR_WHITE); + } + } +} + +// Erase cursor by redrawing the background in that area +static void erase_cursor(int x, int y) { + int sw = get_screen_width(); + int sh = get_screen_height(); + + // Clamp to screen + int x1 = x < 0 ? 0 : x; + int y1 = y < 0 ? 0 : y; + int x2 = x + 10 > sw ? sw : x + 10; + int y2 = y + 10 > sh ? sh : y + 10; + int w = x2 - x1; + int h = y2 - y1; + + // Check what's underneath the cursor and redraw it + if (y1 < sh - 28) { + // Desktop or window area - draw teal background + draw_rect(x1, y1, w, h, COLOR_TEAL); + } else { + // Taskbar area - draw gray background + draw_rect(x1, y1, w, h, COLOR_GRAY); + } +} + +// --- Clock --- +static uint8_t rtc_read(uint8_t reg) { + outb(0x70, reg); + return inb(0x71); +} + +static void draw_clock(int x, int y) { + // Wait for update in progress + while (rtc_read(0x0A) & 0x80); + + uint8_t s = rtc_read(0x00); + uint8_t m = rtc_read(0x02); + uint8_t h = rtc_read(0x04); + uint8_t b = rtc_read(0x0B); + + if (!(b & 0x04)) { + s = (s & 0x0F) + ((s >> 4) * 10); + m = (m & 0x0F) + ((m >> 4) * 10); + h = (h & 0x0F) + ((h >> 4) * 10); + } + + char buf[9]; + buf[0] = '0' + (h / 10); + buf[1] = '0' + (h % 10); + buf[2] = ':'; + buf[3] = '0' + (m / 10); + buf[4] = '0' + (m % 10); + buf[5] = ':'; + buf[6] = '0' + (s / 10); + buf[7] = '0' + (s % 10); + buf[8] = 0; + + draw_string(x, y, buf, COLOR_BLACK); +} + +// --- Main Paint Function --- +void wm_paint(void) { + int sw = get_screen_width(); + int sh = get_screen_height(); + + // First, erase the old cursor (before redrawing anything) + if (cursor_visible) { + erase_cursor(last_cursor_x, last_cursor_y); + } + + // 1. Desktop + draw_desktop_background(); + + draw_folder_icon(20, 20, "Explorer"); + draw_document_icon(20, 80, "Notepad"); + + // 3. Windows - sort by z-index and draw + // Simple bubble sort by z-index (5 windows max) + Window *sorted_windows[6]; + for (int i = 0; i < window_count; i++) { + sorted_windows[i] = all_windows[i]; + } + + for (int i = 0; i < window_count - 1; i++) { + for (int j = 0; j < window_count - i - 1; j++) { + if (sorted_windows[j]->z_index > sorted_windows[j + 1]->z_index) { + Window *temp = sorted_windows[j]; + sorted_windows[j] = sorted_windows[j + 1]; + sorted_windows[j + 1] = temp; + } + } + } + + // Draw windows in z-order (lowest first) + for (int i = 0; i < window_count; i++) { + draw_window(sorted_windows[i]); + } + + // 4. Taskbar + draw_rect(0, sh - 28, sw, 28, COLOR_GRAY); + draw_rect(0, sh - 28, sw, 2, COLOR_WHITE); // Top highlight + + // 5. Start Button + draw_button(2, sh - 26, 60, 24, "Start", start_menu_open); + + // Clock + draw_clock(sw - 80, sh - 20); + + // 6. Start Menu (if open) + if (start_menu_open) { + int menu_h = 230; + int menu_y = sh - 28 - menu_h; + draw_bevel_rect(0, menu_y, 120, menu_h, false); + + // Items + draw_string(8, menu_y + 8, "Explorer", COLOR_BLACK); + draw_string(8, menu_y + 28, "Notepad", COLOR_BLACK); + draw_string(8, menu_y + 48, "Editor", COLOR_BLACK); + draw_string(8, menu_y + 68, "CMD", COLOR_BLACK); + draw_string(8, menu_y + 88, "Calculator", COLOR_BLACK); + draw_string(8, menu_y + 108, "Minesweeper", COLOR_BLACK); + draw_string(8, menu_y + 128, "Control Panel", COLOR_BLACK); + draw_string(8, menu_y + 148, "About BrewOS", COLOR_BLACK); + + // Separator line + draw_rect(5, menu_y + 165, 110, 1, COLOR_BLACK); + + // Power options at bottom + draw_string(8, menu_y + 175, "Shutdown", COLOR_BLACK); + draw_string(8, menu_y + 195, "Restart", COLOR_BLACK); + } + + // 7. Mouse cursor (draw last so it's on top) + draw_cursor(mx, my); + last_cursor_x = mx; + last_cursor_y = my; + + // Flip the buffer - display the rendered frame atomically + graphics_flip_buffer(); +} + +// --- Input Handling --- + +bool rect_contains(int x, int y, int w, int h, int px, int py) { + return px >= x && px < x + w && py >= y && py < y + h; +} + +void wm_handle_click(int x, int y) { + int sh = get_screen_height(); + + // Check Start Button + if (rect_contains(2, sh - 26, 60, 24, x, y)) { + start_menu_open = !start_menu_open; + force_redraw = true; + return; + } + + // Check Start Menu Items + if (start_menu_open) { + int menu_h = 230; + int menu_y = sh - 28 - menu_h; + if (rect_contains(0, menu_y, 120, menu_h, x, y)) { + // Clear focus from all windows first + for (int i = 0; i < window_count; i++) { + all_windows[i]->focused = false; + } + + // Find which item was clicked + if (y < menu_y + 25) { // Explorer + win_explorer.visible = true; + win_explorer.focused = true; + // Bring to front + int max_z = 0; + for (int i = 0; i < window_count; i++) { + if (all_windows[i]->z_index > max_z) { + max_z = all_windows[i]->z_index; + } + } + win_explorer.z_index = max_z + 1; + start_menu_open = false; + } else if (y < menu_y + 45) { // Notepad + win_notepad.visible = true; + win_notepad.focused = true; + int max_z = 0; + for (int i = 0; i < window_count; i++) { + if (all_windows[i]->z_index > max_z) { + max_z = all_windows[i]->z_index; + } + } + win_notepad.z_index = max_z + 1; + start_menu_open = false; + } else if (y < menu_y + 65) { // Editor + win_editor.visible = true; + win_editor.focused = true; + int max_z = 0; + for (int i = 0; i < window_count; i++) { + if (all_windows[i]->z_index > max_z) { + max_z = all_windows[i]->z_index; + } + } + win_editor.z_index = max_z + 1; + start_menu_open = false; + } else if (y < menu_y + 85) { // CMD + win_cmd.visible = true; + win_cmd.focused = true; + int max_z = 0; + for (int i = 0; i < window_count; i++) { + if (all_windows[i]->z_index > max_z) { + max_z = all_windows[i]->z_index; + } + } + win_cmd.z_index = max_z + 1; + start_menu_open = false; + } else if (y < menu_y + 105) { // Calculator + win_calculator.visible = true; + win_calculator.focused = true; + int max_z = 0; + for (int i = 0; i < window_count; i++) { + if (all_windows[i]->z_index > max_z) { + max_z = all_windows[i]->z_index; + } + } + win_calculator.z_index = max_z + 1; + start_menu_open = false; + } else if (y < menu_y + 125) { // Minesweeper + win_minesweeper.visible = true; + win_minesweeper.focused = true; + int max_z = 0; + for (int i = 0; i < window_count; i++) { + if (all_windows[i]->z_index > max_z) { + max_z = all_windows[i]->z_index; + } + } + win_minesweeper.z_index = max_z + 1; + start_menu_open = false; + } else if (y < menu_y + 145) { // Control Panel + win_control_panel.visible = true; + win_control_panel.focused = true; + int max_z = 0; + for (int i = 0; i < window_count; i++) { + if (all_windows[i]->z_index > max_z) { + max_z = all_windows[i]->z_index; + } + } + win_control_panel.z_index = max_z + 1; + start_menu_open = false; + } else if (y < menu_y + 165) { // About BrewOS + win_about.visible = true; + win_about.focused = true; + int max_z = 0; + for (int i = 0; i < window_count; i++) { + if (all_windows[i]->z_index > max_z) { + max_z = all_windows[i]->z_index; + } + } + win_about.z_index = max_z + 1; + start_menu_open = false; + } else if (y < menu_y + 185) { // Shutdown + cli_cmd_shutdown(NULL); + start_menu_open = false; + } else { // Restart + cli_cmd_reboot(NULL); + start_menu_open = false; + } + force_redraw = true; + return; + } + } + + // Find topmost window at click location + Window *topmost = NULL; + int topmost_z = -1; + + for (int i = 0; i < window_count; i++) { + Window *win = all_windows[i]; + if (win->visible && rect_contains(win->x, win->y, win->w, win->h, x, y)) { + if (win->z_index > topmost_z) { + topmost = win; + topmost_z = win->z_index; + } + } + } + + // If a window was clicked + if (topmost != NULL) { + // Clear focus from all windows + for (int i = 0; i < window_count; i++) { + all_windows[i]->focused = false; + } + + // Bring it to front + int max_z = 0; + for (int i = 0; i < window_count; i++) { + if (all_windows[i]->z_index > max_z) { + max_z = all_windows[i]->z_index; + } + } + topmost->z_index = max_z + 1; + topmost->focused = true; + + // Check close button + if (rect_contains(topmost->x + topmost->w - 20, topmost->y + 5, 14, 14, x, y)) { + topmost->visible = false; + // Reset window state on close + if (topmost == &win_explorer) { + explorer_reset(); + } else if (topmost == &win_notepad) { + notepad_reset(); + } else if (topmost == &win_control_panel) { + control_panel_reset(); + } + } else if (y < topmost->y + 24) { + // Dragging the title bar + is_dragging = true; + drag_window = topmost; + drag_offset_x = x - topmost->x; + drag_offset_y = y - topmost->y; + } else { + // Content click + if (topmost->handle_click) { + topmost->handle_click(topmost, x - topmost->x, y - topmost->y); + } + } + } else { + // No window clicked - check desktop icons + // Clear focus from all windows first + for (int i = 0; i < window_count; i++) { + all_windows[i]->focused = false; + } + + if (rect_contains(20, 20, 40, 40, x, y)) { + // Explorer icon + win_explorer.visible = true; + win_explorer.focused = true; + int max_z = 0; + for (int i = 0; i < window_count; i++) { + if (all_windows[i]->z_index > max_z) { + max_z = all_windows[i]->z_index; + } + } + win_explorer.z_index = max_z + 1; + } else if (rect_contains(20, 80, 40, 40, x, y)) { + // Notepad icon + win_notepad.visible = true; + win_notepad.focused = true; + int max_z = 0; + for (int i = 0; i < window_count; i++) { + if (all_windows[i]->z_index > max_z) { + max_z = all_windows[i]->z_index; + } + } + win_notepad.z_index = max_z + 1; + } else if (rect_contains(20, 140, 40, 40, x, y)) { + win_calculator.visible = true; + win_calculator.focused = true; + int max_z = 0; + for (int i = 0; i < window_count; i++) { + if (all_windows[i]->z_index > max_z) { + max_z = all_windows[i]->z_index; + } + } + win_calculator.z_index = max_z + 1; + } else if (rect_contains(20, 200, 40, 40, x, y)) { + win_explorer.visible = true; + win_explorer.focused = true; + int max_z = 0; + for (int i = 0; i < window_count; i++) { + if (all_windows[i]->z_index > max_z) { + max_z = all_windows[i]->z_index; + } + } + win_explorer.z_index = max_z + 1; + } else if (rect_contains(20, 260, 40, 40, x, y)) { + win_editor.visible = true; + win_editor.focused = true; + int max_z = 0; + for (int i = 0; i < window_count; i++) { + if (all_windows[i]->z_index > max_z) { + max_z = all_windows[i]->z_index; + } + } + win_editor.z_index = max_z + 1; + } + } + + // Close start menu if clicked elsewhere + if (start_menu_open) { + start_menu_open = false; + } + + force_redraw = true; +} + +// Handle right click (context menu or special actions) +void wm_handle_right_click(int x, int y) { + int sh = get_screen_height(); + + // Find topmost window at click location + Window *topmost = NULL; + int topmost_z = -1; + + for (int i = 0; i < window_count; i++) { + Window *win = all_windows[i]; + if (win->visible && rect_contains(win->x, win->y, win->w, win->h, x, y)) { + if (win->z_index > topmost_z) { + topmost = win; + topmost_z = win->z_index; + } + } + } + + // If a window was clicked + if (topmost != NULL) { + // Don't process close button or title bar for right click + if (y >= topmost->y + 24) { + // Content right click + if (topmost->handle_right_click) { + topmost->handle_right_click(topmost, x - topmost->x, y - topmost->y); + } + } + } + + force_redraw = true; +}void wm_handle_mouse(int dx, int dy, uint8_t buttons) { + int sw = get_screen_width(); + int sh = get_screen_height(); + + prev_mx = mx; + prev_my = my; + + mx += dx; + my += dy; + + if (mx < 0) mx = 0; + if (my < 0) my = 0; + if (mx >= sw) mx = sw - 1; + if (my >= sh) my = sh - 1; + + static bool prev_left = false; + static bool prev_right = false; + bool left = buttons & 0x01; + bool right = buttons & 0x02; + + if (left && !prev_left) { + wm_handle_click(mx, my); + } else if (right && !prev_right) { + wm_handle_right_click(mx, my); + } else if (left && is_dragging && drag_window) { + drag_window->x = mx - drag_offset_x; + drag_window->y = my - drag_offset_y; + // Mark for full redraw since window moved + force_redraw = true; + } else if (left && !is_dragging && (dx != 0 || dy != 0)) { + // Mouse is moving while left button is held, but window isn't being dragged + force_redraw = true; + } else if (!left && is_dragging) { + is_dragging = false; + drag_window = NULL; + // Mark for full redraw since dragging ended + force_redraw = true; + } + + prev_left = left; + prev_right = right; + + if (prev_mx != mx || prev_my != my) { + // Cursor moved - just mark dirty cursor areas + wm_mark_dirty(prev_mx, prev_my, 10, 10); + wm_mark_dirty(mx, my, 10, 10); + } + + prev_left = left; +} + +void wm_handle_key(char c) { + Window *target = NULL; + if (win_notepad.focused && win_notepad.visible) target = &win_notepad; + else if (win_cmd.focused && win_cmd.visible) target = &win_cmd; + else if (win_calculator.focused && win_calculator.visible) target = &win_calculator; + else if (win_explorer.focused && win_explorer.visible) target = &win_explorer; + else if (win_editor.focused && win_editor.visible) target = &win_editor; + else if (win_control_panel.focused && win_control_panel.visible) target = &win_control_panel; + + if (!target) return; + + if (target->handle_key) { + target->handle_key(target, c); + } + + // Mark window as needing redraw on next timer tick + wm_mark_dirty(target->x, target->y, target->w, target->h); +} + +void wm_mark_dirty(int x, int y, int w, int h) { + graphics_mark_dirty(x, y, w, h); +} + +void wm_refresh(void) { + force_redraw = true; +} + +void wm_init(void) { + notepad_init(); + cmd_init(); + calculator_init(); + explorer_init(); + editor_init(); + control_panel_init(); + about_init(); + minesweeper_init(); + + // Initialize z-indices + win_notepad.z_index = 0; + win_cmd.z_index = 1; + win_calculator.z_index = 2; + win_explorer.z_index = 3; + win_editor.z_index = 4; + win_control_panel.z_index = 5; + win_about.z_index = 6; + win_minesweeper.z_index = 7; + + // Register windows in array + all_windows[0] = &win_notepad; + all_windows[1] = &win_cmd; + all_windows[2] = &win_calculator; + all_windows[3] = &win_explorer; + all_windows[4] = &win_editor; + all_windows[5] = &win_control_panel; + all_windows[6] = &win_about; + all_windows[7] = &win_minesweeper; + window_count = 8; + + // Only show Explorer and Notepad on desktop (Explorer on top) + win_explorer.visible = false; + win_explorer.focused = false; + win_explorer.z_index = 10; + + win_notepad.visible = false; + win_notepad.focused = false; + win_notepad.z_index = 9; + + // Rest are hidden initially + win_cmd.visible = false; + win_calculator.visible = false; + win_editor.visible = false; + win_control_panel.visible = false; + win_about.visible = false; + win_minesweeper.visible = false; + + force_redraw = true; +} + +// Called by timer interrupt ~60Hz +void wm_timer_tick(void) { + timer_ticks++; + + // Only redraw if there are dirty areas (clock updates at most every second, cursor rarely moves in timer only) + // Most of the time, nothing changes between ticks + + static uint8_t last_second = 0xFF; + + outb(0x70, 0x00); + uint8_t current_sec = inb(0x71); + + if (current_sec != last_second) { + last_second = current_sec; + int sw = get_screen_width(); + int sh = get_screen_height(); + // Mark clock area + a bit of buffer + wm_mark_dirty(sw - 90, sh - 30, 90, 20); + } + + // If force_redraw is set, do a full redraw + if (force_redraw) { + graphics_mark_screen_dirty(); + force_redraw = false; + } + + // Perform redraw if there are dirty areas + DirtyRect dirty = graphics_get_dirty_rect(); + if (dirty.active) { + wm_paint(); + graphics_clear_dirty(); + } +} diff --git a/src/kernel/wm.h b/src/kernel/wm.h new file mode 100644 index 0000000..5863f97 --- /dev/null +++ b/src/kernel/wm.h @@ -0,0 +1,52 @@ +#ifndef WM_H +#define WM_H + +#include +#include + +// --- Constants --- +#define COLOR_TEAL 0xFF008080 +#define COLOR_GRAY 0xFFC0C0C0 +#define COLOR_WHITE 0xFFFFFFFF +#define COLOR_BLACK 0xFF000000 +#define COLOR_BLUE 0xFF000080 +#define COLOR_LTGRAY 0xFFDFDFDF +#define COLOR_DKGRAY 0xFF808080 +#define COLOR_RED 0xFFFF0000 +#define COLOR_COFFEE 0xFF6B4423 + +typedef struct Window Window; +struct Window { + char *title; + int x, y, w, h; + bool visible; + char buffer[1024]; + int buf_len; + int cursor_pos; + bool focused; + int z_index; // Layering depth (higher = on top) + + // Callbacks + void (*paint)(Window *win); + void (*handle_key)(Window *win, char c); + void (*handle_click)(Window *win, int x, int y); + void (*handle_right_click)(Window *win, int x, int y); +}; + +void wm_init(void); +void wm_handle_mouse(int dx, int dy, uint8_t buttons); +void wm_handle_key(char c); +void wm_handle_click(int x, int y); +void wm_handle_right_click(int x, int y); + +// Redraw system +void wm_mark_dirty(int x, int y, int w, int h); +void wm_refresh(void); +void wm_paint(void); +void wm_timer_tick(void); + +// Drawing helpers +void draw_bevel_rect(int x, int y, int w, int h, bool sunken); +void draw_button(int x, int y, int w, int h, const char *text, bool pressed); + +#endif