diff -urN orig/CREDITS linux/CREDITS --- orig/CREDITS Mon Aug 5 13:29:38 2002 +++ linux/CREDITS Mon Aug 5 13:41:06 2002 @@ -2215,6 +2215,15 @@ S: Potsdam, New York 13676 S: USA +N: Dave Neuer +E: dneuer@innovation-charter.com +E: mr_fred_smoothie@yahoo.com +D: Helped implement support for Compaq's H31xx series iPAQs +D: Other mostly minor tweaks & bugfixes +S: 325 E. Main St., Suite 3 +S: Carnegie, PA 15105 +S: USA + N: Michael Neuffer E: mike@i-Connect.Net E: neuffer@goofy.zdv.uni-mainz.de diff -urN orig/Documentation/Configure.help linux/Documentation/Configure.help --- orig/Documentation/Configure.help Mon Aug 5 13:29:39 2002 +++ linux/Documentation/Configure.help Thu Feb 27 23:20:12 2003 @@ -4090,6 +4090,13 @@ Say Y to enable support for Permedia2 AGP frame buffer card from 3Dlabs (aka `Graphic Blaster Exxtreme') on the PCI bus. +Permedia3 support (EXPERIMENTAL) +CONFIG_FB_PM3 + This is the frame buffer device driver for the 3DLabs Permedia3 + chipset, used in Formac ProFormance III, 3DLabs Oxygen VX1 & + similar boards, 3DLabs Permedia3 Create!, Appian Jeronimo 2000 + and maybe other boards. + Phase5 CVisionPPC/BVisionPPC support CONFIG_FB_PM2_CVPPC Say Y to enable support for the Amiga Phase 5 CVisionPPC BVisionPPC @@ -9558,7 +9565,7 @@ whenever you want). If you want to compile it as a module, say M here and read . -CSZ packet scheduler +CSZ packet scheduler (experimental) CONFIG_NET_SCH_CSZ Say Y here if you want to use the Clark-Shenker-Zhang (CSZ) packet scheduling algorithm for some of your network devices. At the @@ -11648,6 +11655,17 @@ The module will be called tmspci.o. If you want to compile it as a module, say M here and read . +Altera ether00 support +CONFIG_ETHER00 + This is the driver for Altera's ether00 ethernet mac IP core. Say + Y here if you want to build support for this into the kernel. It + is also available as a module (say M here) that can be inserted/ + removed from the kernel at the same time as the PLD is configured. + If this driver is running on an epxa10 development board then it + will generate a suitable hw address based on the board serial + number (MTD support is required for this). Otherwise you will + need to set a suitable hw address using ifconfig. + Generic TMS380 ISA support CONFIG_TMSISA This tms380 module supports generic TMS380-based ISA cards. @@ -11680,6 +11698,7 @@ The module will be called madgemc.o. If you want to compile it as a module, say M here and read . + SMC ISA/MCA Token Ring adapter support CONFIG_SMCTR This is support for the ISA and MCA SMC Token Ring cards, @@ -13278,6 +13297,16 @@ support" be compiled as a module for this driver to be used properly. +Altera's uart00 serial driver +CONFIG_SERIAL_UART00 + Say Y here if you want to use the hard logic uart on Excalibur. This + driver also supports soft logic implentations of this uart core. + +Serial console on uart00 +CONFIG_SERIAL_UART00_CONSOLE + Say Y here if you want to support a serial console on an Excalibur + hard logic uart or uart00 IP core. + USB ConnectTech WhiteHEAT Serial Driver CONFIG_USB_SERIAL_WHITEHEAT Say Y here if you want to use a ConnectTech WhiteHEAT 4 port @@ -16841,6 +16870,20 @@ . The module will be called i2c-velleman.o. +Guide GPIO adapter +CONFIG_I2C_GUIDE + This supports the Iders GUIDE I2C bit-bashing adapter. If you have + selected the GUIDE A07 as your ARM system type, you cannot deselect + this option, as it is required for proper operation of the GUIDE. + + This interface uses /dev/i2c-0 (major 89, minor 0). + + Say Y if you own such an adapter. + + This driver is also available as a module. If you want to compile + it as a module, say M here and read Documentation/modules.txt. The + module will be called i2c-guide.o. + I2C PCF 8584 interfaces CONFIG_I2C_ALGOPCF This allows you to use a range of I2C adapters called PCF adapters. @@ -17893,6 +17936,17 @@ . The module will be called softdog.o. +SA1100 Internal Watchdog +CONFIG_SA1100_WATCHDOG + Watchdog timer embedded into SA11x0 chips. This will reboot your + system when timeout is reached. + NOTE, that once enabled, this timer cannot be disabled. + + This driver is also available as a module ( = code which can be + inserted in and removed from the running kernel whenever you want). + If you want to compile it as a module, say M here and read + Documentation/modules.txt. The module will be called sa1100_wdt.o. + Berkshire Products PC Watchdog CONFIG_PCWATCHDOG This is the driver for the Berkshire Products PC Watchdog card. @@ -19480,6 +19534,30 @@ from RME. If you want to acess advanced features of the card, read Documentation/sound/rme96xx. +Assabet audio (UDA1341) support +CONFIG_SOUND_ASSABET_UDA1341 + Say Y or M if you have an Intel Assabet evaluation board and want to + use the Philips UDA 1341 audio chip (the one that drives the stereo + audio output) on the SA1100 SSP port. + +Compaq iPAQ audio support +CONFIG_SOUND_H3600_UDA1341 + Say Y or M if you have a Compaq iPaq handheld computer and want to + use its Philips UDA 1341 audio chip. + +Audio support for SA1111/UDA1341 +CONFIG_SOUND_SA1111_UDA1341 + Say Y or M if you have an SA11x0 system with a Philips UDA 1341 + connected to the SA11x1. An example of such a system is the Intel + Assabet evaluation board connected to a Neponset expansion board. + +Generic DAC on the SA11x0 SSP port +CONFIG_SOUND_SA1100SSP + Say Y or M if you have an SA-11x0 system with a DAC on the SSP port. + The LART has an Burr-Brown PCM 1710 digital to analog convertor on + the SSP port, so you want to say Y or M for the LART. It might work + on other SA-1100 platforms, too, but this is not tested. + Are you using a crosscompiler CONFIG_CROSSCOMPILE Say Y here if you are compiling the kernel on a different @@ -22834,6 +22912,20 @@ Say Y if configuring for a Pangolin. Say N otherwise. +Shannon +CONFIG_SA1100_SHANNON + The Shannon (also known as a Tuxscreen, and also as a IS2630) was a + limited edition webphone produced by Philips. The Shannon is a SA1100 + platform with a 640x480 LCD, touchscreen, CIR keyboard, PCMCIA slots, + and a telco interface. + +Simputer +CONFIG_SA1100_SIMPUTER + Say Y here if you are using an Intel(R) StrongARM(R) SA-1110 + based Simputer. See http://www.simputer.org/ for information + on the Simputer. The Simputer software is actively maintained + by PicoPeta Simputers Pvt. Ltd. (http://www.picopeta.com) + Victor CONFIG_SA1100_VICTOR Say Y here if you are using a Visu Aide Intel(R) StrongARM(R) @@ -22841,6 +22933,14 @@ for information on this system. +Radisys Corp. Tulsa +CONFIG_SA1100_PFS168 + The Radisys Corp. PFS-168 (aka Tulsa) is an IntelŪ StrongArmŪ SA-1110 based + computer which includes the SA-1111 Microprocessor Companion Chip and other + custom I/O designed to add connectivity and multimedia features for vending + and business machine applications. Say Y here if you require support for + this target. + # Choice: cerf_ram Cerf on-board RAM size CONFIG_SA1100_CERF_8MB @@ -22908,6 +23008,36 @@ Say Y if you want support for the ARM920T processor. Otherwise, say N. +Support ARM922T processor +CONFIG_CPU_ARM922T + The ARM922T is a version of the ARM920T, but with smaller + instruction and data caches. It is used in Altera's + Excalibur XA device family. + + Say Y if you want support for the ARM922T processor. + Otherwise, say N. + +Support low power wait for interrupt +CONFIG_CPU_ARM922_CPU_IDLE + Saying Y here will allow the processor to enter a low power + mode whilst waiting for an interrupt in idle. If you're unsure + say Y. + +Enable ARM922T instruction cache +CONFIG_CPU_ARM922_I_CACHE_ON + Say Y here to enable the processor instruction cache. Unless + you have a reason not to, say Y. + +Enable ARM922T data cache +CONFIG_CPU_ARM922_D_CACHE_ON + Say Y here to enable the processor data cache. Unless + you have a reason not to, say Y. + +Use data cache in writethrough mode +CONFIG_CPU_ARM922_WRITETHROUGH + Say Y here to use the data cache in writethough mode. Unless you + specifically require this, say N. + Support ARM1020 processor CONFIG_CPU_ARM1020 The ARM1020 is the cached version of the ARM10 processor, @@ -22945,6 +23075,11 @@ Say Y here if you are using the inhand electronics OmniMeter. See for details. +HP Laboratories BadgePAD 4 +CONFIG_SA1100_BADGE4 + Say Y here if you want to build a kernel for the HP Laboratories + BadgePAD 4. + Load kernel using Angel Debug Monitor CONFIG_ANGELBOOT Say Y if you plan to load the kernel using Angel, ARM Ltd's target @@ -22957,6 +23092,15 @@ board includes 2 serial ports, Ethernet, IRDA, and expansion headers. It comes with 16 MB SDRAM and 8 MB flash ROM. +GUIDEA07 +CONFIG_ARCH_GUIDEA07 + Say Y if you are using a GUIDE (A07) board. + + This board is based on the cs89712 processor and shares much common + hardware with the CDB89712 configuration. When you select this + option and the CDB89712 becomes enabled also, don't worry. It's + supposed to be that way. + CLPS-711X internal ROM bootstrap CONFIG_EP72XX_ROM_BOOT If you say Y here, your CLPS711x-based kernel will use the bootstrap @@ -22988,16 +23132,14 @@ FastFPE math emulation CONFIG_FPE_FASTFPE Say Y here to include the FAST floating point emulator in the kernel. - This is an experimental much faster emulator which has only 32 bit + This is an experimental much faster emulator which now also has full precision for the mantissa. It does not support any exceptions. - This makes it very simple, it is approximately 4-8 times faster than - NWFPE. + It is very simple, and approximately 3-6 times faster than NWFPE. - It should be sufficient for most programs. It is definitely not - suitable if you do scientific calculations that need double - precision for iteration formulas that sum up lots of very small - numbers. If you do not feel you need a faster FP emulation you - should better choose NWFPE. + It should be sufficient for most programs. It may be not suitable + for scientific calculations, but you have to check this for yourself. + If you do not feel you need a faster FP emulation you should better + choose NWFPE. It is also possible to say M to build the emulator as a module (fastfpe.o). But keep in mind that you should only load the FP diff -urN orig/Documentation/arm/Booting linux/Documentation/arm/Booting --- orig/Documentation/arm/Booting Thu Jan 1 01:00:00 1970 +++ linux/Documentation/arm/Booting Sat May 18 22:17:57 2002 @@ -0,0 +1,137 @@ + Booting ARM Linux + ================= + +Author: Russell King +Date : 18 May 2002 + +The following documentation is relevant to 2.4.18-rmk6 and beyond. + +In order to boot ARM Linux, you require a boot loader, which is a small +program that runs before the main kernel. The boot loader is expected +to initialise various devices, and eventually call the Linux kernel, +passing information to the kernel. + +Essentially, the boot loader should provide (as a minimum) the +following: + +1. Setup and initialise the RAM. +2. Initialise one serial port. +3. Detect the machine type. +4. Setup the kernel tagged list. +5. Call the kernel image. + + +1. Setup and initialise RAM +--------------------------- + +Existing boot loaders: MANDATORY +New boot loaders: MANDATORY + +The boot loader is expected to find and initialise all RAM that the +kernel will use for volatile data storage in the system. It performs +this in a machine dependent manner. (It may use internal algorithms +to automatically locate and size all RAM, or it may use knowledge of +the RAM in the machine, or any other method the boot loader designer +sees fit.) + + +2. Initialise one serial port +----------------------------- + +Existing boot loaders: OPTIONAL, RECOMMENDED +New boot loaders: OPTIONAL, RECOMMENDED + +The boot loader should initialise and enable one serial port on the +target. This allows the kernel serial driver to automatically detect +which serial port it should use for the kernel console (generally +used for debugging purposes, or communication with the target.) + +As an alternative, the boot loader can pass the relevant 'console=' +option to the kernel via the tagged lists specifing the port, and +serial format options as described in + + linux/Documentation/kernel-parameters.txt. + + +3. Detect the machine type +-------------------------- + +Existing boot loaders: OPTIONAL +New boot loaders: MANDATORY + +The boot loader should detect the machine type its running on by some +method. Whether this is a hard coded value or some algorithm that +looks at the connected hardware is beyond the scope of this document. +The boot loader must ultimately be able to provide a MACH_TYPE_xxx +value to the kernel. (see linux/arch/arm/tools/mach-types). + + +4. Setup the kernel tagged list +------------------------------- + +Existing boot loaders: OPTIONAL, HIGHLY RECOMMENDED +New boot loaders: MANDATORY + +The boot loader must create and initialise the kernel tagged list. +A valid tagged list starts with ATAG_CORE and ends with ATAG_NONE. +The ATAG_CORE tag may or may not be empty. An empty ATAG_CORE tag +has the size field set to '2' (0x00000002). The ATAG_NONE must set +the size field to zero. + +Any number of tags can be placed in the list. It is undefined +whether a repeated tag appends to the information carried by the +previous tag, or whether it replaces the information in its +entirety; some tags behave as the former, others the latter. + +The boot loader must pass at a minimum the size and location of +the system memory, and root filesystem location. Therefore, the +minimum tagged list should look: + + +-----------+ +base -> | ATAG_CORE | | + +-----------+ | + | ATAG_MEM | | increasing address + +-----------+ | + | ATAG_NONE | | + +-----------+ v + +The tagged list should be stored in system RAM. + +The tagged list must be placed in a region of memory where neither +the kernel decompressor nor initrd 'bootp' program will overwrite +it. The recommended placement is in the first 16KiB of RAM. + +5. Calling the kernel image +--------------------------- + +Existing boot loaders: MANDATORY +New boot loaders: MANDATORY + +There are two options for calling the kernel zImage. If the zImage +is stored in flash, and is linked correctly to be run from flash, +then it is legal for the boot loader to call the zImage in flash +directly. + +The zImage may also be placed in system RAM (at any location) and +called there. Note that the kernel uses 16K of RAM below the image +to store page tables. The recommended placement is 32KiB into RAM. + +In either case, the following conditions must be met: + +- CPU register settings + r0 = 0, + r1 = machine type number discovered in (3) above. + r2 = physical address of tagged list in system RAM. + +- CPU mode + All forms of interrupts must be disabled (IRQs and FIQs) + The CPU must be in SVC mode. (A special exception exists for Angel) + +- Caches, MMUs + The MMU must be off. + Instruction cache may be on or off. + Data cache must be off. + +- The boot loader is expected to call the kernel image by jumping + directly to the first instruction of the kernel image. + diff -urN orig/Documentation/arm/ConfigVars linux/Documentation/arm/ConfigVars --- orig/Documentation/arm/ConfigVars Thu Jan 1 01:00:00 1970 +++ linux/Documentation/arm/ConfigVars Thu Jul 20 18:28:19 2000 @@ -0,0 +1,26 @@ +CONFIG_ARCH_CO285 - co-ebsa285 +CONFIG_ARCH_FOOTBRIDGE - all other footbridge systems + + CONFIG_ARCH_CATS - CATS support + CONFIG_ARCH_PERSONAL_SERVER - personal server support + CONFIG_ARCH_EBSA285_ADDIN - add-in ebsa285 support + CONFIG_ARCH_EBSA285_HOST - host ebsa285 support + CONFIG_ARCH_NETWINDER - netwinder support + +CONFIG_FOOTBRIDGE - any system with a footbridge chip + |- (CONFIG_ARCH_CO285) + \- (CONFIG_ARCH_FOOTBRIDGE) + +CONFIG_FOOTBRIDGE_HOST - any system with host mode footbridge support + |- (CONFIG_ARCH_CATS) + |- (CONFIG_ARCH_EBSA285_HOST) + |- (CONFIG_ARCH_NETWINDER) + \- (CONFIG_ARCH_PERSONAL_SERVER) + +CONFIG_FOOTBRIDGE_ADDIN - any system with addin mode footbridge support + |- (CONFIG_ARCH_CO285) + \- (CONFIG_ARCH_EBSA285_ADDIN) + +CONFIG_ARCH_EBSA285 - either host or add-in ebsa285 support, but + |- (CONFIG_ARCH_EBSA285_ADDIN) not co-ebsa285 + \- (CONFIG_ARCH_EBSA285_HOST) diff -urN orig/Documentation/arm/MEMC linux/Documentation/arm/MEMC --- orig/Documentation/arm/MEMC Thu Jan 1 01:00:00 1970 +++ linux/Documentation/arm/MEMC Mon May 3 19:51:25 1999 @@ -0,0 +1,57 @@ +MEMC enhancements for Linux 2.3 +------------------------------- + +The current interface: + + There is a cache of the MEMC settings held in tsk->tss.memcmap, which is + kept up to date by the following functions (from the page tables): + + update_memc_all() hits: 2 + Updates all MEMC caches on all processes. Update the real MEMC + to reflect the `current' tasks page tables. + + update_memc_tsk(tsk) hits: 0 + Update the MEMC cache for the specified task. If tsk is the + `current' task, then update the real MEMC as well. + + update_memc_mm(mm) hits: 16 + Update the MEMC cache for any task which has a mm_struct + corresponding to `mm'. If the `current' tasks mm_struct + includes this, then update the real MEMC as well. + + update_memc_addr(mm, addr, pte) hits: 8 + Update the MEMC cache entry defined by the physical address + in pte for any task which has a mm_struct corresponding to `mm'. + If the `current' tasks mm_struct includes this, then update the + real MEMC as well. + +The proposed interface: + + Couple the MEMC cache into the mm_struct so that we only have to + keep one copy per mm_struct. This also allows us to reduce the + number of loops through all existing tasks on each MEMC change. + + memc_clear(mm, physaddr) hits: 6 + Clear the MEMC mapping associated with the physical address + `physaddr'. If the `current' tasks mm_struct is `mm', then + update the real MEMC as well. (should equate to a possible + two writes and zero reads). + + memc_update_addr(mm, pte, logaddr) hits: 10 + Change the MEMC mapping for the physical address specified + in `pte' to point to the logical address `logaddr', with the + protection specified in `pte'. If the `current' tasks mm_struct + is `mm', then update the real MEMC as well. (should again equate + to a possible two writes and zero reads). + + memc_update_mm(mm) hits: 7 + Rebuild the MEMC mappings for the specified `mm' in the same way + that update_memc_mm used to. If the `current' tasks mm_struct + is `mm', update the real MEMC as well. + + memc_update_all() hits: 2 + Rebuild the MEMC mappings for all mm_structs, including the real + MEMC. + +The hit numbers are approximate usage of each function in the 2.2.7 +memory management (mm) code, and there are other uses outside this area. diff -urN orig/Documentation/arm/SA1100/SA1100_USB linux/Documentation/arm/SA1100/SA1100_USB --- orig/Documentation/arm/SA1100/SA1100_USB Thu Jan 1 01:00:00 1970 +++ linux/Documentation/arm/SA1100/SA1100_USB Tue Oct 16 20:02:23 2001 @@ -0,0 +1,344 @@ +StrongARM SA-1100 USB function Driver +Ward Willats - 08Mar01 + +1. History +'''''''''' +Brad Parker ported the DEC/Compaq "Itsy" +SA-1100 USB Function driver to the 2.4.x code base in late 2000, for +use as an "ethernet over usb" link. His original notes are here in +section 4. Nicolas Pitre rewrote the transmitter and +reciver (endpoints 1 and 2) in late 2000 to use the standard SA DMA +API and I added a bulk character interface and reworked the control +control module code and rewrote endpoint zero in early 2001. + +This release (22Feb01) is the first that completely separates +client modules (usb-eth.c and usb-char.c) from the SA-1100 USB core. +(usb_ctl, usb_ep0, usb_send and usb_receive)and makes the whole +mess a module. Oleg Drokin has done a huge amount of work, fixing +things I break and adding support for the generic usbnet driver +from the AC tree. + +2. Usage +'''''''' +Turn on CONFIG_SA1100_USB_NETLINK to use the "ethernet over usb" +functionality. Turn it off to use the character oriented +interface. The character driver currently uses mknod c 10 240. + +Programming: +The public interface is in sa1100_usb.h. For a client USB service +to use the SA-1100 USB core driver it should: + +1. Call sa1100_usb_open() to get the usb core assigned to it. + +2. Setup descriptors as appropriate for the task at hand. Esp. +important are endpoint max packet lengths, vendor and product IDs, +and type of endpoint (bulk or interrupt). Call +sa1100_get_descriptor_ptr() to get this. + +3. Call sa1100_usb_start() to actually start the usb hardware. +At this time the host will configure the device. + +...at shutdown... + +4. Call sa1100_usb_stop() to stop the USB core. +5. Call sa1100_usb_close() to free the core for use by another +client. + +3. Netlink Usage +'''''''''''''''' +StrongARM SA-1100 USB function "ethernet over usb driver" +Brad Parker + +I ported the DEC "Itsy" usb "ethernet over usb driver" code to the 2.4.x +base and made some enhacements and bug fixes. This code has 2 sides and +implements a simple "ethernet over usb" functionality. + +function (SA1100) side: +- the driver has two endpoints and uses interrupt and bulk transfer to +receive/send packets. the driver does not require any other usb code +and should work on most any sa1100. + +host (SA1111) side: +- because the SA1111 usb host is not working yet I tested this driver +(usb-net-host.c) on a 2.2.14 based PC with the latest usb backport. +It has been fully converted to use URBs and worked well with my UHCI +based controller. + +TESTING: + +To test you need an assabet on the 'function' side, a PC on the 'host' +side and a USB A-B cable to connect them together. + +Boot a kernel on the assabet with "USB function and net-host support" +(CONFIG_SA1100_USB) turned on. This will define an interface named +"usbf". Once it's booted you can setup the interface with + + mount -t proc /proc /proc + /sbin/ifconfig usbf 1.1.1.2 + +I used a 2.2.14 kernel on a x86 PC for the host side. It has a built +in UHCI usb controller chip. I installed the latest USB backport from +http://www.linux-usb.org onto the 2.2.14 kernel sources and turned on +"USB net-host" (CONFIG_USB_NET_HOST) as a module. Load the module +"usb-net-host.o" and connect the USB cable to the assabet. Configure +the usb network interface with + + /sbin/ifconfig usb0 1.1.1.1 + +You should be able to "ping" the assabet now with + + ping -c 1 1.1.1.2 + +If the assabet is running inetd the usual network services such as +telnet and ftp should work. + +Oleg Drokin in 2.4.2-rmk1-np2 (08Mar01) added module config params for +read and write size to the usb-eth.c client to allow dynamic setting +of the DATA0/DATA1 packet size on the usb wire: + +usb_rsize - number of bytes in OUT packets from host to SA1100 +usb_wsize - number of bytes in IN packets from SA1100 to host + +This allows dynamic tuning for performance or to prevent overruning +the the host with data. + +4. Known Issues +''''''''''''''' +- We are fiddling with various ways to set the IMP register in +usb_send.c. A small percentage of the time, this value does not +"take." + +- I've started to bring back the /proc interface, but clients +of the sa-usb core currently don't have a directory or something +to put their stats into. + +- Only a useful subset of ep0 setup calls have been implemented. + + +5. Mysteries of the Universe +'''''''''''''''''''''''''''' +This driver has been hard to develop because the documentation +provided by Intel is incomplete, and the UDC itself seems to have a +variety of bugs. The errata for the part is particularly scary! This +section is an attempt to document some of the discoveries and +questions I have come across while working on this thing. + +pp 11-63 of the "Intel Strong ARM SA-1110 Microprocessor Advanced +Developer Information" give an ominous warning about how "due to +internal synchronization required by the UDC configuration registers, +it is possible for the procesor to write the UDC refisters and FIFOs +too fast." This has led to a variety of approaches that attempt to +bang on the hardware repeatedly and read it back until the write +"sticks." + +All of these approaches have been problematic. Currently some macros +in udc_ctl.h that Nicolas wrote are being used. My hardware guy told +me that writes would never be "lost" but stuck on some internal bus in +the UDC module and propagated to the rest of the circuit when the time +was right. Indeed this seemed to be the case, for example, it seems +impossible to reliably read back the interrupt mask register of the UDC when in +the interrupt service routine. Often times the state was not reflected +on a read until after pending interrupt sources were cleared. + +I was feeling prety good about this and was ripping out the looping +macros right and left until I came upon a situation where, while +receiving a continuous set of 1 character packets, ep1 (usb_receive.c) +could not clear receive packet complete (RPC). After much desperate +faliling about it turns out changing the UDC_flip() macro to bang like +crazy on the RPC bit did in fact clear it, and clear it +consistently. So go figure. + +Other items of interest: + +- Upon emerging from a reset, the UDC will clear the mask register except +for a mask on suspend. + +- USB 1.0 spec says maximum size of a DATA0/1 packet is 64 bytes, +which is what the character driver is using. However, the UDC can do +256 bytes and every host I've tried can handle it, even though they +are not required to. (Perhaps it is a problem when hubs are on the +line, but the SA UDC has other problems in a hub environment -- like +even getting the correct address -- per the errata). + +- Endpoint zero FIFOs: ARGHHH! Just leave those routines alone. +Believe me, I have tried every other variation you can think of. +Probably. + +- Sometimes I get a setup request of 0x80 from Windows hosts. I have +not determined if this is a read_fifo error (none is reported) or if +this is some undocumented secret Redmond handshake only known to +initiates of the inner-order. + +6. Test Program +''''''''''''''' +This is now in the /proc interface. (For good or ill, probably don't +actually need to dump all this stuff..) + +7. Errors and Notes on Intel's 1110 Documentation +''''''''''''''''''''''''''''''''''''''''''''''''' +These corrections apply to "Intel StrongARM SA-1110 Microprocessor, +Advanced Developer's Manual of December 1999" Some of these have been +corrected in later editions, some not. There have been several updates +to this document published through 2000. Always use the latest +available on http://developer.intel.com/design/strong/collateral.htm. + +pp 11-65 section 11.8.3.8 bit 2, reserved is now the resume interrupt +mask. SRM is now SUSIM on SA-1110, and masks only the suspend +interrupt. + +pp 11-67 section 11.8.6, Max IN register, end should be 9 _bytes_ +not 9 bits. + +pp 11-68 section 11.8.7.3, SST. This is set by the CPU _not_ the UDC. +And it looks like you don't get a SST if you FST yourself. + +pp 11-68 section 11.8.7.5, DE. This is set by the CPU _not_ the UDC. + +pp 11-73 section 11.8.9.7, UDCCS2 table, bit 2, Should be "valid only +when _TPC_ (not RPC) is set. + +pp 11-74 section 11.8.10, should end with a GET_DESCRIPTOR _or +similar_ command. (Like, for example, GET_CONFIGURATION). + + +8. Change History +''''''''''''''''' +Following are current chages 8Mar01 (released in 2.4.2-rmk1-np2?) + +- Resetting UDC when coming out of suspend helped enumeration get +going considerably. + +- Added support for client-supplied notify routine to be called +by the USB core when core reaches "configured" state. + +- Added error returns from interrupt reads and buffer flush ioctl +calls to usb-char. Added usb-char.h file for ioctl calls. + +- Fixed bug that kept usb-char transmitter from working the second +time the module was loaded. + +- Turned off a lot of the noise in /proc + +- Added specialty routines in ep0 to set and clear bits. + +- More enumeration fiddling. + +- There are horrible hacks to set max IN length in usb_send + that ARE GOING AWAY SOON! REALLY! + +*** Following changes 26Feb01 (released in 2.4.2-rmk1-np1) + +- usb-eth integration with generic usbnet from AC tree. + +- Creation of public interface for usb clients in sa1100_usb.h +and final separation into a "core" driver (usb_ctl.c, usb_ep0.c +usb_recv.c usb_send.c) and "client" services (usb-eth.c and +usb-char.c). Modularized. + +- Descriptor handling rewritten. Support for string descriptors +added. More bugs in ep0 fixed. More setup packets handled. + +- /proc interface in usb_ctl returning + +- removed client specific stuff from usbd_info_t and hid the +structure in usb_ctl. Removed RAM-backing of address and pktsize +in this structure. Now the descriptor values are gospel. + +- usb_dbg.h eliminated + +- Many bugs fixed in usb-char.c + +- Fiddled startup sequence so should start everytime. + +- Arch specific "soft connect" hook in usb_ctl.c + +- Bumped the interation count in write/set/clear macros +in usb_ctl.h up to 10000. This seemed to help various bit +setting in ep0 and usb_send.c. + +*** Following changes 10Feb01 release: + +- endpoint zero entirely rewitten + +- Various changes by Oleg to make Netlink work again after the + 2.4.1-rmk1-np1 release. + +- Resetting of new max packet length done after clearing TPC +in usb_send, per Nicolas Pitre. + + +*** Following changes 23Jan01 (came out in 2.4.1-rmk1-np1): + +- Moved host initiated SET/GET feature stall into endpoint code of +usb_send.c and usb_receive.c and removed stallep from usb_ctl.c +Opposite of a SET_FEATURE stall is a reset, so no code to unstall is +provided. + +- Added explicit USB state machine to usb_ctl so driver and device +state can be tracked closely and explicitly. Added hard-wired +notification routines in endpoints 1 and 2 so they can track device +state changes as required. State machine has notion of "zombie" state +the covers USB states NONATTACHED, ATTACHED and POWERED since these +are murky, and USB driver currently has no way to differentiate +between the two. + +- Reworked ISR in usb_ctl so reset has higher priority than any other +event. Stopped using sync macros to clear interrupt pending flags and set +mask registers since it appears mask register changes are not +always reflected on a mask register read until the pending flag is +cleared (yet other tests show they are always cleared +eventually). Toggle suspend/resume interrupt masks back and forth during +suspend and resume to debounce and keep UDC internal state machine in +sync per Intel documentation. + +- Flipped UDC flip, clear, write and set macros from do{}while to +while() loops. Theory is you might save a loop iteration if value +becomes valid immediately. Also, my hardware guy says writes are never +"lost", just pipelined and not executed immediately depending on +internal device conditions (like setting int masks when ints +pending), so moved write cycles in macros outside of loops. + +- Added #defines to SA-1110.h for suspend and resume interrupt mask +bits per Intel eratta. Submitted to ARM patch system (444/1). + +- Removed task queue and defered execution of configure() from +usb_ctl. + +- Removed usb_write_reg() from usb_ctl.c, and various cruft from +usb_ctl.h. + +- Added sa1100_usb_xmitter_avail() to usb_send.c. Makes implementing +poll() fileop easier. + +- Added sa1100_usb_send_reset() to usb_send.c. Makes implementing +transmitter timeout easier. + +- Added API to usb_ctl to set vendor and product ID + +- Changed BMATTR descriptor fron int to bulk, when not using netlink. (All +the docs say UDC does not support INT xfers -- though, at the protocol +level I don't see why not, since bulk and int are both just +IN-DATA-ACK. I figure netlink may rely on this, and not just a +continuous pending read from the host, but for "pure bulk" host +polling may not be generally correct.) + +- Removed unused rx_lock and tx_lock from usb_ctl + +- Converted everyone to SA-1100.h and nuked hardware defines in usb_ctl.h + +- Removed udc_init() in usb_ctl.c and folded functionality into udc_start(). + +- Clear force stall (FST) in udc_start and reset so UDC actually runs when +first turned on. + +- Emit NAK in receiver until ep1_start() for error (RPE) case too. + +- Remove enable/disable UDC from reset handler in udc_ctl. The UDC has +already been reset, so no need to do this again. + +- Explicitly set address to zero in ep0_reset() + +- Added "naking" boolean to usb_receiv.c. An attempt to solve a +hypothetical race condition where we are in the critical section +initiating a read from base-level code, a RPC happens, and start() +might clear the condition before the packet is handled by the ISR. diff -urN orig/Documentation/arm/SA1100/nanoEngine linux/Documentation/arm/SA1100/nanoEngine --- orig/Documentation/arm/SA1100/nanoEngine Mon Oct 1 23:10:00 2001 +++ linux/Documentation/arm/SA1100/nanoEngine Fri Jun 21 14:13:59 2002 @@ -7,5 +7,7 @@ (Ref: Stuart Adams ) Also visit Larry Doolittle's "Linux for the nanoEngine" site: -http://recycle.lbl.gov/~ldoolitt/bse/ - + http://recycle.lbl.gov/~ldoolitt/bse/ +This page includes translation code that builds the parameter +blocks for the Linux kernel, which must happen before each +kernel boot. diff -urN orig/Documentation/cpufreq linux/Documentation/cpufreq --- orig/Documentation/cpufreq Thu Jan 1 01:00:00 1970 +++ linux/Documentation/cpufreq Sun Jul 7 13:36:45 2002 @@ -0,0 +1,332 @@ + CPU frequency and voltage scaling code in the Linux(TM) kernel + + + L i n u x C P U F r e q + + + + + Dominik Brodowski + + + + Clock scaling allows you to change the clock speed of the CPUs on the + fly. This is a nice method to save battery power, because the lower + the clock speed, the less power the CPU consumes. + + + +Contents: +--------- +1. Supported architectures +2. User interface +2.1 Sample script for command line interface +3. CPUFreq core and interfaces +3.1 General information +3.2 CPUFreq notifiers +3.3 CPUFreq architecture drivers +4. Mailing list and Links + + + +1. Supported architectures +========================== + +Some architectures detect the lowest and highest possible speed +settings, while others rely on user information on this. For the +latter, a boot parameter is required, for the former, you can specify +one to set the limits between speed settings may occur. +The boot parameter has the following syntax: + + cpufreq=minspeed-maxspeed + +with both minspeed and maxspeed being given in kHz. To set the lower +limit to 59 MHz and the upper limit to 221 MHz, specify: + + cpufreq=59000-221000 + +Check the "Speed Limits Detection" information below on whether +the driver detects the lowest and highest allowed speed setting +automatically. + + +ARM Integrator: + SA 1100, SA1110 +-------------------------------- + Speed Limits Detection: On Integrators, the minimum speed is set + and the maximum speed has to be specified using the boot + parameter. On SA11x0s, the frequencies are fixed (59 - 287 MHz) + + +AMD Elan: + SC400, SC410 +-------------------------------- + Speed Limits Detection: Not implemented. You need to specify the + minimum and maximum frequency in the boot parameter (see above). + + +VIA Cyrix Longhaul: + VIA Samuel/CyrixIII, VIA Cyrix Samuel/C3, + VIA Cyrix Ezra, VIA Cyrix Ezra-T +-------------------------------- + Speed Limits Detection: working. No need for boot parameters. + NOTE: Support for certain processors is currently disabled, + waiting on updated docs from VIA. + + +Intel SpeedStep: + certain mobile Intel Pentium III (Coppermine), and all mobile + Intel Pentium III-M (Tulatin) and mobile Intel Pentium 4 P4-Ms. +-------------------------------- + Speed Limits Detection: working. No need for boot parameters. + NOTE: + 1.) mobile Intel Pentium III (Coppermine): + The SpeedStep interface may only be used on SpeedStep + capable processors. Unforunately, due to lack of documentation, + such detection is not yet possible on mobile Intel PIII + (Coppermine) processors. In order to activate SpeedStep on such a + processor, you have to remove one line manually in + linux/drivers/arch/i386/speedstep.c + + +P4 CPU Clock Modulation: + Intel Pentium 4 Xeon processors +-------------------------------- + Speed Limits Detection: Not implemented. You need to specify the + minimum and maximum frequency in the boot parameter (see above). + + + +2. User Interface +================= + +CPUFreq uses a "sysctl" interface which is located in + /proc/sys/cpu/0/ on UP (uniprocessor) kernels, or + /proc/sys/cpu/any/ on SMP (symmetric multiprocessoring) kernels. + + +In this directory, you will find three files of importance for +CPUFreq: speed-max, speed-min, and speed: + +speed shows the current CPU frequency in kHz, +speed-min the minimal supported CPU frequency, and +speed-max the maximal supported CPU frequency. + +Please note that you might have to specify these limits as a boot +parameter depending on the architecture (see above). + + +To change the CPU frequency, "echo" the desired CPU frequency (in kHz) +to speed. For example, to set the CPU speed to the lowest/highest +allowed frequency do: + +root@notebook:# cat /proc/sys/cpu/0/speed-min > /proc/sys/cpu/0/speed +root@notebook:# cat /proc/sys/cpu/0/speed-max > /proc/sys/cpu/0/speed + + +2.1 Sample script for command line interface +********************************************** + + +Michael Ossmann has written a small command line +interface for the infinitely lazy. + +#!/bin/bash +# +# /usr/local/bin/freq +# simple command line interface to cpufreq + +[ -n "$1" ] && case "$1" in + "min" ) + # set frequency to minimum + cat /proc/sys/cpu/0/speed-min >/proc/sys/cpu/0/speed + ;; + "max" ) + # set frequency to maximum + cat /proc/sys/cpu/0/speed-max >/proc/sys/cpu/0/speed + ;; + * ) + echo "Usage: $0 [min|max]" + echo " min: set frequency to minimum and display new frequency" + echo " max: set frequency to maximum and display new frequency" + echo " no options: display current frequency" + exit 1 + ;; +esac + +# display current frequency +cat /proc/sys/cpu/0/speed +exit 0 + + + +3. CPUFreq core and interfaces +=============================== + +3.1 General information +************************* + +The CPUFreq core code is located in linux/kernel/cpufreq.c. This +cpufreq code offers a standardized interface for the CPUFreq +architecture drivers (those pieces of code that do the actual +frequency transition), as well as to "notifiers". These are device +drivers or other part of the kernel that need to be informed of +frequency changes (like timing code) or even need to force certain +speed limits (like LCD drivers on ARM architecture). Aditionally, the +kernel "constant" loops_per_jiffy is updated on frequency changes +here. + + +3.2 CPUFreq notifiers +*********************** + +CPUFreq notifiers are kernel code that need to be called to either +a) define certain minimum or maximum speed settings, +b) be informed of frequency changes in advance of the transition, or +c) be informed of frequency changes directly after the transition. + +A standard kernel notifier interface is offered for this. See +linux/include/linux/notifier.h for details on notifiers. + + +Data and value passed to CPUFreq notifiers +------------------------------------------ +The second argument passed to any notifier is an unsigned int stating +the phase of the transition: +CPUFREQ_MINMAX during the process of determing a valid new CPU + frequency, +CPUFREQ_PRECHANGE right before the transition, and +CPUFREQ_POSTCHANGE right after the transition. + +The third argument, a void *pointer, points to a struct +cpufreq_freqs. This consists of four values: min, max, cur and new. + +min and max are the current speed limits. Please note: Never update +these values directly, use cpufreq_updateminmax(struct cpufreq_freqs +*freqs, unsigned int min, unsigned int max) instead. cur is the +current/old speed, and new is the new speed, but might only be valid +on CPUFREQ_PRECHANGE or CPUFREQ_POSTCHANGE. + +Each notifier gets called all three times on any transition: + +CPUFREQ_MINMAX +Here the notifier is supposed to update the min and max values to the +limits the protected device / kernel code needs. As stated above, +always use cpufreq_updateminmax for this. + +CPUFREQ_PRECHANGE +CPUFREQ_POSTCHANGE +Here the notifier is supposed to update all internal (e.g. device +driver) code which is dependend on the CPU frequency. + + +3.3 CPUFreq architecture drivers +********************************** + +CPUFreq architecture drivers are the pieces of kernel code that +actually perform CPU frequency transitions. These need to be +initialised seperately (seperate initcalls), and may be +modularized. They interact with the CPUFreq core in the following way: + + +cpufreq_register() +------------------ +cpufreq_register registers an arch driver to the CPUFreq core. Please +note that only one arch driver may be registered at any time, -EBUSY +is returned when an arch driver is already registered. The argument to +cpufreq_register, cpufreq_driver_t driver, is described later. + + +cpufreq_unregister() +-------------------- +cpufreq_unregister unregisters an arch driver, e.g. on module +unloading. Please note that there is no check done that this is called +from the driver which actually registered itself to the core, so +please only call this function when you are sure the arch driver got +registered correctly before. + + +struct cpufreq_driver +---------------- +On initialisation, the arch driver is supposed to pass the following +entries in struct cpufreq_driver cpufreq_driver: + +cpufreq_verify_t validate: This is a pointer to a function with the +following definition: + unsigned int validating_function (unsigned int kHz). +It is called right before a transition occurs. The proposed new +speed setting is passed as an argument in kHz; the validating code +should verify this is a valid speed setting which is currently +supported by the CPU. It shall return the closest valid CPU frequency +in kHz. + +cpufreq_setspeed_t setspeed: This is a pointer to a function with the +following definition: + void setspeed_function (unsigned int kHz). +This function shall perform the transition to the new CPU frequency +given as argument in kHz. Note that this argument is exactly the same +as the one returned by cpufreq_verify_t validate. + + +unsigned int freq.cur: The current CPU core frequency. Note that this +is a requirement while the next two entries are optional. + + +unsigned int freq.min (optional): The minimal CPU core frequency this +CPU supports. This value may be limited further by the +cpufreq_verify_t validate function, and so this value should be the +minimal core frequency allowed "theoretically" on this system in this +configuration. + + +unsigned int freq.max (optional): The maximum CPU core frequency this +CPU supports. This value may be limited further by the +cpufreq_verify_t validate function, and so this value should be the +maximum core frequency allowed "theoretically" on this system in this +configuration. + + +Some Requirements to CPUFreq architecture drivers +------------------------------------------------- +* Only call cpufreq_register() when the ability to switch CPU + frequencies is _verified_ or can't be missing +* cpufreq_unregister() may only be called if cpufreq_register() has + been successfully(!) called before +* All CPUs have to be set to the same speed whenever setspeed() is + called +* Be aware that there is currently no error management in the + setspeed() code in the CPUFreq core. So only call yourself a + cpufreq_driver if you are really a working cpufreq_driver! + + + +4. Mailing list and Links +************************** + + +Mailing List +------------ +There is a CPU frequency changing CVS commit and general list where +you can report bugs, problems or submit patches. To post a message, +send an email to cpufreq@www.linux.org.uk, to subscribe go to +http://www.linux.org.uk/mailman/listinfo/cpufreq. Previous post to the +mailing list are available to subscribers at +http://www.linux.org.uk/mailman/private/cpufreq/. + + +Links +----- +the FTP archives: +* ftp://ftp.linux.org.uk/pub/linux/cpufreq/ + +how to access the CVS repository: +* http://www.arm.linux.org.uk/cvs/ + +the CPUFreq Mailing list: +* http://www.linux.org.uk/mailman/listinfo/cpufreq + +Clock and voltage scaling for the SA-1100: +* http://www.lart.tudelft.nl/projects/scaling + +CPUFreq project homepage +* http://www.brodo.de/cpufreq/ diff -urN orig/Documentation/cpufreq-old linux/Documentation/cpufreq-old --- orig/Documentation/cpufreq-old Thu Jan 1 01:00:00 1970 +++ linux/Documentation/cpufreq-old Sun Jul 7 13:36:45 2002 @@ -0,0 +1,332 @@ + CPU frequency and voltage scaling code in the Linux(TM) kernel + + + L i n u x C P U F r e q + + + + + Dominik Brodowski + + + + Clock scaling allows you to change the clock speed of the CPUs on the + fly. This is a nice method to save battery power, because the lower + the clock speed, the less power the CPU consumes. + + + +Contents: +--------- +1. Supported architectures +2. User interface +2.1 Sample script for command line interface +3. CPUFreq core and interfaces +3.1 General information +3.2 CPUFreq notifiers +3.3 CPUFreq architecture drivers +4. Mailing list and Links + + + +1. Supported architectures +========================== + +Some architectures detect the lowest and highest possible speed +settings, while others rely on user information on this. For the +latter, a boot parameter is required, for the former, you can specify +one to set the limits between speed settings may occur. +The boot parameter has the following syntax: + + cpufreq=minspeed-maxspeed + +with both minspeed and maxspeed being given in kHz. To set the lower +limit to 59 MHz and the upper limit to 221 MHz, specify: + + cpufreq=59000-221000 + +Check the "Speed Limits Detection" information below on whether +the driver detects the lowest and highest allowed speed setting +automatically. + + +ARM Integrator: + SA 1100, SA1110 +-------------------------------- + Speed Limits Detection: On Integrators, the minimum speed is set + and the maximum speed has to be specified using the boot + parameter. On SA11x0s, the frequencies are fixed (59 - 287 MHz) + + +AMD Elan: + SC400, SC410 +-------------------------------- + Speed Limits Detection: Not implemented. You need to specify the + minimum and maximum frequency in the boot parameter (see above). + + +VIA Cyrix Longhaul: + VIA Samuel/CyrixIII, VIA Cyrix Samuel/C3, + VIA Cyrix Ezra, VIA Cyrix Ezra-T +-------------------------------- + Speed Limits Detection: working. No need for boot parameters. + NOTE: Support for certain processors is currently disabled, + waiting on updated docs from VIA. + + +Intel SpeedStep: + certain mobile Intel Pentium III (Coppermine), and all mobile + Intel Pentium III-M (Tulatin) and mobile Intel Pentium 4 P4-Ms. +-------------------------------- + Speed Limits Detection: working. No need for boot parameters. + NOTE: + 1.) mobile Intel Pentium III (Coppermine): + The SpeedStep interface may only be used on SpeedStep + capable processors. Unforunately, due to lack of documentation, + such detection is not yet possible on mobile Intel PIII + (Coppermine) processors. In order to activate SpeedStep on such a + processor, you have to remove one line manually in + linux/drivers/arch/i386/speedstep.c + + +P4 CPU Clock Modulation: + Intel Pentium 4 Xeon processors +-------------------------------- + Speed Limits Detection: Not implemented. You need to specify the + minimum and maximum frequency in the boot parameter (see above). + + + +2. User Interface +================= + +CPUFreq uses a "sysctl" interface which is located in + /proc/sys/cpu/0/ on UP (uniprocessor) kernels, or + /proc/sys/cpu/any/ on SMP (symmetric multiprocessoring) kernels. + + +In this directory, you will find three files of importance for +CPUFreq: speed-max, speed-min, and speed: + +speed shows the current CPU frequency in kHz, +speed-min the minimal supported CPU frequency, and +speed-max the maximal supported CPU frequency. + +Please note that you might have to specify these limits as a boot +parameter depending on the architecture (see above). + + +To change the CPU frequency, "echo" the desired CPU frequency (in kHz) +to speed. For example, to set the CPU speed to the lowest/highest +allowed frequency do: + +root@notebook:# cat /proc/sys/cpu/0/speed-min > /proc/sys/cpu/0/speed +root@notebook:# cat /proc/sys/cpu/0/speed-max > /proc/sys/cpu/0/speed + + +2.1 Sample script for command line interface +********************************************** + + +Michael Ossmann has written a small command line +interface for the infinitely lazy. + +#!/bin/bash +# +# /usr/local/bin/freq +# simple command line interface to cpufreq + +[ -n "$1" ] && case "$1" in + "min" ) + # set frequency to minimum + cat /proc/sys/cpu/0/speed-min >/proc/sys/cpu/0/speed + ;; + "max" ) + # set frequency to maximum + cat /proc/sys/cpu/0/speed-max >/proc/sys/cpu/0/speed + ;; + * ) + echo "Usage: $0 [min|max]" + echo " min: set frequency to minimum and display new frequency" + echo " max: set frequency to maximum and display new frequency" + echo " no options: display current frequency" + exit 1 + ;; +esac + +# display current frequency +cat /proc/sys/cpu/0/speed +exit 0 + + + +3. CPUFreq core and interfaces +=============================== + +3.1 General information +************************* + +The CPUFreq core code is located in linux/kernel/cpufreq.c. This +cpufreq code offers a standardized interface for the CPUFreq +architecture drivers (those pieces of code that do the actual +frequency transition), as well as to "notifiers". These are device +drivers or other part of the kernel that need to be informed of +frequency changes (like timing code) or even need to force certain +speed limits (like LCD drivers on ARM architecture). Aditionally, the +kernel "constant" loops_per_jiffy is updated on frequency changes +here. + + +3.2 CPUFreq notifiers +*********************** + +CPUFreq notifiers are kernel code that need to be called to either +a) define certain minimum or maximum speed settings, +b) be informed of frequency changes in advance of the transition, or +c) be informed of frequency changes directly after the transition. + +A standard kernel notifier interface is offered for this. See +linux/include/linux/notifier.h for details on notifiers. + + +Data and value passed to CPUFreq notifiers +------------------------------------------ +The second argument passed to any notifier is an unsigned int stating +the phase of the transition: +CPUFREQ_MINMAX during the process of determing a valid new CPU + frequency, +CPUFREQ_PRECHANGE right before the transition, and +CPUFREQ_POSTCHANGE right after the transition. + +The third argument, a void *pointer, points to a struct +cpufreq_freqs. This consists of four values: min, max, cur and new. + +min and max are the current speed limits. Please note: Never update +these values directly, use cpufreq_updateminmax(struct cpufreq_freqs +*freqs, unsigned int min, unsigned int max) instead. cur is the +current/old speed, and new is the new speed, but might only be valid +on CPUFREQ_PRECHANGE or CPUFREQ_POSTCHANGE. + +Each notifier gets called all three times on any transition: + +CPUFREQ_MINMAX +Here the notifier is supposed to update the min and max values to the +limits the protected device / kernel code needs. As stated above, +always use cpufreq_updateminmax for this. + +CPUFREQ_PRECHANGE +CPUFREQ_POSTCHANGE +Here the notifier is supposed to update all internal (e.g. device +driver) code which is dependend on the CPU frequency. + + +3.3 CPUFreq architecture drivers +********************************** + +CPUFreq architecture drivers are the pieces of kernel code that +actually perform CPU frequency transitions. These need to be +initialised seperately (seperate initcalls), and may be +modularized. They interact with the CPUFreq core in the following way: + + +cpufreq_register() +------------------ +cpufreq_register registers an arch driver to the CPUFreq core. Please +note that only one arch driver may be registered at any time, -EBUSY +is returned when an arch driver is already registered. The argument to +cpufreq_register, cpufreq_driver_t driver, is described later. + + +cpufreq_unregister() +-------------------- +cpufreq_unregister unregisters an arch driver, e.g. on module +unloading. Please note that there is no check done that this is called +from the driver which actually registered itself to the core, so +please only call this function when you are sure the arch driver got +registered correctly before. + + +struct cpufreq_driver +---------------- +On initialisation, the arch driver is supposed to pass the following +entries in struct cpufreq_driver cpufreq_driver: + +cpufreq_verify_t validate: This is a pointer to a function with the +following definition: + unsigned int validating_function (unsigned int kHz). +It is called right before a transition occurs. The proposed new +speed setting is passed as an argument in kHz; the validating code +should verify this is a valid speed setting which is currently +supported by the CPU. It shall return the closest valid CPU frequency +in kHz. + +cpufreq_setspeed_t setspeed: This is a pointer to a function with the +following definition: + void setspeed_function (unsigned int kHz). +This function shall perform the transition to the new CPU frequency +given as argument in kHz. Note that this argument is exactly the same +as the one returned by cpufreq_verify_t validate. + + +unsigned int freq.cur: The current CPU core frequency. Note that this +is a requirement while the next two entries are optional. + + +unsigned int freq.min (optional): The minimal CPU core frequency this +CPU supports. This value may be limited further by the +cpufreq_verify_t validate function, and so this value should be the +minimal core frequency allowed "theoretically" on this system in this +configuration. + + +unsigned int freq.max (optional): The maximum CPU core frequency this +CPU supports. This value may be limited further by the +cpufreq_verify_t validate function, and so this value should be the +maximum core frequency allowed "theoretically" on this system in this +configuration. + + +Some Requirements to CPUFreq architecture drivers +------------------------------------------------- +* Only call cpufreq_register() when the ability to switch CPU + frequencies is _verified_ or can't be missing +* cpufreq_unregister() may only be called if cpufreq_register() has + been successfully(!) called before +* All CPUs have to be set to the same speed whenever setspeed() is + called +* Be aware that there is currently no error management in the + setspeed() code in the CPUFreq core. So only call yourself a + cpufreq_driver if you are really a working cpufreq_driver! + + + +4. Mailing list and Links +************************** + + +Mailing List +------------ +There is a CPU frequency changing CVS commit and general list where +you can report bugs, problems or submit patches. To post a message, +send an email to cpufreq@www.linux.org.uk, to subscribe go to +http://www.linux.org.uk/mailman/listinfo/cpufreq. Previous post to the +mailing list are available to subscribers at +http://www.linux.org.uk/mailman/private/cpufreq/. + + +Links +----- +the FTP archives: +* ftp://ftp.linux.org.uk/pub/linux/cpufreq/ + +how to access the CVS repository: +* http://www.arm.linux.org.uk/cvs/ + +the CPUFreq Mailing list: +* http://www.linux.org.uk/mailman/listinfo/cpufreq + +Clock and voltage scaling for the SA-1100: +* http://www.lart.tudelft.nl/projects/scaling + +CPUFreq project homepage +* http://www.brodo.de/cpufreq/ diff -urN orig/Documentation/l3/structure linux/Documentation/l3/structure --- orig/Documentation/l3/structure Thu Jan 1 01:00:00 1970 +++ linux/Documentation/l3/structure Sun Aug 5 18:40:00 2001 @@ -0,0 +1,36 @@ +L3 Bus Driver +------------- + +The structure of the driver is as follows: + + +----------+ +----------+ +----------+ + | client 1 | | client 2 | | client 3 | + +-----^----+ +----^-----+ +----^-----+ + | | | + +-----v--------------v---------------v-----+ + | | + +-----^-------+ +-------^-----+ + | | core | | + +-----v----+ | | +----v-----+ + | device | | | | device | + | driver 1 | | | | driver 2 | + +-----^----+ | | +----^-----+ + | | services | | + +-----v-------+ +-------v-----+ + | | + +-----------------^----^-------------------+ + | | + | +-v---------+ + | | algorithm | + | | driver | + | +-v---------+ + | | + +-v----v-+ + | bus | + | driver | + +--------+ + +Clients talk to the core to attach device drivers and bus adapters, and +to instruct device drivers to perform actions. Device drivers then talk +to the core to perform L3 bus transactions via the algorithm driver and +ultimately bus driver. diff -urN orig/Documentation/serial/driver linux/Documentation/serial/driver --- orig/Documentation/serial/driver Thu Jan 1 01:00:00 1970 +++ linux/Documentation/serial/driver Sat Nov 24 23:24:47 2001 @@ -0,0 +1,208 @@ + + Low Level Serial API + -------------------- + + + $Id: driver,v 1.3 2001/11/24 23:24:47 rmk Exp $ + + +This document is meant as a brief overview of some aspects of the new serial +driver. It is not complete, any questions you have should be directed to + + +The reference implementation is contained within serial_amba.c. + + + +Low Level Serial Hardware Driver +-------------------------------- + +The low level serial hardware driver is responsible for supplying port +information (defined by uart_port) and a set of control methods (defined +by uart_ops) to the core serial driver. The low level driver is also +responsible for handling interrupts for the port, and providing any +console support. + + +Console Support +--------------- + +The serial core provides a few helper functions. This includes identifing +the correct port structure (via uart_get_console) and decoding command line +arguments (uart_parse_options). + + +Locking +------- + +Generally, all locking is done by the core driver, except for the interrupt +functions. It is the responsibility of the low level hardware driver to +perform the necessary locking there using info->lock. (since it is running +in an interrupt, you only need to use spin_lock() and spin_unlock() from +the interrupt handler). + + +uart_ops +-------- + +The uart_ops structure is the main interface between serial_core and the +hardware specific driver. It contains all the methods to control the +hardware. + + tx_empty(port) + This function tests whether the transmitter fifo and shifter + for the port described by 'port' is empty. If it is empty, + this function should return TIOCSER_TEMT, otherwise return 0. + If the port does not support this operation, then it should + return TIOCSER_TEMT. + + set_mctrl(port, mctrl) + This function sets the modem control lines for port described + by 'port' to the state described by mctrl. The relevant bits + of mctrl are: + - TIOCM_RTS RTS signal. + - TIOCM_DTR DTR signal. + - TIOCM_OUT1 OUT1 signal. + - TIOCM_OUT2 OUT2 signal. + If the appropriate bit is set, the signal should be driven + active. If the bit is clear, the signal should be driven + inactive. + + get_mctrl(port) + Returns the current state of modem control inputs. The state + of the outputs should not be returned, since the core keeps + track of their state. The state information should include: + - TIOCM_DCD state of DCD signal + - TIOCM_CTS state of CTS signal + - TIOCM_DSR state of DSR signal + - TIOCM_RI state of RI signal + The bit is set if the signal is currently driven active. If + the port does not support CTS, DCD or DSR, the driver should + indicate that the signal is permanently active. If RI is + not available, the signal should not be indicated as active. + + stop_tx(port,from_tty) + Stop transmitting characters. This might be due to the CTS + line becoming inactive or the tty layer indicating we want + to stop transmission. + + start_tx(port,nonempty,from_tty) + start transmitting characters. (incidentally, nonempty will + always be nonzero, and shouldn't be used - it will be dropped). + + stop_rx(port) + Stop receiving characters; the port is in the process of + being closed. + + enable_ms(port) + Enable the modem status interrupts. + + break_ctl(port,ctl) + Control the transmission of a break signal. If ctl is + nonzero, the break signal should be transmitted. The signal + should be terminated when another call is made with a zero + ctl. + + startup(port,info) + Grab any interrupt resources and initialise any low level driver + state. Enable the port for reception. It should not activate + RTS nor DTR; this will be done via a separate call to set_mctrl. + + shutdown(port,info) + Disable the port, disable any break condition that may be in + effect, and free any interrupt resources. It should not disable + RTS nor DTR; this will have already been done via a separate + call to set_mctrl. + + change_speed(port,cflag,iflag,quot) + Change the port parameters, including word length, parity, stop + bits. Update read_status_mask and ignore_status_mask to indicate + the types of events we are interested in receiving. Relevant + cflag bits are: + CSIZE - word size + CSTOPB - 2 stop bits + PARENB - parity enable + PARODD - odd parity (when PARENB is in force) + CREAD - enable reception of characters (if not set, + still receive characters from the port, but + throw them away. + CRTSCTS - if set, enable CTS status change reporting + CLOCAL - if not set, enable modem status change + reporting. + Relevant iflag bits are: + INPCK - enable frame and parity error events to be + passed to the TTY layer. + BRKINT + PARMRK - both of these enable break events to be + passed to the TTY layer. + + IGNPAR - ignore parity and framing errors + IGNBRK - ignore break errors, If IGNPAR is also + set, ignore overrun errors as well. + The interaction of the iflag bits is as follows (parity error + given as an example): + Parity error INPCK IGNPAR + None n/a n/a character received + Yes n/a 0 character discarded + Yes 0 1 character received, marked as + TTY_NORMAL + Yes 1 1 character received, marked as + TTY_PARITY + + pm(port,state,oldstate) + perform any power management related activities on the specified + port. state indicates the new state (defined by ACPI D0-D3), + oldstate indicates the previous state. Essentially, D0 means + fully on, D3 means powered down. + + This function should not be used to grab any resources. + + type(port) + Return a pointer to a string constant describing the specified + port, or return NULL, in which case the string 'unknown' is + substituted. + + release_port(port) + Release any memory and IO region resources currently in use by + the port. + + request_port(port) + Request any memory and IO region resources required by the port. + If any fail, no resources should be registered when this function + returns, and it should return -EBUSY on failure. + + config_port(port,type) + Perform any autoconfiguration steps required for the port. `type` + contains a bit mask of the required configuration. UART_CONFIG_TYPE + indicates that the port requires detection and identification. + port->type should be set to the type found, or PORT_UNKNOWN if + no port was detected. + + UART_CONFIG_IRQ indicates autoconfiguration of the interrupt signal, + which should be probed using standard kernel autoprobing techniques. + This is not necessary on platforms where ports have interrupts + internally hard wired (eg, system on a chip implementations). + + verify_port(port,serinfo) + Verify the new serial port information contained within serinfo is + suitable for this port type. + + ioctl(port,cmd,arg) + Perform any port specific IOCTLs. IOCTL commands must be defined + using the standard numbering system found in + + +Other notes +----------- + +It is intended some day to drop the 'unused' entries from uart_port, and +allow low level drivers to register their own individual uart_port's with +the core. This will allow drivers to use uart_port as a pointer to a +structure containing both the uart_port entry with their own extensions, +thus: + + struct my_port { + struct uart_port port; + int my_stuff; + }; + diff -urN orig/MAINTAINERS linux/MAINTAINERS --- orig/MAINTAINERS Mon Aug 5 13:29:41 2002 +++ linux/MAINTAINERS Mon Aug 5 23:25:12 2002 @@ -191,6 +191,13 @@ L: linux-scsi@vger.kernel.org S: Maintained +ALTERA EPXA1/EPXA10 DEVELOPMENT BOARD PORT +P: Clive Davies +M: cdavies@altera.com +L: linux-arm-kernel@lists.arm.linux.org.uk +W: http://www.arm.linux.org.uk/ +S: Maintained + APM DRIVER P: Stephen Rothwell M: sfr@canb.auug.org.au @@ -209,6 +216,13 @@ M: linux@treblig.org S: Maintained +ARM/PT DIGITAL BOARD PORT +P: Stefan Eletzhofer +M: stefan.eletzhofer@eletztrick.de +L: linux-arm-kernel@lists.arm.linux.org.uk +W: http://www.arm.linux.org.uk/ +S: Maintained + ARM/SHARK MACHINE SUPPORT P: Alexander Schulz M: alex@shark-linux.de @@ -1376,7 +1390,7 @@ SA1100 SUPPORT P: Nicolas Pitre M: nico@cam.org -L: sa1100-linux@pa.dec.com +L: linux-arm@lists.arm.linux.org.uk S: Maintained SBPCD CDROM DRIVER diff -urN orig/Makefile linux/Makefile --- orig/Makefile Mon Aug 5 13:29:41 2002 +++ linux/Makefile Wed Mar 19 22:20:01 2003 @@ -1,7 +1,7 @@ VERSION = 2 PATCHLEVEL = 4 SUBLEVEL = 19 -EXTRAVERSION = +EXTRAVERSION = -rmk7 KERNELRELEASE=$(VERSION).$(PATCHLEVEL).$(SUBLEVEL)$(EXTRAVERSION) @@ -134,7 +134,10 @@ DRIVERS-$(CONFIG_ACPI) += drivers/acpi/acpi.o DRIVERS-$(CONFIG_PARPORT) += drivers/parport/driver.o -DRIVERS-y += drivers/char/char.o \ +DRIVERS-$(CONFIG_I2C) += drivers/i2c/i2c.o +DRIVERS-$(CONFIG_L3) += drivers/l3/l3.o +DRIVERS-y += drivers/serial/serial.o \ + drivers/char/char.o \ drivers/block/block.o \ drivers/misc/misc.o \ drivers/net/net.o \ @@ -159,6 +162,7 @@ DRIVERS-y += drivers/cdrom/driver.o endif +DRIVERS-$(CONFIG_SSI) += drivers/ssi/ssi.o DRIVERS-$(CONFIG_SOUND) += drivers/sound/sounddrivers.o DRIVERS-$(CONFIG_PCI) += drivers/pci/driver.o DRIVERS-$(CONFIG_MTD) += drivers/mtd/mtdlink.o @@ -182,12 +186,12 @@ DRIVERS-$(CONFIG_INPUT) += drivers/input/inputdrv.o DRIVERS-$(CONFIG_I2O) += drivers/message/i2o/i2o.o DRIVERS-$(CONFIG_IRDA) += drivers/net/irda/irda.o -DRIVERS-$(CONFIG_I2C) += drivers/i2c/i2c.o DRIVERS-$(CONFIG_PHONE) += drivers/telephony/telephony.o DRIVERS-$(CONFIG_MD) += drivers/md/mddev.o DRIVERS-$(CONFIG_BLUEZ) += drivers/bluetooth/bluetooth.o DRIVERS-$(CONFIG_HOTPLUG_PCI) += drivers/hotplug/vmlinux-obj.o DRIVERS-$(CONFIG_ISDN_BOOL) += drivers/isdn/vmlinux-obj.o +DRIVERS-$(CONFIG_PLD) += drivers/pld/pld.o DRIVERS := $(DRIVERS-y) @@ -264,11 +268,6 @@ export CPPFLAGS CFLAGS CFLAGS_KERNEL AFLAGS AFLAGS_KERNEL export NETWORKS DRIVERS LIBS HEAD LDFLAGS LINKFLAGS MAKEBOOT ASFLAGS - -.S.s: - $(CPP) $(AFLAGS) $(AFLAGS_KERNEL) -traditional -o $*.s $< -.S.o: - $(CC) $(AFLAGS) $(AFLAGS_KERNEL) -traditional -c -o $*.o $< Version: dummy @rm -f include/linux/compile.h diff -urN orig/Rules.make linux/Rules.make --- orig/Rules.make Mon Aug 5 13:29:42 2002 +++ linux/Rules.make Mon Aug 5 13:54:24 2002 @@ -51,15 +51,15 @@ # %.s: %.c - $(CC) $(CFLAGS) $(EXTRA_CFLAGS_nostdinc) -DKBUILD_BASENAME=$(subst $(comma),_,$(subst -,_,$(*F))) $(CFLAGS_$@) -S $< -o $@ + $(CC) $(CFLAGS) $(EXTRA_CFLAGS_nostdinc) -DKBUILD_BASENAME=$(subst $(comma),_,$(subst -,_,$(*F))) $(CFLAGS_$(*F)) $(CFLAGS_$@) -S $< -o $@ %.i: %.c - $(CPP) $(CFLAGS) $(EXTRA_CFLAGS_nostdinc) -DKBUILD_BASENAME=$(subst $(comma),_,$(subst -,_,$(*F))) $(CFLAGS_$@) $< > $@ + $(CPP) $(CFLAGS) $(EXTRA_CFLAGS_nostdinc) -DKBUILD_BASENAME=$(subst $(comma),_,$(subst -,_,$(*F))) $(CFLAGS_$(*F)) $(CFLAGS_$@) $< > $@ %.o: %.c - $(CC) $(CFLAGS) $(EXTRA_CFLAGS_nostdinc) -DKBUILD_BASENAME=$(subst $(comma),_,$(subst -,_,$(*F))) $(CFLAGS_$@) -c -o $@ $< + $(CC) $(CFLAGS) $(EXTRA_CFLAGS_nostdinc) -DKBUILD_BASENAME=$(subst $(comma),_,$(subst -,_,$(*F))) $(CFLAGS_$(*F)) $(CFLAGS_$@) -c -o $@ $< @ ( \ - echo 'ifeq ($(strip $(subst $(comma),:,$(CFLAGS) $(EXTRA_CFLAGS_nostdinc) $(CFLAGS_$@))),$$(strip $$(subst $$(comma),:,$$(CFLAGS) $$(EXTRA_CFLAGS_nostdinc) $$(CFLAGS_$@))))' ; \ + echo 'ifeq ($(strip $(subst $(comma),:,$(CFLAGS) $(EXTRA_CFLAGS_nostdinc) $(CFLAGS_$(*F)) $(CFLAGS_$@))),$$(strip $$(subst $$(comma),:,$$(CFLAGS) $$(EXTRA_CFLAGS_nostdinc) $$(CFLAGS_$(*F)) $$(CFLAGS_$@))))' ; \ echo 'FILES_FLAGS_UP_TO_DATE += $@' ; \ echo 'endif' \ ) > $(dir $@)/.$(notdir $@).flags @@ -272,7 +272,8 @@ endif # CONFIG_MODVERSIONS ifneq "$(strip $(export-objs))" "" -$(export-objs): $(export-objs:.o=.c) $(TOPDIR)/include/linux/modversions.h +$(export-objs): $(TOPDIR)/include/linux/modversions.h +$(export-objs): %.o: %.c $(CC) $(CFLAGS) $(EXTRA_CFLAGS_nostdinc) -DKBUILD_BASENAME=$(subst $(comma),_,$(subst -,_,$(*F))) $(CFLAGS_$@) -DEXPORT_SYMTAB -c $(@:.o=.c) @ ( \ echo 'ifeq ($(strip $(subst $(comma),:,$(CFLAGS) $(EXTRA_CFLAGS_nostdinc) $(CFLAGS_$@) -DEXPORT_SYMTAB)),$$(strip $$(subst $$(comma),:,$$(CFLAGS) $$(EXTRA_CFLAGS_nostdinc) $$(CFLAGS_$@) -DEXPORT_SYMTAB)))' ; \ diff -urN orig/arch/alpha/config.in linux/arch/alpha/config.in --- orig/arch/alpha/config.in Mon Aug 5 13:29:42 2002 +++ linux/arch/alpha/config.in Mon Aug 5 13:41:15 2002 @@ -7,6 +7,7 @@ define_bool CONFIG_UID16 n define_bool CONFIG_RWSEM_GENERIC_SPINLOCK n define_bool CONFIG_RWSEM_XCHGADD_ALGORITHM y +define_bool CONFIG_GENERIC_ISA_DMA y mainmenu_name "Kernel configuration of Linux for Alpha machines" diff -urN orig/arch/arm/Makefile linux/arch/arm/Makefile --- orig/arch/arm/Makefile Mon Aug 5 13:29:42 2002 +++ linux/arch/arm/Makefile Wed Jan 22 23:46:12 2003 @@ -8,8 +8,15 @@ # Copyright (C) 1995-2001 by Russell King LINKFLAGS :=-p -X -T arch/arm/vmlinux.lds +OBJCOPYFLAGS :=-O binary -R .note -R .comment -S GZFLAGS :=-9 -CFLAGS +=-fno-common -pipe +CFLAGS +=-Uarm -fno-common -pipe + +ifeq ($(CONFIG_FRAME_POINTER),y) +CFLAGS :=$(CFLAGS:-fomit-frame-pointer=-mapcs -mno-sched-prolog) +endif + +CFLAGS :=$(CFLAGS:-O2=-Os) ifeq ($(CONFIG_DEBUG_INFO),y) CFLAGS +=-g @@ -24,10 +31,13 @@ apcs-$(CONFIG_CPU_26) :=-mapcs-26 -mcpu=arm3 -Os # This selects which instruction set is used. +# Note that GCC is lame - it doesn't numerically define an +# architecture version macro, but instead defines a whole +# series of macros. arch-y := -arch-$(CONFIG_CPU_32v3) :=-march=armv3 -arch-$(CONFIG_CPU_32v4) :=-march=armv4 -arch-$(CONFIG_CPU_32v5) :=-march=armv5 +arch-$(CONFIG_CPU_32v3) :=-D__LINUX_ARM_ARCH__=3 -march=armv3 +arch-$(CONFIG_CPU_32v4) :=-D__LINUX_ARM_ARCH__=4 -march=armv4 +arch-$(CONFIG_CPU_32v5) :=-D__LINUX_ARM_ARCH__=5 -march=armv5 # This selects how we optimise for the processor. tune-y := @@ -40,12 +50,12 @@ tune-$(CONFIG_CPU_SA110) :=-mtune=strongarm110 tune-$(CONFIG_CPU_SA1100) :=-mtune=strongarm1100 -CFLAGS_BOOT :=$(apcs-y) $(arch-y) $(tune-y) -mshort-load-bytes -msoft-float -CFLAGS +=$(apcs-y) $(arch-y) $(tune-y) -mshort-load-bytes -msoft-float +CFLAGS_BOOT :=$(apcs-y) $(arch-y) $(tune-y) -mshort-load-bytes -msoft-float -Uarm +CFLAGS +=$(apcs-y) $(arch-y) $(tune-y) -mshort-load-bytes -msoft-float -Uarm AFLAGS +=$(apcs-y) $(arch-y) -mno-fpu -msoft-float ifeq ($(CONFIG_CPU_26),y) -PROCESSOR = armo +PROCESSOR := armo ifeq ($(CONFIG_ROM_KERNEL),y) DATAADDR = 0x02080000 TEXTADDR = 0x03800000 @@ -81,6 +91,10 @@ INCDIR = cl7500 endif +ifeq ($(CONFIG_ARCH_RISCSTATION),y) +MACHINE = riscstation +endif + ifeq ($(CONFIG_FOOTBRIDGE),y) MACHINE = footbridge INCDIR = ebsa285 @@ -121,12 +135,18 @@ MACHINE = integrator endif + +ifeq ($(CONFIG_ARCH_AT91RM9200DK),y) +TEXTADDR = 0xc0008000 +MACHINE = at91rm9200dk +endif + ifeq ($(CONFIG_ARCH_MX1ADS),y) MACHINE = mx1ads endif ifeq ($(CONFIG_ARCH_CAMELOT),y) -MACHINE = epxa10db +MACHINE = epxa endif ifeq ($(CONFIG_ARCH_CLPS711X),y) @@ -135,19 +155,23 @@ endif ifeq ($(CONFIG_ARCH_FORTUNET),y) -TEXTADDR = 0xc0008000 +TEXTADDR = 0xc0208000 endif ifeq ($(CONFIG_ARCH_ANAKIN),y) MACHINE = anakin endif -export MACHINE PROCESSOR TEXTADDR GZFLAGS CFLAGS_BOOT +ifeq ($(CONFIG_ARCH_OMAHA),y) +MACHINE = omaha +endif + +export MACHINE PROCESSOR TEXTADDR GZFLAGS CFLAGS_BOOT OBJCOPYFLAGS # Only set INCDIR if its not already defined above # Grr, ?= doesn't work as all the other assignment operators do. Make bug? ifeq ($(origin INCDIR), undefined) -INCDIR := $(MACHINE) +INCDIR :=$(MACHINE) endif ifeq ($(origin DATAADDR), undefined) @@ -161,32 +185,37 @@ CORE_FILES := $(MACHDIR)/$(MACHINE).o $(CORE_FILES) endif -HEAD := arch/arm/kernel/head-$(PROCESSOR).o \ - arch/arm/kernel/init_task.o -SUBDIRS += arch/arm/kernel arch/arm/mm arch/arm/lib arch/arm/nwfpe -CORE_FILES := arch/arm/kernel/kernel.o arch/arm/mm/mm.o $(CORE_FILES) -LIBS := arch/arm/lib/lib.a $(LIBS) +HEAD :=arch/arm/kernel/head-$(PROCESSOR).o \ + arch/arm/kernel/init_task.o +SUBDIRS +=arch/arm/kernel arch/arm/mm arch/arm/lib arch/arm/nwfpe +CORE_FILES :=arch/arm/kernel/kernel.o arch/arm/mm/mm.o $(CORE_FILES) +LIBS :=arch/arm/lib/lib.a $(LIBS) ifeq ($(CONFIG_FPE_NWFPE),y) -LIBS := arch/arm/nwfpe/math-emu.o $(LIBS) +LIBS :=arch/arm/nwfpe/math-emu.o $(LIBS) endif # Only include fastfpe if it is part of the kernel tree. -FASTFPE := arch/arm/fastfpe +FASTFPE :=arch/arm/fastfpe ifeq ($(FASTFPE),$(wildcard $(FASTFPE))) -SUBDIRS += $(FASTFPE) +SUBDIRS +=$(FASTFPE) ifeq ($(CONFIG_FPE_FASTFPE),y) -LIBS := arch/arm/fastfpe/fast-math-emu.o $(LIBS) +LIBS :=arch/arm/fastfpe/fast-math-emu.o $(LIBS) endif endif ifeq ($(findstring y,$(CONFIG_ARCH_CLPS7500) $(CONFIG_ARCH_L7200)),y) -SUBDIRS += drivers/acorn/char -DRIVERS += drivers/acorn/char/acorn-char.o +SUBDIRS +=drivers/acorn/char +DRIVERS +=drivers/acorn/char/acorn-char.o +endif + +ifeq ($(CONFIG_ARCH_RISCSTATION),y) +SUBDIRS +=drivers/acorn/char +DRIVERS +=drivers/acorn/char/acorn-char.o endif -MAKEBOOT = $(MAKE) -C arch/$(ARCH)/boot -MAKETOOLS = $(MAKE) -C arch/$(ARCH)/tools +MAKEBOOT =$(MAKE) -C arch/$(ARCH)/boot +MAKETOOLS =$(MAKE) -C arch/$(ARCH)/tools # The following is a hack to get 'constants.h' up # to date before starting compilation @@ -194,15 +223,44 @@ $(patsubst %,_dir_%, $(SUBDIRS)): maketools $(patsubst %,_modsubdir_%,$(MOD_DIRS)): maketools -symlinks: archsymlinks +symlinks: include/asm-arm/.arch include/asm-arm/.proc -archsymlinks: - $(RM) include/asm-arm/arch include/asm-arm/proc - (cd include/asm-arm; ln -sf arch-$(INCDIR) arch; ln -sf proc-$(PROCESSOR) proc) +# Update machine arch and proc symlinks if something which affects +# them changed. We use .arch and .proc to indicate when they were +# updated last, otherwise make uses the target directory mtime. + +include/asm-arm/.arch: $(wildcard include/config/arch/*.h) +ifneq ("$(INCDIR)","") + @echo ' Making asm-arm/arch -> asm-arm/arch-$(INCDIR) symlink' + @rm -f include/asm-arm/arch + @ln -sf arch-$(INCDIR) include/asm-arm/arch + @touch $@ +else + @echo ' No architecture defined. You may want to use a pre-packaged config. make a5k_config, ebsa110_config, footbridge_config, etc.' +endif + +include/asm-arm/.proc: $(wildcard include/config/cpu/32.h) $(wildcard include/config/cpu/26.h) +ifneq ("$(INCDIR)","") + @echo ' Making asm-arm/proc -> asm-arm/proc-$(PROCESSOR) symlink' + @rm -f include/asm-arm/proc + @ln -sf proc-$(PROCESSOR) include/asm-arm/proc + @touch $@ +else + @echo ' No architecture defined. You may want to use a pre-packaged config. make a5k_config, ebsa110_config, footbridge_config, etc.' +endif + +.PHONY: maketools +maketools: include/asm-arm/.arch include/asm-arm/.proc \ + include/asm-arm/constants.h include/linux/version.h checkbin + @$(MAKETOOLS) vmlinux: arch/arm/vmlinux.lds -arch/arm/vmlinux.lds: $(LDSCRIPT) dummy +arch/arm/vmlinux.lds: arch/arm/Makefile $(LDSCRIPT) \ + $(wildcard include/config/cpu/32.h) \ + $(wildcard include/config/cpu/26.h) \ + $(wildcard include/config/arch/*.h) + @echo ' Generating $@' @sed 's/TEXTADDR/$(TEXTADDR)/;s/DATAADDR/$(DATAADDR)/' $(LDSCRIPT) >$@ arch/arm/kernel arch/arm/mm arch/arm/lib: dummy @@ -216,38 +274,36 @@ MRPROPER_FILES += \ arch/arm/tools/constants.h* \ - include/asm-arm/arch \ - include/asm-arm/proc \ + include/asm-arm/arch include/asm-arm/.arch \ + include/asm-arm/proc include/asm-arm/.proc \ include/asm-arm/constants.h* \ include/asm-arm/mach-types.h # We use MRPROPER_FILES and CLEAN_FILES now -archmrproper: +archmrproper: FORCE @/bin/true -archclean: +archclean: FORCE @$(MAKEBOOT) clean -archdep: scripts/mkdep archsymlinks +archdep: scripts/mkdep symlinks @$(MAKETOOLS) dep @$(MAKEBOOT) dep -# we need version.h -maketools: checkbin include/linux/version.h - @$(MAKETOOLS) all - -# Ensure this is ld "2.9.4" or later +# Ensure this is ld "2.9.5" or later NEW_LINKER := $(shell $(LD) --gc-sections --version >/dev/null 2>&1; echo $$?) ifneq ($(NEW_LINKER),0) -checkbin: +checkbin: FORCE @echo '*** ${VERSION}.${PATCHLEVEL} kernels no longer build correctly with old versions of binutils.' @echo '*** Please upgrade your binutils to 2.9.5.' @false else -checkbin: +checkbin: FORCE @true endif + +.PHONY: FORCE # My testing targets (that short circuit a few dependencies) zImg:; @$(MAKEBOOT) zImage diff -urN orig/arch/arm/boot/Makefile linux/arch/arm/boot/Makefile --- orig/arch/arm/boot/Makefile Mon Aug 5 13:29:42 2002 +++ linux/arch/arm/boot/Makefile Wed Jan 22 23:46:12 2003 @@ -5,34 +5,42 @@ # License. See the file "COPYING" in the main directory of this archive # for more details. # -# Copyright (C) 1995-2000 Russell King +# Copyright (C) 1995-2002 Russell King # SYSTEM =$(TOPDIR)/vmlinux +# Note: the following conditions must always be true: +# ZRELADDR == virt_to_phys(TEXTADDR) +# PARAMS_PHYS must be with 4MB of ZRELADDR +# INITRD_PHYS must be in RAM + ifeq ($(CONFIG_CPU_26),y) -ZTEXTADDR = 0x02080000 +ZRELADDR = 0x02080000 PARAMS_PHYS = 0x0207c000 INITRD_PHYS = 0x02180000 -INITRD_VIRT = 0x02180000 endif ifeq ($(CONFIG_ARCH_RPC),y) -ZTEXTADDR = 0x10008000 +ZRELADDR = 0x10008000 +PARAMS_PHYS = 0x10000100 +INITRD_PHYS = 0x18000000 +endif + +ifeq ($(CONFIG_ARCH_RISCSTATION),y) +ZRELADDR = 0x10008000 PARAMS_PHYS = 0x10000100 INITRD_PHYS = 0x18000000 -INITRD_VIRT = 0xc8000000 endif ifeq ($(CONFIG_ARCH_CLPS7500),y) -ZTEXTADDR = 0x10008000 +ZRELADDR = 0x10008000 endif ifeq ($(CONFIG_ARCH_EBSA110),y) -ZTEXTADDR = 0x00008000 +ZRELADDR = 0x00008000 PARAMS_PHYS = 0x00000400 INITRD_PHYS = 0x00800000 -INITRD_VIRT = 0xc0800000 endif ifeq ($(CONFIG_ARCH_SHARK),y) @@ -41,126 +49,114 @@ endif ifeq ($(CONFIG_FOOTBRIDGE),y) -ZTEXTADDR = 0x00008000 +ZRELADDR = 0x00008000 PARAMS_PHYS = 0x00000100 INITRD_PHYS = 0x00800000 -INITRD_VIRT = 0xc0800000 endif ifeq ($(CONFIG_ARCH_INTEGRATOR),y) -ZTEXTADDR = 0x00008000 +ZRELADDR = 0x00008000 PARAMS_PHYS = 0x00000100 INITRD_PHYS = 0x00800000 -INITRD_VIRT = 0xc0800000 +endif + +ifeq ($(CONFIG_ARCH_AT91RM9200DK),y) +ZRELADDR = 0x20008000 endif ifeq ($(CONFIG_ARCH_MX1ADS),y) -ZTEXTADDR = 0x08008000 +ZRELADDR = 0x08008000 endif ifeq ($(CONFIG_ARCH_CAMELOT),y) -ZTEXTADDR = 0x00008000 +ZRELADDR = 0x00008000 endif ifeq ($(CONFIG_ARCH_NEXUSPCI),y) -ZTEXTADDR = 0x40008000 +ZRELADDR = 0x40008000 endif ifeq ($(CONFIG_ARCH_L7200),y) -# RAM based kernel -#ZTEXTADDR = 0xf0400000 -#ZRELADDR = 0xf0008000 - -# FLASH based kernel -ZTEXTADDR = 0x00010000 ZRELADDR = 0xf0008000 -ZBSSADDR = 0xf03e0000 endif # The standard locations for stuff on CLPS711x type processors ifeq ($(CONFIG_ARCH_CLPS711X),y) -ZTEXTADDR = 0xc0028000 +ZRELADDR = 0xc0028000 PARAMS_PHYS = 0xc0000100 endif # Should probably have some agreement on these... ifeq ($(CONFIG_ARCH_P720T),y) INITRD_PHYS = 0xc0400000 -INITRD_VIRT = 0xc0400000 endif ifeq ($(CONFIG_ARCH_CDB89712),y) INITRD_PHYS = 0x00700000 -INITRD_VIRT = 0xc0300000 +endif + +ifeq ($(CONFIG_ARCH_OMAHA),y) +ZTEXTADDR = 0x0c008000 +PARAMS_PHYS = 0x0C000100 +INITRD_PHYS = 0x0C800000 +INITRD_VIRT = 0x0C800000 endif ifeq ($(CONFIG_ARCH_SA1100),y) -ZTEXTADDR = 0xc0008000 ZRELADDR = 0xc0008000 -ifeq ($(CONFIG_SA1100_VICTOR),y) - ZTEXTADDR = 0x00002000 - ZBSSADDR = 0xc0200000 -endif -ifeq ($(CONFIG_SA1100_SHERMAN),y) - ZTEXTADDR = 0x00050000 - ZBSSADDR = 0xc0200000 -endif -ifeq ($(CONFIG_SA1100_GRAPHICSCLIENT),y) - ZTEXTADDR = 0xC0200000 -endif -ifeq ($(CONFIG_SA1100_GRAPHICSMASTER),y) - ZTEXTADDR = 0xC0400000 -endif -ifeq ($(CONFIG_SA1100_ADSBITSY),y) - ZTEXTADDR = 0xC0400000 -endif -ifeq ($(CONFIG_SA1100_YOPY),y) - ZTEXTADDR = 0x00080000 - ZBSSADDR = 0xc0200000 -endif +# No defconfig file to move this into... +#ifeq ($(CONFIG_SA1100_YOPY),y) +# ZTEXTADDR = 0x00080000 +# ZBSSADDR = 0xc0200000 +#endif ifeq ($(CONFIG_SA1111),y) ZRELADDR = 0xc0208000 endif endif ifeq ($(CONFIG_ARCH_ANAKIN),y) -ZTEXTADDR = 0x20008000 +ZRELADDR = 0x20008000 endif # -# If you don't define ZRELADDR above, -# then it defaults to ZTEXTADDR +# We now have a PIC decompressor implementation. Decompressors running +# from RAM should not define ZTEXTADDR. Decompressors running directly +# from ROM or Flash must define ZTEXTADDR (preferably via the config) # -ifeq ($(ZRELADDR),) -ZRELADDR = $(ZTEXTADDR) +ifeq ($(CONFIG_ZBOOT_ROM),y) +ZTEXTADDR =0x$(CONFIG_ZBOOT_ROM_TEXT) +ZBSSADDR =0x$(CONFIG_ZBOOT_ROM_BSS) +else +ZTEXTADDR =0 +ZBSSADDR =ALIGN(4) endif -export SYSTEM ZTEXTADDR ZBSSADDR ZRELADDR INITRD_PHYS INITRD_VIRT PARAMS_PHYS +export SYSTEM ZTEXTADDR ZBSSADDR ZRELADDR INITRD_PHYS PARAMS_PHYS -Image: $(CONFIGURE) $(SYSTEM) - $(OBJCOPY) -O binary -R .note -R .comment -S $(SYSTEM) $@ +Image: $(SYSTEM) + $(OBJCOPY) $(OBJCOPYFLAGS) $< $@ bzImage: zImage -zImage: $(CONFIGURE) compressed/vmlinux - $(OBJCOPY) -O binary -R .note -R .comment -S compressed/vmlinux $@ +zImage: compressed/vmlinux + $(OBJCOPY) $(OBJCOPYFLAGS) $< $@ bootpImage: bootp/bootp - $(OBJCOPY) -O binary -R .note -R .comment -S bootp/bootp $@ + $(OBJCOPY) $(OBJCOPYFLAGS) $< $@ -compressed/vmlinux: $(TOPDIR)/vmlinux dep +compressed/vmlinux: $(TOPDIR)/vmlinux FORCE @$(MAKE) -C compressed vmlinux -bootp/bootp: zImage initrd +bootp/bootp: zImage initrd FORCE @$(MAKE) -C bootp bootp initrd: - @test "$(INITRD_VIRT)" != "" || (echo This architecture does not support INITRD; exit -1) + @test "$(INITRD_PHYS)" != "" || (echo This machine does not support INITRD; exit -1) @test "$(INITRD)" != "" || (echo You must specify INITRD; exit -1) -install: $(CONFIGURE) Image +install: Image sh ./install.sh $(VERSION).$(PATCHLEVEL).$(SUBLEVEL)$(EXTRAVERSION) Image $(TOPDIR)/System.map "$(INSTALL_PATH)" -zinstall: $(CONFIGURE) zImage +zinstall: zImage sh ./install.sh $(VERSION).$(PATCHLEVEL).$(SUBLEVEL)$(EXTRAVERSION) zImage $(TOPDIR)/System.map "$(INSTALL_PATH)" clean: @@ -169,3 +165,5 @@ @$(MAKE) -C bootp clean dep: + +FORCE: diff -urN orig/arch/arm/boot/bootp/Makefile linux/arch/arm/boot/bootp/Makefile --- orig/arch/arm/boot/bootp/Makefile Fri Oct 26 16:45:49 2001 +++ linux/arch/arm/boot/bootp/Makefile Thu Sep 5 22:59:42 2002 @@ -5,9 +5,7 @@ ZSYSTEM =$(TOPDIR)/arch/arm/boot/zImage ZLDFLAGS =-p -X -T bootp.lds \ --defsym initrd_addr=$(INITRD_PHYS) \ - --defsym initrd_virt=$(INITRD_VIRT) \ - --defsym params=$(PARAMS_PHYS) \ - --defsym kernel_addr=$(ZTEXTADDR) + --defsym params=$(PARAMS_PHYS) all: bootp diff -urN orig/arch/arm/boot/bootp/bootp.lds linux/arch/arm/boot/bootp/bootp.lds --- orig/arch/arm/boot/bootp/bootp.lds Tue Oct 3 20:08:18 2000 +++ linux/arch/arm/boot/bootp/bootp.lds Thu Sep 5 22:58:17 2002 @@ -1,7 +1,7 @@ /* * linux/arch/arm/boot/bootp/bootp.lds * - * Copyright (C) 2000 Russell King + * Copyright (C) 2000-2002 Russell King * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as @@ -15,13 +15,9 @@ _text = .; .text : { _stext = .; - _start = .; - init.o(.start) - kernel_start = .; + *(.start) kernel.o - kernel_len = . - kernel_start; . = ALIGN(32); - *(.text) initrd_start = .; initrd.o initrd_len = . - initrd_start; diff -urN orig/arch/arm/boot/bootp/init.S linux/arch/arm/boot/bootp/init.S --- orig/arch/arm/boot/bootp/init.S Fri Oct 26 16:45:49 2001 +++ linux/arch/arm/boot/bootp/init.S Thu Sep 5 22:59:25 2002 @@ -1,7 +1,7 @@ /* * linux/arch/arm/boot/bootp/init.S * - * Copyright (C) 2000 Russell King + * Copyright (C) 2000-2002 Russell King * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as @@ -11,8 +11,9 @@ * r0 through to r3 straight through. */ .section .start,#alloc,#execinstr - .type _entry, #function -_entry: adr r10, initdata + .type _start, #function + .globl _start +_start: adr r10, initdata ldr r11, initdata sub r11, r10, r11 @ work out exec offset b splitify @@ -22,17 +23,11 @@ initdata: .word initdata @ compiled address of this .size initdata,. - initdata - .text splitify: adr r13, data ldmia r13!, {r4-r6} @ move the initrd add r4, r4, r11 @ correction bl move - ldmia r13!, {r4-r6} @ then the kernel - mov r12, r5 - add r4, r4, r11 @ correction - bl move - /* * Setup the initrd parameters to pass to the kernel. This can either be * passed in via a param_struct or a tag list. We spot the param_struct @@ -76,6 +71,7 @@ mov r4, #16 @ length of initrd tag mov r9, #0 @ end of tag list terminator stmia r8, {r4, r5, r6, r7, r9} + adr r12, kernel_start mov pc, r12 @ call kernel /* @@ -97,15 +93,12 @@ .word initrd_addr .word initrd_len - .word kernel_start - .word kernel_addr - .word kernel_len - .word 0x54410001 @ r4 = ATAG_CORE - .word 0x54410005 @ r5 = ATAG_INITRD - .word initrd_virt @ r6 + .word 0x54420005 @ r5 = ATAG_INITRD + .word initrd_addr @ r6 .word initrd_len @ r7 .word params @ r8 - .type kernel_start,#object .type initrd_start,#object + +kernel_start: diff -urN orig/arch/arm/boot/compressed/Makefile linux/arch/arm/boot/compressed/Makefile --- orig/arch/arm/boot/compressed/Makefile Mon Aug 5 13:29:42 2002 +++ linux/arch/arm/boot/compressed/Makefile Wed Jan 22 23:46:12 2003 @@ -9,7 +9,7 @@ HEAD = head.o OBJS = misc.o -CFLAGS = $(CPPFLAGS) -O2 -DSTDC_HEADERS $(CFLAGS_BOOT) +CFLAGS = $(CPPFLAGS) -O2 -DSTDC_HEADERS $(CFLAGS_BOOT) -fpic -Uarm FONTC = $(TOPDIR)/drivers/video/font_acorn_8x8.c ZLDFLAGS = -p -X -T vmlinux.lds @@ -18,7 +18,12 @@ # ifeq ($(CONFIG_ARCH_ACORN),y) OBJS += ll_char_wr.o font.o -ZLDFLAGS += -defsym params=$(PARAMS_PHYS) +CFLAGS += -DPARAMS_PHYS=$(PARAMS_PHYS) +endif + +ifeq ($(CONFIG_ARCH_RISCSTATION),y) +OBJS += ll_char_wr.o font.o +CFLAGS += -DPARAMS_PHYS=$(PARAMS_PHYS) endif ifeq ($(CONFIG_ARCH_NETWINDER),y) @@ -33,6 +38,10 @@ OBJS += head-integrator.o endif +ifeq ($(CONFIG_ARCH_AT91RM9200DK),y) +OBJS += head-at91rm9200dk.o +endif + ifeq ($(CONFIG_ARCH_MX1ADS),y) OBJS += head-mx1ads.o endif @@ -60,18 +69,9 @@ ifeq ($(CONFIG_ARCH_SA1100),y) OBJS += head-sa1100.o -ifeq ($(CONFIG_SA1100_NANOENGINE),y) - OBJS += hw-bse.o -endif endif -SEDFLAGS = s/TEXT_START/$(ZTEXTADDR)/;s/LOAD_ADDR/$(ZRELADDR)/; - -ifneq ($(ZBSSADDR),) -SEDFLAGS += s/BSS_START/$(ZBSSADDR)/ -else -SEDFLAGS += s/BSS_START/ALIGN(4)/ -endif +SEDFLAGS = s/TEXT_START/$(ZTEXTADDR)/;s/LOAD_ADDR/$(ZRELADDR)/;s/BSS_START/$(ZBSSADDR)/ LIBGCC := $(shell $(CC) $(CFLAGS) --print-libgcc-file-name) @@ -80,11 +80,14 @@ vmlinux: $(HEAD) $(OBJS) piggy.o vmlinux.lds $(LD) $(ZLDFLAGS) $(HEAD) $(OBJS) piggy.o $(LIBGCC) -o vmlinux -$(HEAD): $(HEAD:.o=.S) +$(HEAD): $(HEAD:.o=.S) \ + $(wildcard $(TOPDIR)/include/config/zboot/rom.h) \ + $(wildcard $(TOPDIR)/include/config/cpu/32.h) \ + $(wildcard $(TOPDIR)/include/config/cpu/26.h) $(CC) $(AFLAGS) -traditional -c $(HEAD:.o=.S) piggy.o: $(SYSTEM) - $(OBJCOPY) -O binary -R .note -R .comment -S $(SYSTEM) piggy + $(OBJCOPY) $(OBJCOPYFLAGS) $(SYSTEM) piggy gzip $(GZFLAGS) < piggy > piggy.gz $(LD) -r -o $@ -b binary piggy.gz rm -f piggy piggy.gz diff -urN orig/arch/arm/boot/compressed/head-at91rm9200dk.S linux/arch/arm/boot/compressed/head-at91rm9200dk.S --- orig/arch/arm/boot/compressed/head-at91rm9200dk.S Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/boot/compressed/head-at91rm9200dk.S Wed Jan 22 23:46:12 2003 @@ -0,0 +1,13 @@ +/* + * linux/arch/arm/boot/compressed/head-at91rm9200dk.S + * + * Copyright (C) 2002 ATMEL Rousset + * + * This is merged into head.S by the linker. + * + */ +#include + + .section ".start", "ax" + mov r7,#0xFF + add r7,r7,#MACH_TYPE_AT91RM9200DK-0xFF diff -urN orig/arch/arm/boot/compressed/head-clps7500.S linux/arch/arm/boot/compressed/head-clps7500.S --- orig/arch/arm/boot/compressed/head-clps7500.S Sat Apr 28 11:24:53 2001 +++ linux/arch/arm/boot/compressed/head-clps7500.S Sat May 18 14:11:39 2002 @@ -13,7 +13,7 @@ /* This branch is taken if the CPU memory width matches the actual device in use. The default at power on is 16 bits so we must be prepared for a mismatch. */ - .section ".start", #alloc, #execinstr + .section ".start", "ax" 2: b 1f .word 0xffff diff -urN orig/arch/arm/boot/compressed/head-epxa10db.S linux/arch/arm/boot/compressed/head-epxa10db.S --- orig/arch/arm/boot/compressed/head-epxa10db.S Mon Aug 5 13:29:42 2002 +++ linux/arch/arm/boot/compressed/head-epxa10db.S Sat May 18 14:11:47 2002 @@ -1,5 +1,5 @@ #include #include - .section ".start", #alloc, #execinstr + .section ".start", "ax" mov r7, #MACH_TYPE_CAMELOT diff -urN orig/arch/arm/boot/compressed/head-ftvpci.S linux/arch/arm/boot/compressed/head-ftvpci.S --- orig/arch/arm/boot/compressed/head-ftvpci.S Thu Feb 22 11:24:58 2001 +++ linux/arch/arm/boot/compressed/head-ftvpci.S Sat May 18 14:11:56 2002 @@ -13,7 +13,7 @@ * 2 of the License, or (at your option) any later version. */ - .section ".start", #alloc, #execinstr + .section ".start", "ax" ftv_start: mcr p15, 0, r0, c7, c5, 0 @ flush I cache mrc p15, 0, r0, c1, c0 diff -urN orig/arch/arm/boot/compressed/head-integrator.S linux/arch/arm/boot/compressed/head-integrator.S --- orig/arch/arm/boot/compressed/head-integrator.S Wed Jul 4 19:53:31 2001 +++ linux/arch/arm/boot/compressed/head-integrator.S Sat May 18 14:12:03 2002 @@ -1,4 +1,4 @@ #include - .section ".start", #alloc, #execinstr + .section ".start", "ax" mov r7, #MACH_TYPE_INTEGRATOR diff -urN orig/arch/arm/boot/compressed/head-l7200.S linux/arch/arm/boot/compressed/head-l7200.S --- orig/arch/arm/boot/compressed/head-l7200.S Thu Feb 22 11:24:58 2001 +++ linux/arch/arm/boot/compressed/head-l7200.S Sat May 18 14:12:45 2002 @@ -8,12 +8,13 @@ */ #include +#include #ifndef CONFIG_ARCH_L7200 #error What am I doing here... #endif - .section ".start", #alloc, #execinstr + .section ".start", "ax" __L7200_start: mov r0, #0x00100000 @ FLASH address of initrd @@ -26,4 +27,4 @@ ble 1b mov r8, #0 @ Zero it out - mov r7, #19 @ Set architecture ID + mov r7, #MACH_TYPE_L7200 @ Set architecture ID diff -urN orig/arch/arm/boot/compressed/head-netwinder.S linux/arch/arm/boot/compressed/head-netwinder.S --- orig/arch/arm/boot/compressed/head-netwinder.S Tue Oct 3 20:08:18 2000 +++ linux/arch/arm/boot/compressed/head-netwinder.S Sat May 18 14:19:35 2002 @@ -1,50 +1,13 @@ /* * linux/arch/arm/boot/compressed/head-netwinder.S * - * Copyright (C) 2000 Russell King + * Copyright (C) 2000-2002 Russell King * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as * published by the Free Software Foundation. */ -#define K(a,b,c) ((a) << 24 | (b) << 12 | (c)) + .section ".start", "ax" - .section ".start", #alloc, #execinstr - - /* - * check to see if we are running from the correct address. - * If not, we move ourselves in a two stage process. Firstly, - * we copy the start of the kernel (which includes this code) - * to 0x8000, and then jump to this code to continue with the - * rest (since this code will get overwritten). - */ - adr r2, 1f - ldmdb r2, {r9, r10} - and r3, r2, #0xc000 - teq r3, #0x8000 @ correctly located? - beq 2f @ skip this code - bic r3, r2, #0xc000 - orr r3, r3, #0x8000 - mov r0, r3 @ new address if '1' - mov r4, #64 @ number of bytes to copy - sub r5, r10, r9 @ total number of bytes to copy - b 1f - - .word _start - .word __bss_start - -1: - .rept 4 - ldmia r2!, {r6, r9, r10, r11} - stmia r3!, {r6, r9, r10, r11} - .endr - subs r4, r4, #64 - bcs 1b - movs r4, r5 @ remaining length - mov r5, #0 @ no more to copy - movne pc, r0 @ jump back to 1 (in the newly copied - @ code) - mov r7, #5 @ only here to fix NeTTroms which dont - mov r8, #2 << 24 @ scheduled for removal in 2.5.xx - orr r8, r8, #5 << 12 -2: + mov r7, #5 + mov r8, #0 diff -urN orig/arch/arm/boot/compressed/head-sa1100.S linux/arch/arm/boot/compressed/head-sa1100.S --- orig/arch/arm/boot/compressed/head-sa1100.S Fri Oct 26 16:45:49 2001 +++ linux/arch/arm/boot/compressed/head-sa1100.S Sat May 18 14:12:56 2002 @@ -11,7 +11,7 @@ #include #include - .section ".start", #alloc, #execinstr + .section ".start", "ax" __SA1100_start: diff -urN orig/arch/arm/boot/compressed/head-shark.S linux/arch/arm/boot/compressed/head-shark.S --- orig/arch/arm/boot/compressed/head-shark.S Mon Aug 5 13:29:42 2002 +++ linux/arch/arm/boot/compressed/head-shark.S Sat May 18 14:19:55 2002 @@ -16,7 +16,7 @@ #include - .section ".start", #alloc, #execinstr + .section ".start", "ax" b __beginning @@ -38,7 +38,7 @@ adr r1, __ofw_data add r2, r1, #4 mov lr, pc - b SYMBOL_NAME(ofw_init) + b ofw_init mov r1, #0 adr r2, __mmu_off @ calculate physical address @@ -109,7 +109,7 @@ add sp, sp, #128 adr r0, __ofw_data mov lr, pc - b SYMBOL_NAME(create_params) + b create_params mov r8, #0 mov r7, #15 diff -urN orig/arch/arm/boot/compressed/head.S linux/arch/arm/boot/compressed/head.S --- orig/arch/arm/boot/compressed/head.S Mon Aug 5 13:29:42 2002 +++ linux/arch/arm/boot/compressed/head.S Wed Jan 22 23:46:12 2003 @@ -1,7 +1,7 @@ /* * linux/arch/arm/boot/compressed/head.S * - * Copyright (C) 1996-1999 Russell King + * Copyright (C) 1996-2002 Russell King * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as @@ -19,6 +19,13 @@ */ #ifdef DEBUG #if defined(CONFIG_DEBUG_DC21285_PORT) + .macro loadsp, rb + mov \rb, #0x42000000 + .endm + .macro writeb, rb + str \rb, [r3, #0x160] + .endm +#elif defined(CONFIG_FOOTBRIDGE) .macro loadsp, rb mov \rb, #0x7c000000 .endm @@ -40,6 +47,13 @@ .macro writeb, rb strb \rb, [r3, #0] .endm +#elif defined(CONFIG_ARCH_AT91RM9200DK) + .macro loadsp, rb + mov \rb, =0xFFFFF200 @AT91C_BASE_DBGU + .endm + .macro writeb, rb + strb \rb, [r3, #0x1C] @DBGU_THR] + .endm #elif defined(CONFIG_ARCH_SA1100) .macro loadsp, rb mov \rb, #0x80000000 @ physical base address @@ -50,6 +64,11 @@ # endif .endm .macro writeb, rb +/* + * "The ARM peripheral bus does not support byte or half-word operations. + * All reads and writes of the UART by the CPU should be wordwide." + * - SA-1100 Developer's Manual, August 1999 + */ str \rb, [r3, #0x14] @ UTDR .endm #else @@ -144,33 +163,98 @@ */ .text -1: adr r2, LC0 - ldmia r2, {r2, r3, r4, r5, sp} + adr r0, LC0 + ldmia r0, {r1, r2, r3, r4, r5, r6, ip, sp} + subs r0, r0, r1 @ calculate the delta offset + + teq r0, #0 @ if delta is zero, we're + beq not_relocated @ running at the address we + @ were linked at. - mov r0, #0 + /* + * We're running at a different address. We need to fix + * up various pointers: + * r5 - zImage base address + * r6 - GOT start + * ip - GOT end + */ + add r5, r5, r0 + add r6, r6, r0 + add ip, ip, r0 + +#ifndef CONFIG_ZBOOT_ROM + /* + * If we're running fully PIC === CONFIG_ZBOOT_ROM = n, + * we need to fix up pointers into the BSS region. + * r2 - BSS start + * r3 - BSS end + * sp - stack pointer + */ + add r2, r2, r0 + add r3, r3, r0 + add sp, sp, r0 + + /* + * Relocate all entries in the GOT table. + */ +1: ldr r1, [r6, #0] + add r1, r1, r0 + str r1, [r6], #4 + cmp r6, ip + blo 1b +#else + + /* + * Relocate entries in the GOT table. We only relocate + * the entries that are outside the (relocated) BSS region. + */ +1: ldr r1, [r6, #0] + cmp r1, r2 @ entry < bss_start || + cmphs r3, r1 @ _end < entry + addlo r1, r1, r0 + str r1, [r6], #4 + cmp r6, ip + blo 1b + +#endif + +not_relocated: mov r0, #0 1: str r0, [r2], #4 @ clear bss str r0, [r2], #4 str r0, [r2], #4 str r0, [r2], #4 cmp r2, r3 - blt 1b + blo 1b - mrc p15, 0, r6, c0, c0 @ get processor ID + /* + * The C runtime environment should now be setup + * sufficiently. Turn the cache on, set up some + * pointers, and start decompressing. + */ bl cache_on mov r1, sp @ malloc space above stack add r2, sp, #0x10000 @ 64k max - teq r4, r5 @ will we overwrite ourselves? - moveq r5, r2 @ decompress after image - movne r5, r4 @ decompress to final location +/* + * Check to see if we will overwrite ourselves. + * r4 = final kernel address + * r5 = start of this image + * r2 = end of malloc space (and therefore this image) + * We basically want: + * r4 >= r2 -> OK + * r4 + image length <= r5 -> OK + */ + cmp r4, r2 + bhs wont_overwrite + add r0, r4, #4096*1024 @ 4MB largest kernel size + cmp r0, r5 + bls wont_overwrite + mov r5, r2 @ decompress after malloc space mov r0, r5 mov r3, r7 - bl SYMBOL_NAME(decompress_kernel) - - teq r4, r5 @ do we need to relocate - beq call_kernel @ the kernel? + bl decompress_kernel add r0, r0, #127 bic r0, r0, #127 @ align the kernel length @@ -185,23 +269,39 @@ */ add r1, r5, r0 @ end of decompressed kernel adr r2, reloc_start - adr r3, reloc_end + ldr r3, LC1 + add r3, r2, r3 1: ldmia r2!, {r8 - r13} @ copy relocation code stmia r1!, {r8 - r13} ldmia r2!, {r8 - r13} stmia r1!, {r8 - r13} cmp r2, r3 - blt 1b + blo 1b bl cache_clean_flush add pc, r5, r0 @ call relocation code +/* + * We're not in danger of overwriting ourselves. Do this the simple way. + * + * r4 = kernel execution address + * r7 = architecture ID + */ +wont_overwrite: mov r0, r4 + mov r3, r7 + bl decompress_kernel + b call_kernel + .type LC0, #object -LC0: .word __bss_start - .word _end - .word _load_addr - .word _start - .word user_stack+4096 +LC0: .word LC0 @ r1 + .word __bss_start @ r2 + .word _end @ r3 + .word _load_addr @ r4 + .word _start @ r5 + .word _got_start @ r6 + .word _got_end @ ip + .word user_stack+4096 @ sp +LC1: .word reloc_end - reloc_start .size LC0, . - LC0 /* @@ -218,22 +318,15 @@ * r7 = architecture number * r8 = run-time address of "start" * On exit, - * r0, r1, r2, r3, r8, r9 corrupted + * r1, r2, r3, r8, r9, r12 corrupted * This routine must preserve: * r4, r5, r6, r7 */ .align 5 -cache_on: ldr r1, proc_sa110_type - eor r1, r1, r6 - movs r1, r1, lsr #5 @ catch SA110 and SA1100 - beq 1f - ldr r1, proc_sa1110_type - eor r1, r1, r6 - movs r1, r1, lsr #4 -@ movne pc, lr - bne cache_off -1: - sub r3, r4, #16384 @ Page directory size +cache_on: mov r3, #8 @ cache_on function + b call_cache_fn + +__setup_mmu: sub r3, r4, #16384 @ Page directory size bic r3, r3, #0xff @ Align the pointer bic r3, r3, #0x3f00 /* @@ -248,9 +341,9 @@ orr r1, r1, #3 << 10 add r2, r3, #16384 1: cmp r1, r8 @ if virt > start of RAM - orrge r1, r1, #0x0c @ set cacheable, bufferable + orrhs r1, r1, #0x0c @ set cacheable, bufferable cmp r1, r9 @ if virt > end of RAM - bicge r1, r1, #0x0c @ clear cacheable, bufferable + bichs r1, r1, #0x0c @ clear cacheable, bufferable str r1, [r0], #4 @ 1:1 mapping add r1, r1, #1048576 teq r0, r2 @@ -269,24 +362,42 @@ str r1, [r0], #4 add r1, r1, #1048576 str r1, [r0] + mov pc, lr +__armv4_cache_on: + mov r12, lr + bl __setup_mmu mov r0, #0 mcr p15, 0, r0, c7, c10, 4 @ drain write buffer - mcr p15, 0, r0, c8, c7 @ flush I,D TLBs - mcr p15, 0, r3, c2, c0 @ load page table pointer + mcr p15, 0, r0, c8, c7, 0 @ flush I,D TLBs + mcr p15, 0, r3, c2, c0, 0 @ load page table pointer mov r0, #-1 - mcr p15, 0, r0, c3, c0 @ load domain access register - mrc p15, 0, r0, c1, c0 + mcr p15, 0, r0, c3, c0, 0 @ load domain access register + mrc p15, 0, r0, c1, c0, 0 orr r0, r0, #0x1000 @ I-cache enable #ifndef DEBUG orr r0, r0, #0x003d @ Write buffer, mmu #endif - mcr p15, 0, r0, c1, c0 - mov pc, lr + mcr p15, 0, r0, c1, c0, 0 + mov pc, r12 + +__arm6_cache_on: + mov r12, lr + bl __setup_mmu + mov r0, #0 + mcr p15, 0, r0, c7, c0, 0 @ invalidate whole cache v3 + mcr p15, 0, r0, c5, c0, 0 @ invalidate whole TLB v3 + mcr p15, 0, r3, c2, c0, 0 @ load page table pointer + mov r0, #-1 + mcr p15, 0, r0, c3, c0, 0 @ load domain access control + mov r0, #0x3d + mcr p15, 0, r0, c1, c0, 0 @ load control register + mov pc, r12 /* - * This code is relocatable. It is relocated by the above code to the end - * of the kernel and executed there. During this time, we have no stacks. + * All code following this line is relocatable. It is relocated by + * the above code to the end of the decompressed kernel image and + * executed there. During this time, we have no stacks. * * r0 = decompressed kernel length * r1-r3 = unused @@ -307,7 +418,7 @@ .endr cmp r5, r8 - blt 1b + blo 1b debug_reloc_end call_kernel: bl cache_clean_flush @@ -317,47 +428,117 @@ mov pc, r4 @ call kernel /* - * Here follow the relocatable cache support functions for - * the various processors. + * Here follow the relocatable cache support functions for the + * various processors. This is a generic hook for locating an + * entry and jumping to an instruction at the specified offset + * from the start of the block. Please note this is all position + * independent code. + * + * r1 = corrupted + * r2 = corrupted + * r3 = block offset + * r6 = corrupted + * r12 = corrupted */ - .type proc_sa110_type,#object -proc_sa110_type: - .word 0x4401a100 - .size proc_sa110_type, . - proc_sa110_type - - .type proc_sa1110_type,#object -proc_sa1110_type: - .word 0x6901b110 - .size proc_sa1110_type, . - proc_sa1110_type +call_cache_fn: adr r12, proc_types + mrc p15, 0, r6, c0, c0 @ get processor ID +1: ldr r1, [r12, #0] @ get value + ldr r2, [r12, #4] @ get mask + eor r1, r1, r6 @ (real ^ match) + tst r1, r2 @ & mask + addeq pc, r12, r3 @ call cache function + add r12, r12, #4*5 + b 1b + +/* + * Table for cache operations. This is basically: + * - CPU ID match + * - CPU ID mask + * - 'cache on' method instruction + * - 'cache off' method instruction + * - 'cache flush' method instruction + * + * We match an entry using: ((real_id ^ match) & mask) == 0 + * + * Writethrough caches generally only need 'on' and 'off' + * methods. Writeback caches _must_ have the flush method + * defined. + */ + .type proc_types,#object +proc_types: + .word 0x41560600 @ ARM6/610 + .word 0xffffffe0 + b __arm6_cache_off @ works, but slow + b __arm6_cache_off + mov pc, lr +@ b __arm6_cache_on @ untested +@ b __arm6_cache_off +@ b __armv3_cache_flush + + .word 0x41007000 @ ARM7/710 + .word 0xfff8fe00 + b __arm7_cache_off + b __arm7_cache_off + mov pc, lr + + .word 0x41807200 @ ARM720T (writethrough) + .word 0xffffff00 + b __armv4_cache_on + b __armv4_cache_off + mov pc, lr + + .word 0x41129200 @ ARM920T + .word 0xff00fff0 + b __armv4_cache_on + b __armv4_cache_off + b __armv4_cache_flush + + .word 0x41029220 @ ARM922T + .word 0xff00fff0 + b __armv4_cache_on + b __armv4_cache_off + b __armv4_cache_flush + + .word 0x4401a100 @ sa110 / sa1100 + .word 0xffffffe0 + b __armv4_cache_on + b __armv4_cache_off + b __armv4_cache_flush + + .word 0x6901b110 @ sa1110 + .word 0xfffffff0 + b __armv4_cache_on + b __armv4_cache_off + b __armv4_cache_flush + + .word 0x69050000 @ xscale + .word 0xffff0000 + b __armv4_cache_on + b __armv4_cache_off + b __armv4_cache_flush + + .word 0 @ unrecognised type + .word 0 + mov pc, lr + mov pc, lr + mov pc, lr + + .size proc_types, . - proc_types /* * Turn off the Cache and MMU. ARMv3 does not support * reading the control register, but ARMv4 does. * * On entry, r6 = processor ID - * On exit, r0, r1 corrupted + * On exit, r0, r1, r2, r3, r12 corrupted * This routine must preserve: r4, r6, r7 */ .align 5 -cache_off: -#ifdef CONFIG_CPU_ARM610 - eor r1, r6, #0x41000000 - eor r1, r1, #0x00560000 - bic r1, r1, #0x0000001f - teq r1, #0x00000600 - mov r0, #0x00000060 @ ARM6 control reg. - beq __armv3_cache_off -#endif -#ifdef CONFIG_CPU_ARM710 - eor r1, r6, #0x41000000 - bic r1, r1, #0x00070000 - bic r1, r1, #0x000000ff - teq r1, #0x00007000 @ ARM7 - teqne r1, #0x00007100 @ ARM710 - mov r0, #0x00000070 @ ARM7 control reg. - beq __armv3_cache_off -#endif +cache_off: mov r3, #12 @ cache_off function + b call_cache_fn + +__armv4_cache_off: mrc p15, 0, r0, c1, c0 bic r0, r0, #0x000d mcr p15, 0, r0, c1, c0 @ turn MMU and cache off @@ -366,11 +547,19 @@ mcr p15, 0, r0, c8, c7 @ invalidate whole TLB v4 mov pc, lr +__arm6_cache_off: + mov r0, #0x00000030 @ ARM6 control reg. + b __armv3_cache_off + +__arm7_cache_off: + mov r0, #0x00000070 @ ARM7 control reg. + b __armv3_cache_off + __armv3_cache_off: - mcr p15, 0, r0, c1, c0 @ turn MMU and cache off + mcr p15, 0, r0, c1, c0, 0 @ turn MMU and cache off mov r0, #0 - mcr p15, 0, r0, c7, c0 @ invalidate whole cache v3 - mcr p15, 0, r0, c5, c0 @ invalidate whole TLB v3 + mcr p15, 0, r0, c7, c0, 0 @ invalidate whole cache v3 + mcr p15, 0, r0, c5, c0, 0 @ invalidate whole TLB v3 mov pc, lr /* @@ -379,23 +568,18 @@ * On entry, * r6 = processor ID * On exit, - * r1, r2, r12 corrupted + * r1, r2, r3, r12 corrupted * This routine must preserve: - * r4, r6, r7 + * r0, r4, r5, r6, r7 */ .align 5 cache_clean_flush: - ldr r1, proc_sa110_type - eor r1, r1, r6 - movs r1, r1, lsr #5 @ catch SA110 and SA1100 - beq 1f - ldr r1, proc_sa1110_type - eor r1, r1, r6 - movs r1, r1, lsr #4 - movne pc, lr -1: + mov r3, #16 + b call_cache_fn + +__armv4_cache_flush: bic r1, pc, #31 - add r2, r1, #32768 + add r2, r1, #65536 @ 2x the largest dcache size 1: ldr r12, [r1], #32 @ s/w flush D cache teq r1, r2 bne 1b @@ -404,6 +588,11 @@ mcr p15, 0, r1, c7, c10, 4 @ drain WB mov pc, lr +__armv3_cache_flush: + mov r1, #0 + mcr p15, 0, r0, c7, c0, 0 @ invalidate whole cache v3 + mov pc, lr + /* * Various debugging routines for printing hex characters and * memory, which again must be relocatable. @@ -479,5 +668,5 @@ reloc_end: .align - .section ".stack" + .section ".stack", "w" user_stack: .space 4096 diff -urN orig/arch/arm/boot/compressed/hw-bse.c linux/arch/arm/boot/compressed/hw-bse.c --- orig/arch/arm/boot/compressed/hw-bse.c Sat Aug 26 19:55:39 2000 +++ linux/arch/arm/boot/compressed/hw-bse.c Thu Jan 1 01:00:00 1970 @@ -1,74 +0,0 @@ -/* - * Bright Star Engineering Inc. - * - * code for readng parameters from the - * parameter blocks of the boot block - * flash memory - * - */ - -static int strcmp(const char *s1, const char *s2) -{ - while (*s1 != '\0' && *s1 == *s2) - { - s1++; - s2++; - } - - return (*(unsigned char *) s1) - (*(unsigned char *) s2); -} - -struct pblk_t { - char type; - unsigned short size; -}; - -static char *bse_getflashparam(char *name) { - unsigned int esize; - char *q,*r; - unsigned char *p,*e; - struct pblk_t *thepb = (struct pblk_t *) 0x00004000; - struct pblk_t *altpb = (struct pblk_t *) 0x00006000; - if (thepb->type&1) { - if (altpb->type&1) { - /* no valid param block */ - return (char*)0; - } else { - /* altpb is valid */ - struct pblk_t *tmp; - tmp = thepb; - thepb = altpb; - altpb = tmp; - } - } - p = (char*)thepb + sizeof(struct pblk_t); - e = p + thepb->size; - while (p < e) { - q = p; - esize = *p; - if (esize == 0xFF) break; - if (esize == 0) break; - if (esize > 127) { - esize = (esize&0x7F)<<8 | p[1]; - q++; - } - q++; - r=q; - if (*r && ((name == 0) || (!strcmp(name,r)))) { - while (*q++) ; - return q; - } - p+=esize; - } - return (char*)0; -} - -void bse_setup(void) { - /* extract the linux cmdline from flash */ - char *name=bse_getflashparam("linuxboot"); - char *x = (char *)0xc0000100; - if (name) { - while (*name) *x++=*name++; - } - *x=0; -} diff -urN orig/arch/arm/boot/compressed/ll_char_wr.S linux/arch/arm/boot/compressed/ll_char_wr.S --- orig/arch/arm/boot/compressed/ll_char_wr.S Sat Apr 28 11:24:53 2001 +++ linux/arch/arm/boot/compressed/ll_char_wr.S Wed Sep 11 23:17:36 2002 @@ -19,144 +19,116 @@ #include #include - .text + .text -#define BOLD 0x01 -#define ITALIC 0x02 -#define UNDERLINE 0x04 -#define FLASH 0x08 -#define INVERSE 0x10 - -LC0: .word SYMBOL_NAME(bytes_per_char_h) - .word SYMBOL_NAME(video_size_row) - .word SYMBOL_NAME(acorndata_8x8) - .word SYMBOL_NAME(con_charconvtable) +LC0: .word LC0 + .word bytes_per_char_h + .word video_size_row + .word acorndata_8x8 + .word con_charconvtable +/* + * r0 = ptr + * r1 = char + * r2 = white + */ ENTRY(ll_write_char) - stmfd sp!, {r4 - r7, lr} + stmfd sp!, {r4 - r7, lr} @ @ Smashable regs: {r0 - r3}, [r4 - r7], (r8 - fp), [ip], (sp), [lr], (pc) @ - eor ip, r1, #UNDERLINE << 9 -/* - * calculate colours - */ - tst r1, #INVERSE << 9 - moveq r2, r1, lsr #16 - moveq r3, r1, lsr #24 - movne r2, r1, lsr #24 - movne r3, r1, lsr #16 - and r3, r3, #255 - and r2, r2, #255 -/* - * calculate offset into character table - */ - mov r1, r1, lsl #23 - mov r1, r1, lsr #20 -/* - * calculate offset required for each row [maybe I should make this an argument to this fn. - * Have to see what the register usage is like in the calling routines. - */ - adr r4, LC0 - ldmia r4, {r4, r5, r6, lr} - ldr r4, [r4] - ldr r5, [r5] -/* - * Go to resolution-dependent routine... - */ - cmp r4, #4 - blt Lrow1bpp - eor r2, r3, r2 @ Create eor mask to change colour from bg - orr r3, r3, r3, lsl #8 @ to fg. - orr r3, r3, r3, lsl #16 - add r0, r0, r5, lsl #3 @ Move to bottom of character - add r1, r1, #7 - ldrb r7, [r6, r1] - tst ip, #UNDERLINE << 9 - eoreq r7, r7, #255 - teq r4, #8 - beq Lrow8bpplp + /* + * calculate offset into character table + */ + mov r1, r1, lsl #3 + /* + * calculate offset required for each row. + */ + adr ip, LC0 + ldmia ip, {r3, r4, r5, r6, lr} + sub ip, ip, r3 + add r6, r6, ip + add lr, lr, ip + ldr r4, [r4, ip] + ldr r5, [r5, ip] + /* + * Go to resolution-dependent routine... + */ + cmp r4, #4 + blt Lrow1bpp + add r0, r0, r5, lsl #3 @ Move to bottom of character + orr r1, r1, #7 + ldrb r7, [r6, r1] + teq r4, #8 + beq Lrow8bpplp @ @ Smashable regs: {r0 - r3}, [r4], {r5 - r7}, (r8 - fp), [ip], (sp), {lr}, (pc) @ - orr r3, r3, r3, lsl #4 -Lrow4bpplp: ldr r7, [lr, r7, lsl #2] - mul r7, r2, r7 - tst r1, #7 @ avoid using r7 directly after - eor ip, r3, r7 - str ip, [r0, -r5]! - LOADREGS(eqfd, sp!, {r4 - r7, pc}) - sub r1, r1, #1 - ldrb r7, [r6, r1] - ldr r7, [lr, r7, lsl #2] - mul r7, r2, r7 - tst r1, #7 @ avoid using r7 directly after - eor ip, r3, r7 - str ip, [r0, -r5]! - subne r1, r1, #1 - ldrneb r7, [r6, r1] - bne Lrow4bpplp - LOADREGS(fd, sp!, {r4 - r7, pc}) +Lrow4bpplp: + ldr r7, [lr, r7, lsl #2] + mul r7, r2, r7 + sub r1, r1, #1 @ avoid using r7 directly after + str r7, [r0, -r5]! + ldrb r7, [r6, r1] + ldr r7, [lr, r7, lsl #2] + mul r7, r2, r7 + tst r1, #7 @ avoid using r7 directly after + str r7, [r0, -r5]! + subne r1, r1, #1 + ldrneb r7, [r6, r1] + bne Lrow4bpplp + LOADREGS(fd, sp!, {r4 - r7, pc}) @ @ Smashable regs: {r0 - r3}, [r4], {r5 - r7}, (r8 - fp), [ip], (sp), {lr}, (pc) @ -Lrow8bpplp: mov ip, r7, lsr #4 - ldr ip, [lr, ip, lsl #2] - mul r4, r2, ip - and ip, r7, #15 @ avoid r4 - ldr ip, [lr, ip, lsl #2] @ avoid r4 - mul ip, r2, ip @ avoid r4 - eor r4, r3, r4 @ avoid ip - tst r1, #7 @ avoid ip - sub r0, r0, r5 @ avoid ip - eor ip, r3, ip - stmia r0, {r4, ip} - LOADREGS(eqfd, sp!, {r4 - r7, pc}) - sub r1, r1, #1 - ldrb r7, [r6, r1] - mov ip, r7, lsr #4 - ldr ip, [lr, ip, lsl #2] - mul r4, r2, ip - and ip, r7, #15 @ avoid r4 - ldr ip, [lr, ip, lsl #2] @ avoid r4 - mul ip, r2, ip @ avoid r4 - eor r4, r3, r4 @ avoid ip - tst r1, #7 @ avoid ip - sub r0, r0, r5 @ avoid ip - eor ip, r3, ip - stmia r0, {r4, ip} - subne r1, r1, #1 - ldrneb r7, [r6, r1] - bne Lrow8bpplp - LOADREGS(fd, sp!, {r4 - r7, pc}) +Lrow8bpplp: + mov ip, r7, lsr #4 + ldr ip, [lr, ip, lsl #2] + mul r4, r2, ip + and ip, r7, #15 @ avoid r4 + ldr ip, [lr, ip, lsl #2] @ avoid r4 + mul ip, r2, ip @ avoid r4 + sub r1, r1, #1 @ avoid ip + sub r0, r0, r5 @ avoid ip + stmia r0, {r4, ip} + ldrb r7, [r6, r1] + mov ip, r7, lsr #4 + ldr ip, [lr, ip, lsl #2] + mul r4, r2, ip + and ip, r7, #15 @ avoid r4 + ldr ip, [lr, ip, lsl #2] @ avoid r4 + mul ip, r2, ip @ avoid r4 + tst r1, #7 @ avoid ip + sub r0, r0, r5 @ avoid ip + stmia r0, {r4, ip} + subne r1, r1, #1 + ldrneb r7, [r6, r1] + bne Lrow8bpplp + LOADREGS(fd, sp!, {r4 - r7, pc}) @ @ Smashable regs: {r0 - r3}, [r4], {r5, r6}, [r7], (r8 - fp), [ip], (sp), [lr], (pc) @ -Lrow1bpp: add r6, r6, r1 - ldmia r6, {r4, r7} - tst ip, #INVERSE << 9 - mvnne r4, r4 - mvnne r7, r7 - strb r4, [r0], r5 - mov r4, r4, lsr #8 - strb r4, [r0], r5 - mov r4, r4, lsr #8 - strb r4, [r0], r5 - mov r4, r4, lsr #8 - strb r4, [r0], r5 - strb r7, [r0], r5 - mov r7, r7, lsr #8 - strb r7, [r0], r5 - mov r7, r7, lsr #8 - strb r7, [r0], r5 - mov r7, r7, lsr #8 - tst ip, #UNDERLINE << 9 - mvneq r7, r7 - strb r7, [r0], r5 - LOADREGS(fd, sp!, {r4 - r7, pc}) +Lrow1bpp: + add r6, r6, r1 + ldmia r6, {r4, r7} + strb r4, [r0], r5 + mov r4, r4, lsr #8 + strb r4, [r0], r5 + mov r4, r4, lsr #8 + strb r4, [r0], r5 + mov r4, r4, lsr #8 + strb r4, [r0], r5 + strb r7, [r0], r5 + mov r7, r7, lsr #8 + strb r7, [r0], r5 + mov r7, r7, lsr #8 + strb r7, [r0], r5 + mov r7, r7, lsr #8 + strb r7, [r0], r5 + LOADREGS(fd, sp!, {r4 - r7, pc}) - .bss + .bss ENTRY(con_charconvtable) - .space 1024 + .space 1024 diff -urN orig/arch/arm/boot/compressed/misc.c linux/arch/arm/boot/compressed/misc.c --- orig/arch/arm/boot/compressed/misc.c Mon Aug 5 13:29:42 2002 +++ linux/arch/arm/boot/compressed/misc.c Thu Sep 5 22:57:20 2002 @@ -22,7 +22,6 @@ #include #include -#include #ifdef STANDALONE_DEBUG #define puts printf @@ -291,7 +290,6 @@ free_mem_ptr_end = free_mem_ptr_end_p; __machine_arch_type = arch_id; - proc_decomp_setup(); arch_decomp_setup(); makecrc(); @@ -315,4 +313,3 @@ return 0; } #endif - diff -urN orig/arch/arm/boot/compressed/vmlinux.lds.in linux/arch/arm/boot/compressed/vmlinux.lds.in --- orig/arch/arm/boot/compressed/vmlinux.lds.in Wed Jul 4 19:53:31 2001 +++ linux/arch/arm/boot/compressed/vmlinux.lds.in Sat May 18 14:08:17 2002 @@ -35,29 +35,26 @@ _etext = .; - .data : { - *(.data) - } - + _got_start = .; + .got : { *(.got) } + _got_end = .; + .got.plt : { *(.got.plt) } + .data : { *(.data) } _edata = .; . = BSS_START; __bss_start = .; - .bss : { - *(.bss) - } + .bss : { *(.bss) } _end = .; - .stack : { - *(.stack) - } + .stack (NOLOAD) : { *(.stack) } - .stab 0 : { *(.stab) } - .stabstr 0 : { *(.stabstr) } - .stab.excl 0 : { *(.stab.excl) } - .stab.exclstr 0 : { *(.stab.exclstr) } - .stab.index 0 : { *(.stab.index) } - .stab.indexstr 0 : { *(.stab.indexstr) } - .comment 0 : { *(.comment) } + .stab 0 : { *(.stab) } + .stabstr 0 : { *(.stabstr) } + .stab.excl 0 : { *(.stab.excl) } + .stab.exclstr 0 : { *(.stab.exclstr) } + .stab.index 0 : { *(.stab.index) } + .stab.indexstr 0 : { *(.stab.indexstr) } + .comment 0 : { *(.comment) } } diff -urN orig/arch/arm/config.in linux/arch/arm/config.in --- orig/arch/arm/config.in Mon Aug 5 13:29:42 2002 +++ linux/arch/arm/config.in Thu Feb 27 23:20:10 2003 @@ -2,7 +2,7 @@ # For a description of the syntax of this configuration file, # see Documentation/kbuild/config-language.txt. # -mainmenu_name "Linux Kernel Configuration" +mainmenu_name "ARM Linux Kernel Configuration" define_bool CONFIG_ARM y define_bool CONFIG_EISA n @@ -39,14 +39,17 @@ CLPS711x/EP721x-based CONFIG_ARCH_CLPS711X \ Co-EBSA285 CONFIG_ARCH_CO285 \ EBSA-110 CONFIG_ARCH_EBSA110 \ - Epxa10db CONFIG_ARCH_CAMELOT \ + Excalibur-ARM CONFIG_ARCH_CAMELOT \ FootBridge CONFIG_ARCH_FOOTBRIDGE \ Integrator CONFIG_ARCH_INTEGRATOR \ + Omaha CONFIG_ARCH_OMAHA \ LinkUp-L7200 CONFIG_ARCH_L7200 \ Motorola-MX1ADS CONFIG_ARCH_MX1ADS \ RiscPC CONFIG_ARCH_RPC \ - SA1100-based CONFIG_ARCH_SA1100 \ - Shark CONFIG_ARCH_SHARK" RiscPC + RiscStation CONFIG_ARCH_RISCSTATION \ + SA1100-based CONFIG_ARCH_SA1100 \ + Shark CONFIG_ARCH_SHARK \ + At91rm9200dk CONFIG_ARCH_AT91RM9200DK " RiscPC mainmenu_option next_comment comment 'Archimedes/A5000 Implementations' @@ -69,10 +72,14 @@ mainmenu_option next_comment comment 'SA11x0 Implementations' +dep_bool ' ACCELENT DevBoard' CONFIG_SA1100_ACCELENT $CONFIG_ARCH_SA1100 dep_bool ' Assabet' CONFIG_SA1100_ASSABET $CONFIG_ARCH_SA1100 dep_bool ' Include support for Neponset' CONFIG_ASSABET_NEPONSET $CONFIG_SA1100_ASSABET +dep_bool ' ADS Advanced Graphics Client' CONFIG_SA1100_ADSAGC $CONFIG_ARCH_SA1100 dep_bool ' ADS Bitsy' CONFIG_SA1100_ADSBITSY $CONFIG_ARCH_SA1100 +dep_bool ' ADS Bitsy Plus' CONFIG_SA1100_ADSBITSYPLUS $CONFIG_ARCH_SA1100 dep_bool ' Brutus' CONFIG_SA1100_BRUTUS $CONFIG_ARCH_SA1100 +dep_bool ' Cep' CONFIG_SA1100_CEP $CONFIG_ARCH_SA1100 dep_bool ' CerfBoard' CONFIG_SA1100_CERF $CONFIG_ARCH_SA1100 if [ "$CONFIG_SA1100_CERF" = "y" ]; then choice 'Cerf RAM available' \ @@ -94,6 +101,8 @@ else define_bool CONFIG_SA1100_H3XXX n fi +dep_tristate 'Compaq iPAQ Handheld sleeve support' CONFIG_H3600_SLEEVE $CONFIG_SA1100_H3600 +#dep_bool ' Consus' CONFIG_SA1100_CONSUS $CONFIG_ARCH_SA1100 #dep_bool ' Empeg' CONFIG_SA1100_EMPEG $CONFIG_ARCH_SA1100 dep_bool ' Extenex HandHeld Theater (Squashtail)' CONFIG_SA1100_EXTENEX1 $CONFIG_ARCH_SA1100 if [ "$CONFIG_SA1100_EXTENEX1" = "y" ]; then @@ -104,6 +113,7 @@ dep_bool ' Frodo' CONFIG_SA1100_FRODO $CONFIG_ARCH_SA1100 dep_bool ' GraphicsClient Plus' CONFIG_SA1100_GRAPHICSCLIENT $CONFIG_ARCH_SA1100 dep_bool ' GraphicsMaster' CONFIG_SA1100_GRAPHICSMASTER $CONFIG_ARCH_SA1100 +dep_bool ' HackKit Core Board' CONFIG_SA1100_HACKKIT $CONFIG_ARCH_SA1100 dep_bool ' HP Labs BadgePAD 4' CONFIG_SA1100_BADGE4 $CONFIG_ARCH_SA1100 dep_bool ' HP Jornada 720' CONFIG_SA1100_JORNADA720 $CONFIG_ARCH_SA1100 dep_bool ' HuW WebPanel' CONFIG_SA1100_HUW_WEBPANEL $CONFIG_ARCH_SA1100 @@ -117,29 +127,18 @@ dep_bool ' Shannon' CONFIG_SA1100_SHANNON $CONFIG_ARCH_SA1100 dep_bool ' Sherman' CONFIG_SA1100_SHERMAN $CONFIG_ARCH_SA1100 dep_bool ' Simpad' CONFIG_SA1100_SIMPAD $CONFIG_ARCH_SA1100 +dep_bool ' Simputer' CONFIG_SA1100_SIMPUTER $CONFIG_ARCH_SA1100 dep_bool ' Tulsa' CONFIG_SA1100_PFS168 $CONFIG_ARCH_SA1100 dep_bool ' Victor' CONFIG_SA1100_VICTOR $CONFIG_ARCH_SA1100 dep_bool ' XP860' CONFIG_SA1100_XP860 $CONFIG_ARCH_SA1100 dep_bool ' Yopy' CONFIG_SA1100_YOPY $CONFIG_ARCH_SA1100 -# Determine if SA1111 support is required -if [ "$CONFIG_ASSABET_NEPONSET" = "y" -o \ - "$CONFIG_SA1100_JORNADA720" = "y" -o \ - "$CONFIG_SA1100_PFS168" = "y" -o \ - "$CONFIG_SA1100_XP860" = "y" -o \ - "$CONFIG_SA1100_GRAPHICSMASTER" = "y" -o \ - "$CONFIG_SA1100_PT_SYSTEM3" = "y" -o \ - "$CONFIG_SA1100_ADSBITSY" = "y" -o \ - "$CONFIG_SA1100_BADGE4" = "y" ]; then - define_bool CONFIG_SA1111 y - define_int CONFIG_FORCE_MAX_ZONEORDER 9 -fi +dep_tristate 'SA1100 USB function support' CONFIG_SA1100_USB $CONFIG_ARCH_SA1100 +dep_tristate ' Support for SA11x0 USB network link function' CONFIG_SA1100_USB_NETLINK $CONFIG_SA1100_USB +dep_tristate ' Support for SA11x0 USB character device emulation' CONFIG_SA1100_USB_CHAR $CONFIG_SA1100_USB -#dep_tristate 'SA1100 USB function support' CONFIG_SA1100_USB $CONFIG_ARCH_SA1100 -#dep_tristate ' Support for SA11x0 USB network link function' CONFIG_SA1100_USB_NETLINK $CONFIG_SA1100_USB -#dep_tristate ' Support for SA11x0 USB character device emulation' CONFIG_SA1100_USB_CHAR $CONFIG_SA1100_USB +dep_tristate 'SA1100 Generic PIO SSP support' CONFIG_SA1100_SSP $CONFIG_ARCH_SA1100 -dep_tristate 'Compaq iPAQ Handheld sleeve support' CONFIG_H3600_SLEEVE $CONFIG_SA1100_H3600 endmenu mainmenu_option next_comment @@ -148,8 +147,10 @@ dep_bool ' CDB89712' CONFIG_ARCH_CDB89712 $CONFIG_ARCH_CLPS711X dep_bool ' CLEP7312' CONFIG_ARCH_CLEP7312 $CONFIG_ARCH_CLPS711X dep_bool ' EDB7211' CONFIG_ARCH_EDB7211 $CONFIG_ARCH_CLPS711X -dep_bool ' P720T' CONFIG_ARCH_P720T $CONFIG_ARCH_CLPS711X dep_bool ' FORTUNET' CONFIG_ARCH_FORTUNET $CONFIG_ARCH_CLPS711X +dep_bool ' GUIDEA07' CONFIG_ARCH_GUIDEA07 $CONFIG_ARCH_CLPS711X +dep_bool ' P720T' CONFIG_ARCH_P720T $CONFIG_ARCH_CLPS711X + # XXX Maybe these should indicate register compatibility # instead of being mutually exclusive. @@ -168,6 +169,11 @@ "$CONFIG_ARCH_EP7212" = "y" ]; then bool ' EP72xx ROM boot' CONFIG_EP72XX_ROM_BOOT fi + +if [ "$CONFIG_ARCH_GUIDEA07" = "y"]; then + define_bool CONFIG_ARCH_CDB89712 y +fi + endmenu # Definitions to make life easier @@ -178,7 +184,8 @@ define_bool CONFIG_ARCH_ACORN n fi -# see Documentation/arm/ConfigVars for a description of these +##################################################################### +# Footbridge support if [ "$CONFIG_ARCH_CO285" = "y" -o \ "$CONFIG_ARCH_FOOTBRIDGE" = "y" ]; then define_bool CONFIG_FOOTBRIDGE y @@ -204,6 +211,42 @@ define_bool CONFIG_ARCH_EBSA285 y fi +##################################################################### +# SA1111 support +if [ "$CONFIG_ASSABET_NEPONSET" = "y" -o \ + "$CONFIG_SA1100_ACCELENT" = "y" -o \ + "$CONFIG_SA1100_ADSAGC" = "y" -o \ + "$CONFIG_SA1100_ADSBITSY" = "y" -o \ + "$CONFIG_SA1100_ADSBITSYPLUS" = "y" -o \ + "$CONFIG_SA1100_BADGE4" = "y" -o \ + "$CONFIG_SA1100_CONSUS" = "y" -o \ + "$CONFIG_SA1100_GRAPHICSMASTER" = "y" -o \ + "$CONFIG_SA1100_JORNADA720" = "y" -o \ + "$CONFIG_SA1100_PFS168" = "y" -o \ + "$CONFIG_SA1100_PT_SYSTEM3" = "y" -o \ + "$CONFIG_SA1100_XP860" = "y" ]; then + define_bool CONFIG_SA1111 y + define_int CONFIG_FORCE_MAX_ZONEORDER 9 +fi + +if [ "$CONFIG_ARCH_CAMELOT" = "y" ]; then + choice 'EPXA system type' \ + "EPXA10DB CONFIG_EPXA10DB \ + EPXA1DB CONFIG_EPXA1DB" EPXA10DB + +if [ "$CONFIG_EPXA10DB" = "y" ]; then + choice 'EPXA10B version' \ + "SDR CONFIG_EPXA10DB_R2 \ + DDR CONFIG_EPXA10DB_R3" DDR +fi + + comment 'PLD hotswap support' + define_bool CONFIG_PLD y + dep_bool 'Support for PLD device hotplugging' CONFIG_PLD_HOTSWAP $CONFIG_EXPERIMENTAL +fi + +comment 'Processor Type' + # Figure out whether this system uses 26-bit or 32-bit CPUs. if [ "$CONFIG_ARCH_ARCA5K" = "y" ]; then define_bool CONFIG_CPU_32 n @@ -214,28 +257,9 @@ define_bool CONFIG_CPU_26 n fi -comment 'Processor Type' - -# Firstly, figure out what processor architecture version we should be using. -if [ "$CONFIG_ARCH_RPC" = "y" -o "$CONFIG_ARCH_CLPS7500" = "y" ]; then - define_bool CONFIG_CPU_32v3 y -else - define_bool CONFIG_CPU_32v3 n -fi -if [ "$CONFIG_ARCH_EBSA110" = "y" -o "$CONFIG_FOOTBRIDGE" = "y" -o \ - "$CONFIG_ARCH_TBOX" = "y" -o "$CONFIG_ARCH_SHARK" = "y" -o \ - "$CONFIG_ARCH_NEXUSPCI" = "y" -o "$CONFIG_ARCH_CLPS711X" = "y" -o \ - "$CONFIG_ARCH_INTEGRATOR" = "y" -o "$CONFIG_ARCH_SA1100" = "y" -o \ - "$CONFIG_ARCH_L7200" = "y" -o "$CONFIG_ARCH_ANAKIN" = "y" -o \ - "$CONFIG_ARCH_CAMELOT" = "y" -o "$CONFIG_ARCH_MX1ADS" = "y" ]; then - define_bool CONFIG_CPU_32v4 y -else - define_bool CONFIG_CPU_32v4 n -fi - -# Select CPU types depending on the architecture selected. -# We use this to select which CPUs are supported, and to select -# the compiler tuning options. +# Select CPU types depending on the architecture selected. This selects +# which CPUs we support in the kernel image, and the compiler instruction +# optimiser behaviour. # ARM610 if [ "$CONFIG_ARCH_RPC" = "y" ]; then @@ -245,7 +269,8 @@ fi # ARM710 -if [ "$CONFIG_ARCH_CLPS7500" = "y" ]; then +if [ "$CONFIG_ARCH_CLPS7500" = "y" -o \ + "$CONFIG_ARCH_RISCSTATION" = "y" ]; then define_bool CONFIG_CPU_ARM710 y else if [ "$CONFIG_ARCH_RPC" = "y" ]; then @@ -272,7 +297,9 @@ if [ "$CONFIG_ARCH_MX1ADS" = "y" ]; then define_bool CONFIG_CPU_ARM920T y else - if [ "$CONFIG_ARCH_INTEGRATOR" = "y" ]; then + if [ "$CONFIG_ARCH_INTEGRATOR" = "y" -o \ + "$CONFIG_ARCH_OMAHA" = "y" -o \ + "$CONFIG_ARCH_AT91RM9200DK" = "y" ]; then bool 'Support ARM920T processor' CONFIG_CPU_ARM920T else define_bool CONFIG_CPU_ARM920T n @@ -280,28 +307,13 @@ fi -if [ "$CONFIG_CPU_ARM920T" = "y" ]; then - bool ' ARM920T CPU idle' CONFIG_CPU_ARM920_CPU_IDLE - bool ' ARM920T I-Cache on' CONFIG_CPU_ARM920_I_CACHE_ON - bool ' ARM920T D-Cache on' CONFIG_CPU_ARM920_D_CACHE_ON - if [ "$CONFIG_CPU_ARM920_D_CACHE_ON" = "y" ] ; then - bool ' Force write through caches on ARM920T' CONFIG_CPU_ARM920_WRITETHROUGH - fi -fi - # ARM922T if [ "$CONFIG_ARCH_CAMELOT" = "y" ]; then - define_bool CONFIG_CPU_ARM922T y + define_bool CONFIG_CPU_ARM922T y + define_bool CONFIG_PLD y else - define_bool CONFIG_CPU_ARM922T n -fi -if [ "$CONFIG_CPU_ARM922T" = "y" ]; then - bool ' ARM922T CPU idle' CONFIG_CPU_ARM922_CPU_IDLE - bool ' ARM922T I-Cache on' CONFIG_CPU_ARM922_I_CACHE_ON - bool ' ARM922T D-Cache on' CONFIG_CPU_ARM922_D_CACHE_ON - if [ "$CONFIG_CPU_ARM922_D_CACHE_ON" = "y" ] ; then - bool ' Force write through caches on ARM922T' CONFIG_CPU_ARM922_WRITETHROUGH - fi + define_bool CONFIG_CPU_ARM922T n + define_bool CONFIG_PLD n fi # ARM926T @@ -310,18 +322,6 @@ else define_bool CONFIG_CPU_ARM926T n fi -if [ "$CONFIG_CPU_ARM926T" = "y" ]; then - bool ' ARM926T CPU idle' CONFIG_CPU_ARM926_CPU_IDLE - bool ' ARM926T I-Cache on' CONFIG_CPU_ARM926_I_CACHE_ON - bool ' ARM926T D-Cache on' CONFIG_CPU_ARM926_D_CACHE_ON - if [ "$CONFIG_CPU_ARM926_D_CACHE_ON" = "y" ] ; then - bool ' Force write through caches on ARM926T' CONFIG_CPU_ARM926_WRITETHROUGH - fi - if [ "$CONFIG_CPU_ARM926_I_CACHE_ON" = "y" -o \ - "$CONFIG_CPU_ARM926_D_CACHE_ON" = "y" ]; then - bool ' Round robin I and D cache replacement algorithm' CONFIG_CPU_ARM926_ROUND_ROBIN - fi -fi # ARM1020 if [ "$CONFIG_ARCH_INTEGRATOR" = "y" ]; then @@ -329,16 +329,12 @@ else define_bool CONFIG_CPU_ARM1020 n fi -if [ "$CONFIG_CPU_ARM1020" = "y" ]; then - bool ' ARM1020 I-Cache on' CONFIG_CPU_ARM1020_I_CACHE_ON - bool ' ARM10 D-Cache on' CONFIG_CPU_ARM1020_D_CACHE_ON - if [ "$CONFIG_CPU_ARM1020_D_CACHE_ON" = "y" ] ; then - bool ' Force write through caches on ARM1020' CONFIG_CPU_ARM1020_FORCE_WRITE_THROUGH - fi - if [ "$CONFIG_CPU_ARM1020_I_CACHE_ON" = "y" -o \ - "$CONFIG_CPU_ARM1020_D_CACHE_ON" = "y" ]; then - bool ' Round robin I and D cache replacement algorithm' CONFIG_CPU_ARM1020_ROUND_ROBIN - fi + +# ARM1026EJ-S +if [ "$CONFIG_ARCH_INTEGRATOR" = "y" ]; then + bool 'Support ARM1026EJ-S processor' CONFIG_CPU_ARM1026 +else + define_bool CONFIG_CPU_ARM1026 n fi # SA110 @@ -361,13 +357,59 @@ define_bool CONFIG_CPU_SA1100 n fi -if [ "$CONFIG_CPU_32" = "y" ]; then - dep_bool 'Support Thumb instructions (experimental)' CONFIG_ARM_THUMB $CONFIG_EXPERIMENTAL +# Figure out what processor architecture version we should be using. +# This defines the compiler instruction set which depends on the machine type. + +if [ "$CONFIG_ARCH_RPC" = "y" -o \ + "$CONFIG_ARCH_RISCSTATION" = "y" -o \ + "$CONFIG_ARCH_CLPS7500" = "y" ]; then + define_bool CONFIG_CPU_32v3 y +else + define_bool CONFIG_CPU_32v3 n +fi +if [ "$CONFIG_ARCH_EBSA110" = "y" -o "$CONFIG_FOOTBRIDGE" = "y" -o \ + "$CONFIG_ARCH_TBOX" = "y" -o "$CONFIG_ARCH_SHARK" = "y" -o \ + "$CONFIG_ARCH_NEXUSPCI" = "y" -o "$CONFIG_ARCH_CLPS711X" = "y" -o \ + "$CONFIG_ARCH_INTEGRATOR" = "y" -o "$CONFIG_ARCH_SA1100" = "y" -o \ + "$CONFIG_ARCH_L7200" = "y" -o "$CONFIG_ARCH_ANAKIN" = "y" -o \ + "$CONFIG_ARCH_CAMELOT" = "y" -o "$CONFIG_ARCH_MX1ADS" = "y" -o \ + "$CONFIG_ARCH_OMAHA" = "y" -o "$CONFIG_ARCH_AT91RM9200DK" = "y" ]; then + define_bool CONFIG_CPU_32v4 y +else + define_bool CONFIG_CPU_32v4 n fi +comment 'Processor Features' + +if [ "$CONFIG_CPU_ARM720T" = "y" -o "$CONFIG_CPU_ARM920T" = "y" -o \ + "$CONFIG_CPU_ARM922T" = "y" -o "$CONFIG_CPU_ARM926T" = "y" -o \ + "$CONFIG_CPU_ARM1020" = "y" -o "$CONFIG_CPU_ARM1026" = "y" ]; then + dep_bool 'Support Thumb instructions (EXPERIMENTAL)' CONFIG_ARM_THUMB $CONFIG_EXPERIMENTAL +fi +if [ "$CONFIG_CPU_ARM920T" = "y" -o "$CONFIG_CPU_ARM922T" = "y" -o \ + "$CONFIG_CPU_ARM926T" = "y" -o "$CONFIG_CPU_ARM1020" = "y" -o \ + "$CONFIG_CPU_ARM1026" = "y" ]; then + bool 'Disable I-Cache' CONFIG_CPU_ICACHE_DISABLE + bool 'Disable D-Cache' CONFIG_CPU_DCACHE_DISABLE + if [ "$CONFIG_CPU_DISABLE_DCACHE" = "n" ]; then + bool 'Force write through D-cache' CONFIG_CPU_DCACHE_WRITETHROUGH + fi +fi +if [ "$CONFIG_CPU_ARM926T" = "y" -o "$CONFIG_CPU_ARM1020" = "y" -o \ + "$CONFIG_CPU_ARM1026" = "y" ]; then + if [ "$CONFIG_CPU_ICACHE_DISABLE" = "n" -o "$CONFIG_CPU_DCACHE_DISABLE" = "n" ]; then + bool 'Round robin I and D cache replacement algorithm' CONFIG_CPU_CACHE_ROUND_ROBIN + fi +fi +if [ "$CONFIG_CPU_ARM1020" = "y" -o "$CONFIG_CPU_ARM1026" = "y" ]; then + bool 'Disable branch prediction' CONFIG_CPU_BPREDICT_DISABLE +fi + + # Select various configuration options depending on the machine type if [ "$CONFIG_ARCH_EDB7211" = "y" -o \ - "$CONFIG_ARCH_SA1100" = "y" ]; then + "$CONFIG_ARCH_SA1100" = "y" -o \ + "$CONFIG_ARCH_RISCSTATION" = "y" ]; then define_bool CONFIG_DISCONTIGMEM y else define_bool CONFIG_DISCONTIGMEM n @@ -395,6 +437,7 @@ if [ "$CONFIG_FOOTBRIDGE_HOST" = "y" -o \ "$CONFIG_ARCH_SHARK" = "y" -o \ "$CONFIG_ARCH_CLPS7500" = "y" -o \ + "$CONFIG_ARCH_RISCSTATION" = "y" -o \ "$CONFIG_ARCH_EBSA110" = "y" -o \ "$CONFIG_ARCH_CDB89712" = "y" -o \ "$CONFIG_ARCH_EDB7211" = "y" -o \ @@ -411,25 +454,38 @@ define_bool CONFIG_ISA_DMA n fi +# Compressed boot loader in ROM. Yes, we really want to ask about +# TEXT and BSS so we preserve their values in the config files. +bool 'Compressed boot loader in ROM/flash' CONFIG_ZBOOT_ROM +hex 'Compressed ROM boot loader base address' CONFIG_ZBOOT_ROM_TEXT 0 +hex 'Compressed ROM boot loader BSS address' CONFIG_ZBOOT_ROM_BSS 0 + if [ "$CONFIG_ARCH_SA1100" = "y" -o \ "$CONFIG_ARCH_INTEGRATOR" = "y" ]; then dep_bool 'Support CPU clock change (EXPERIMENTAL)' CONFIG_CPU_FREQ $CONFIG_EXPERIMENTAL fi source drivers/pci/Config.in -bool 'Support hot-pluggable devices' CONFIG_HOTPLUG +bool 'Support for hot-pluggable devices' CONFIG_HOTPLUG if [ "$CONFIG_HOTPLUG" = "y" ]; then source drivers/pcmcia/Config.in else define_bool CONFIG_PCMCIA n fi +if [ "$CONFIG_SA1100_ACCELENT" = "y" ]; then + if [ "$CONFIG_PCMCIA" != "n" ]; then + bool ' Use second PCMCIA/CF slot (disables on-board IDE)' CONFIG_SA_PCMCIA_SLOT_1 y + fi +fi bool 'Networking support' CONFIG_NET bool 'System V IPC' CONFIG_SYSVIPC bool 'BSD Process Accounting' CONFIG_BSD_PROCESS_ACCT bool 'Sysctl support' CONFIG_SYSCTL comment 'At least one math emulation must be selected' tristate 'NWFPE math emulation' CONFIG_FPE_NWFPE -dep_tristate 'FastFPE math emulation (experimental)' CONFIG_FPE_FASTFPE $CONFIG_EXPERIMENTAL +if [ "$CONFIG_CPU_26" = "n" -a "$CONFIG_CPU_32v3" = "n" ]; then + dep_tristate 'FastFPE math emulation (EXPERIMENTAL)' CONFIG_FPE_FASTFPE $CONFIG_EXPERIMENTAL +fi choice 'Kernel core (/proc/kcore) format' \ "ELF CONFIG_KCORE_ELF \ A.OUT CONFIG_KCORE_AOUT" ELF @@ -438,19 +494,8 @@ tristate 'Kernel support for MISC binaries' CONFIG_BINFMT_MISC dep_bool 'Power Management support (experimental)' CONFIG_PM $CONFIG_EXPERIMENTAL dep_tristate 'RISC OS personality' CONFIG_ARTHUR $CONFIG_CPU_32 +string 'Default kernel command string' CONFIG_CMDLINE "" -if [ "$CONFIG_ARCH_EBSA110" = "y" -o \ - "$CONFIG_ARCH_SA1100" = "y" -o \ - "$CONFIG_ARCH_CLPS7500" = "y" -o \ - "$CONFIG_ARCH_PERSONAL_SERVER" = "y" -o \ - "$CONFIG_ARCH_CATS" = "y" -o \ - "$CONFIG_ARCH_P720T" = "y" -o \ - "$CONFIG_ARCH_CDB89712" = "y" -o \ - "$CONFIG_ARCH_CAMELOT" = "y" -o \ - "$CONFIG_ARCH_ANAKIN" = "y" -o \ - "$CONFIG_ARCH_MX1ADS" = "y" ]; then - string 'Default kernel command string' CONFIG_CMDLINE "" -fi if [ "$CONFIG_ARCH_NETWINDER" = "y" -o \ "$CONFIG_ARCH_EBSA110" = "y" -o \ "$CONFIG_ARCH_EBSA285" = "y" -o \ @@ -460,7 +505,8 @@ "$CONFIG_ARCH_SA1100" = "y" -o \ "$CONFIG_ARCH_INTEGRATOR" = "y" -o \ "$CONFIG_ARCH_CDB89712" = "y" -o \ - "$CONFIG_ARCH_P720T" = "y" ]; then + "$CONFIG_ARCH_P720T" = "y" -o \ + "$CONFIG_ARCH_OMAHA" = "y" ]; then bool 'Timer and CPU usage LEDs' CONFIG_LEDS if [ "$CONFIG_LEDS" = "y" ]; then if [ "$CONFIG_ARCH_NETWINDER" = "y" -o \ @@ -469,17 +515,24 @@ "$CONFIG_ARCH_CO285" = "y" -o \ "$CONFIG_ARCH_SA1100" = "y" -o \ "$CONFIG_ARCH_INTEGRATOR" = "y" -o \ - "$CONFIG_ARCH_P720T" = "y" ]; then + "$CONFIG_ARCH_P720T" = "y" -o \ + "$CONFIG_ARCH_OMAHA" = "y" ]; then bool ' Timer LED' CONFIG_LEDS_TIMER bool ' CPU usage LED' CONFIG_LEDS_CPU fi - fi - if [ "$CONFIG_ARCH_EBSA110" = "y" ]; then - define_bool CONFIG_LEDS_TIMER y + if [ "$CONFIG_ARCH_EBSA110" = "y" ]; then + define_bool CONFIG_LEDS_TIMER y + fi fi fi -if [ "$CONFIG_CPU_32" = "y" -a "$CONFIG_ARCH_EBSA110" != "y" ]; then - bool 'Kernel-mode alignment trap handler' CONFIG_ALIGNMENT_TRAP +if [ "$CONFIG_ARCH_SA1100" = "y" ]; then + define_bool CONFIG_ALIGNMENT_TRAP y +else + if [ "$CONFIG_CPU_32" = "y" -a "$CONFIG_ARCH_EBSA110" != "y" ]; then + bool 'Kernel-mode alignment trap handler' CONFIG_ALIGNMENT_TRAP + else + define_bool CONFIG_ALIGNMENT_TRAP n + fi fi endmenu @@ -487,6 +540,8 @@ if [ "$CONFIG_ALIGNMENT_TRAP" = "y" ]; then source drivers/mtd/Config.in +else + define_bool CONFIG_MTD n fi source drivers/pnp/Config.in @@ -503,7 +558,7 @@ mainmenu_option next_comment comment 'Network device support' - bool 'Network device support?' CONFIG_NETDEVICES + bool 'Network device support' CONFIG_NETDEVICES if [ "$CONFIG_NETDEVICES" = "y" ]; then source drivers/net/Config.in fi @@ -515,9 +570,9 @@ fi mainmenu_option next_comment -comment 'ATA/IDE/MFM/RLL support' +comment 'ATA/ATAPI/MFM/RLL support' -tristate 'ATA/IDE/MFM/RLL support' CONFIG_IDE +tristate 'ATA/ATAPI/MFM/RLL support' CONFIG_IDE if [ "$CONFIG_IDE" != "n" ]; then source drivers/ide/Config.in @@ -530,7 +585,7 @@ mainmenu_option next_comment comment 'SCSI support' -tristate 'SCSI support?' CONFIG_SCSI +tristate 'SCSI support' CONFIG_SCSI if [ "$CONFIG_SCSI" != "n" ]; then source drivers/scsi/Config.in @@ -602,7 +657,7 @@ "$CONFIG_ARCH_MX1ADS" = "y" ]; then define_bool CONFIG_PC_KEYMAP y fi - if [ "$CONFIG_ARCH_ACORN" != "y" -a "$CONFIG_ARCH_EBSA110" != "y" ]; then + if [ "$CONFIG_ARCH_ACORN" != "y" -a "$CONFIG_ARCH_EBSA110" != "y" -a "$CONFIG_ARCH_RISCSTATION" != "y" ]; then bool 'VGA text console' CONFIG_VGA_CONSOLE fi source drivers/video/Config.in @@ -611,6 +666,7 @@ if [ "$CONFIG_ARCH_ACORN" = "y" -o \ "$CONFIG_ARCH_CLPS7500" = "y" -o \ + "$CONFIG_ARCH_RISCSTATION" = "y" -o \ "$CONFIG_ARCH_TBOX" = "y" -o \ "$CONFIG_ARCH_SHARK" = "y" -o \ "$CONFIG_ARCH_SA1100" = "y" -o \ @@ -629,7 +685,9 @@ source drivers/usb/Config.in -source net/bluetooth/Config.in +if [ "$CONFIG_EXPERIMENTAL" = "y" ]; then + source net/bluetooth/Config.in +fi mainmenu_option next_comment comment 'Kernel hacking' @@ -638,7 +696,16 @@ # you know what you are doing and are willing to live without stack traces, you # can get a slightly smaller kernel by setting this option to n, but then RMK # will have to kill you ;). -define_bool CONFIG_FRAME_POINTER y +if [ "$CONFIG_ARCH_OMAHA" = "y" ]; then + # + # Note: We want all the performance we can get, so this means + # we accept the debugging limitations of setting CONFIG_FRAME_POINTER=n + # -- ahaigh@arm.com (23 August 2002) + define_bool CONFIG_FRAME_POINTER n +else + define_bool CONFIG_FRAME_POINTER y +fi + bool 'Verbose user fault messages' CONFIG_DEBUG_USER bool 'Include debugging information in kernel binary' CONFIG_DEBUG_INFO dep_bool 'Disable pgtable cache' CONFIG_NO_PGT_CACHE $CONFIG_CPU_26 @@ -650,7 +717,7 @@ dep_bool ' Wait queue debugging' CONFIG_DEBUG_WAITQ $CONFIG_DEBUG_KERNEL dep_bool ' Verbose BUG() reporting (adds 70K)' CONFIG_DEBUG_BUGVERBOSE $CONFIG_DEBUG_KERNEL dep_bool ' Verbose kernel error messages' CONFIG_DEBUG_ERRORS $CONFIG_DEBUG_KERNEL -# These options are only for real kernel hackers who want to get their hands dirty. +# These options are only for real kernel hackers who want to get their hands dirty. dep_bool ' Kernel low-level debugging functions' CONFIG_DEBUG_LL $CONFIG_DEBUG_KERNEL dep_bool ' Kernel low-level debugging messages via footbridge serial port' CONFIG_DEBUG_DC21285_PORT $CONFIG_DEBUG_LL $CONFIG_FOOTBRIDGE dep_bool ' Kernel low-level debugging messages via UART2' CONFIG_DEBUG_CLPS711X_UART2 $CONFIG_DEBUG_LL $CONFIG_ARCH_CLPS711X diff -urN orig/arch/arm/def-configs/accelent_sa linux/arch/arm/def-configs/accelent_sa --- orig/arch/arm/def-configs/accelent_sa Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/def-configs/accelent_sa Mon Aug 5 22:25:08 2002 @@ -0,0 +1,1099 @@ +# +# Automatically generated make config: don't edit +# +CONFIG_ARM=y +# CONFIG_EISA is not set +# CONFIG_SBUS is not set +# CONFIG_MCA is not set +CONFIG_UID16=y +CONFIG_RWSEM_GENERIC_SPINLOCK=y +# CONFIG_RWSEM_XCHGADD_ALGORITHM is not set +# CONFIG_GENERIC_BUST_SPINLOCK is not set +# CONFIG_GENERIC_ISA_DMA is not set + +# +# Code maturity level options +# +CONFIG_EXPERIMENTAL=y +# CONFIG_OBSOLETE is not set + +# +# Loadable module support +# +CONFIG_MODULES=y +# CONFIG_MODVERSIONS is not set +CONFIG_KMOD=y + +# +# System Type +# +# CONFIG_ARCH_ANAKIN is not set +# CONFIG_ARCH_ARCA5K is not set +# CONFIG_ARCH_CLPS7500 is not set +# CONFIG_ARCH_CLPS711X is not set +# CONFIG_ARCH_CO285 is not set +# CONFIG_ARCH_EBSA110 is not set +# CONFIG_ARCH_CAMELOT is not set +# CONFIG_ARCH_FOOTBRIDGE is not set +# CONFIG_ARCH_INTEGRATOR is not set +# CONFIG_ARCH_L7200 is not set +# CONFIG_ARCH_MX1ADS is not set +# CONFIG_ARCH_RPC is not set +CONFIG_ARCH_SA1100=y +# CONFIG_ARCH_SHARK is not set + +# +# Archimedes/A5000 Implementations +# + +# +# Archimedes/A5000 Implementations (select only ONE) +# +# CONFIG_ARCH_ARC is not set +# CONFIG_ARCH_A5K is not set + +# +# Footbridge Implementations +# +# CONFIG_ARCH_CATS is not set +# CONFIG_ARCH_PERSONAL_SERVER is not set +# CONFIG_ARCH_EBSA285_ADDIN is not set +# CONFIG_ARCH_EBSA285_HOST is not set +# CONFIG_ARCH_NETWINDER is not set + +# +# SA11x0 Implementations +# +CONFIG_SA1100_ACCELENT=y +# CONFIG_SA1100_ASSABET is not set +# CONFIG_ASSABET_NEPONSET is not set +# CONFIG_SA1100_ADSBITSY is not set +# CONFIG_SA1100_BRUTUS is not set +# CONFIG_SA1100_CEP is not set +# CONFIG_SA1100_CERF is not set +# CONFIG_SA1100_H3100 is not set +# CONFIG_SA1100_H3600 is not set +# CONFIG_SA1100_H3800 is not set +# CONFIG_SA1100_H3XXX is not set +# CONFIG_SA1100_EXTENEX1 is not set +# CONFIG_SA1100_FLEXANET is not set +# CONFIG_SA1100_FREEBIRD is not set +# CONFIG_SA1100_FRODO is not set +# CONFIG_SA1100_GRAPHICSCLIENT is not set +# CONFIG_SA1100_GRAPHICSMASTER is not set +# CONFIG_SA1100_BADGE4 is not set +# CONFIG_SA1100_JORNADA720 is not set +# CONFIG_SA1100_HUW_WEBPANEL is not set +# CONFIG_SA1100_ITSY is not set +# CONFIG_SA1100_LART is not set +# CONFIG_SA1100_NANOENGINE is not set +# CONFIG_SA1100_OMNIMETER is not set +# CONFIG_SA1100_PANGOLIN is not set +# CONFIG_SA1100_PLEB is not set +# CONFIG_SA1100_PT_SYSTEM3 is not set +# CONFIG_SA1100_SHANNON is not set +# CONFIG_SA1100_SHERMAN is not set +# CONFIG_SA1100_SIMPAD is not set +# CONFIG_SA1100_PFS168 is not set +# CONFIG_SA1100_VICTOR is not set +# CONFIG_SA1100_XP860 is not set +# CONFIG_SA1100_YOPY is not set +CONFIG_SA1111=y +CONFIG_FORCE_MAX_ZONEORDER=9 +CONFIG_SA1100_USB=m +CONFIG_SA1100_USB_NETLINK=m +CONFIG_SA1100_USB_CHAR=m +# CONFIG_H3600_SLEEVE is not set + +# +# CLPS711X/EP721X Implementations +# +# CONFIG_ARCH_AUTCPU12 is not set +# CONFIG_ARCH_CDB89712 is not set +# CONFIG_ARCH_CLEP7312 is not set +# CONFIG_ARCH_EDB7211 is not set +# CONFIG_ARCH_P720T is not set +# CONFIG_ARCH_FORTUNET is not set +# CONFIG_ARCH_EP7211 is not set +# CONFIG_ARCH_EP7212 is not set +# CONFIG_ARCH_ACORN is not set +# CONFIG_FOOTBRIDGE is not set +# CONFIG_FOOTBRIDGE_HOST is not set +# CONFIG_FOOTBRIDGE_ADDIN is not set +CONFIG_CPU_32=y +# CONFIG_CPU_26 is not set + +# +# Processor Type +# +# CONFIG_CPU_32v3 is not set +CONFIG_CPU_32v4=y +# CONFIG_CPU_ARM610 is not set +# CONFIG_CPU_ARM710 is not set +# CONFIG_CPU_ARM720T is not set +# CONFIG_CPU_ARM920T is not set +# CONFIG_CPU_ARM922T is not set +# CONFIG_PLD is not set +# CONFIG_CPU_ARM926T is not set +# CONFIG_CPU_ARM1020 is not set +# CONFIG_CPU_SA110 is not set +CONFIG_CPU_SA1100=y +# CONFIG_ARM_THUMB is not set +CONFIG_DISCONTIGMEM=y + +# +# General setup +# +# CONFIG_PCI is not set +CONFIG_ISA=y +# CONFIG_ISA_DMA is not set +# CONFIG_ZBOOT_ROM is not set +CONFIG_ZBOOT_ROM_TEXT=C0400000 +CONFIG_ZBOOT_ROM_BSS=C0200000 +CONFIG_CPU_FREQ=y +CONFIG_HOTPLUG=y + +# +# PCMCIA/CardBus support +# +CONFIG_PCMCIA=m +CONFIG_PCMCIA_PROBE=y +# CONFIG_I82092 is not set +# CONFIG_I82365 is not set +# CONFIG_TCIC is not set +# CONFIG_PCMCIA_CLPS6700 is not set +CONFIG_PCMCIA_SA1100=m +# CONFIG_SA_PCMCIA_SLOT_1 is not set +CONFIG_NET=y +CONFIG_SYSVIPC=y +# CONFIG_BSD_PROCESS_ACCT is not set +CONFIG_SYSCTL=y + +# +# At least one math emulation must be selected +# +CONFIG_FPE_NWFPE=y +# CONFIG_FPE_FASTFPE is not set +CONFIG_KCORE_ELF=y +# CONFIG_KCORE_AOUT is not set +# CONFIG_BINFMT_AOUT is not set +CONFIG_BINFMT_ELF=y +# CONFIG_BINFMT_MISC is not set +CONFIG_PM=y +# CONFIG_ARTHUR is not set +CONFIG_CMDLINE="root=1f02" +CONFIG_LEDS=y +CONFIG_LEDS_TIMER=y +CONFIG_LEDS_CPU=y +CONFIG_ALIGNMENT_TRAP=y + +# +# Parallel port support +# +CONFIG_PARPORT=m +# CONFIG_PARPORT_PC is not set +CONFIG_PARPORT_IDP=m +# CONFIG_PARPORT_AMIGA is not set +# CONFIG_PARPORT_MFC3 is not set +# CONFIG_PARPORT_ATARI is not set +# CONFIG_PARPORT_GSC is not set +# CONFIG_PARPORT_SUNBPP is not set +# CONFIG_PARPORT_OTHER is not set +# CONFIG_PARPORT_1284 is not set + +# +# Memory Technology Devices (MTD) +# +CONFIG_MTD=y +# CONFIG_MTD_DEBUG is not set +CONFIG_MTD_PARTITIONS=y +# CONFIG_MTD_CONCAT is not set +# CONFIG_MTD_REDBOOT_PARTS is not set +# CONFIG_MTD_CMDLINE_PARTS is not set +# CONFIG_MTD_AFS_PARTS is not set + +# +# User Modules And Translation Layers +# +CONFIG_MTD_CHAR=y +CONFIG_MTD_BLOCK=y +# CONFIG_FTL is not set +# CONFIG_NFTL is not set + +# +# RAM/ROM/Flash chip drivers +# +CONFIG_MTD_CFI=y +# CONFIG_MTD_JEDECPROBE is not set +CONFIG_MTD_GEN_PROBE=y +CONFIG_MTD_CFI_ADV_OPTIONS=y +CONFIG_MTD_CFI_NOSWAP=y +# CONFIG_MTD_CFI_BE_BYTE_SWAP is not set +# CONFIG_MTD_CFI_LE_BYTE_SWAP is not set +CONFIG_MTD_CFI_GEOMETRY=y +# CONFIG_MTD_CFI_B1 is not set +# CONFIG_MTD_CFI_B2 is not set +CONFIG_MTD_CFI_B4=y +# CONFIG_MTD_CFI_B8 is not set +# CONFIG_MTD_CFI_I1 is not set +CONFIG_MTD_CFI_I2=y +# CONFIG_MTD_CFI_I4 is not set +# CONFIG_MTD_CFI_I8 is not set +CONFIG_MTD_CFI_INTELEXT=y +# CONFIG_MTD_CFI_AMDSTD is not set +# CONFIG_MTD_RAM is not set +# CONFIG_MTD_ROM is not set +# CONFIG_MTD_ABSENT is not set +# CONFIG_MTD_OBSOLETE_CHIPS is not set +# CONFIG_MTD_AMDSTD is not set +# CONFIG_MTD_SHARP is not set +# CONFIG_MTD_JEDEC is not set + +# +# Mapping drivers for chip access +# +# CONFIG_MTD_PHYSMAP is not set +# CONFIG_MTD_NORA is not set +# CONFIG_MTD_ARM_INTEGRATOR is not set +# CONFIG_MTD_CDB89712 is not set +CONFIG_MTD_SA1100=y +# CONFIG_MTD_DC21285 is not set +# CONFIG_MTD_IQ80310 is not set +# CONFIG_MTD_FORTUNET is not set +# CONFIG_MTD_EPXA10DB is not set +# CONFIG_MTD_AUTCPU12 is not set +# CONFIG_MTD_EDB7312 is not set +# CONFIG_MTD_IMPA7 is not set +# CONFIG_MTD_PCI is not set + +# +# Self-contained MTD device drivers +# +# CONFIG_MTD_PMC551 is not set +# CONFIG_MTD_SLRAM is not set +# CONFIG_MTD_MTDRAM is not set +# CONFIG_MTD_BLKMTD is not set + +# +# Disk-On-Chip Device Drivers +# +# CONFIG_MTD_DOC1000 is not set +# CONFIG_MTD_DOC2000 is not set +# CONFIG_MTD_DOC2001 is not set +# CONFIG_MTD_DOCPROBE is not set + +# +# NAND Flash Device Drivers +# +# CONFIG_MTD_NAND is not set + +# +# Plug and Play configuration +# +# CONFIG_PNP is not set +# CONFIG_ISAPNP is not set + +# +# Block devices +# +# CONFIG_BLK_DEV_FD is not set +# CONFIG_BLK_DEV_XD is not set +# CONFIG_PARIDE is not set +# CONFIG_BLK_CPQ_DA is not set +# CONFIG_BLK_CPQ_CISS_DA is not set +# CONFIG_BLK_DEV_DAC960 is not set +CONFIG_BLK_DEV_LOOP=y +# CONFIG_BLK_DEV_NBD is not set +CONFIG_BLK_DEV_RAM=y +CONFIG_BLK_DEV_RAM_SIZE=4096 +CONFIG_BLK_DEV_INITRD=y + +# +# Multi-device support (RAID and LVM) +# +# CONFIG_MD is not set +# CONFIG_BLK_DEV_MD is not set +# CONFIG_MD_LINEAR is not set +# CONFIG_MD_RAID0 is not set +# CONFIG_MD_RAID1 is not set +# CONFIG_MD_RAID5 is not set +# CONFIG_MD_MULTIPATH is not set +# CONFIG_BLK_DEV_LVM is not set + +# +# Networking options +# +CONFIG_PACKET=y +# CONFIG_PACKET_MMAP is not set +# CONFIG_NETLINK_DEV is not set +# CONFIG_NETFILTER is not set +# CONFIG_FILTER is not set +CONFIG_UNIX=y +CONFIG_INET=y +# CONFIG_IP_MULTICAST is not set +# CONFIG_IP_ADVANCED_ROUTER is not set +# CONFIG_IP_PNP is not set +# CONFIG_NET_IPIP is not set +# CONFIG_NET_IPGRE is not set +# CONFIG_ARPD is not set +# CONFIG_INET_ECN is not set +# CONFIG_SYN_COOKIES is not set +# CONFIG_IPV6 is not set +# CONFIG_KHTTPD is not set +# CONFIG_ATM is not set +# CONFIG_VLAN_8021Q is not set + +# +# +# +# CONFIG_IPX is not set +# CONFIG_ATALK is not set +# CONFIG_DECNET is not set +# CONFIG_BRIDGE is not set +# CONFIG_X25 is not set +# CONFIG_LAPB is not set +# CONFIG_LLC is not set +# CONFIG_NET_DIVERT is not set +# CONFIG_ECONET is not set +# CONFIG_WAN_ROUTER is not set +# CONFIG_NET_FASTROUTE is not set +# CONFIG_NET_HW_FLOWCONTROL is not set + +# +# QoS and/or fair queueing +# +# CONFIG_NET_SCHED is not set + +# +# Network device support +# +CONFIG_NETDEVICES=y + +# +# ARCnet devices +# +# CONFIG_ARCNET is not set +# CONFIG_DUMMY is not set +# CONFIG_BONDING is not set +# CONFIG_EQUALIZER is not set +# CONFIG_TUN is not set +# CONFIG_ETHERTAP is not set + +# +# Ethernet (10 or 100Mbit) +# +CONFIG_NET_ETHERNET=y +# CONFIG_ARM_AM79C961A is not set +# CONFIG_SUNLANCE is not set +# CONFIG_SUNBMAC is not set +# CONFIG_SUNQE is not set +# CONFIG_SUNGEM is not set +# CONFIG_NET_VENDOR_3COM is not set +# CONFIG_LANCE is not set +# CONFIG_NET_VENDOR_SMC is not set +# CONFIG_NET_VENDOR_RACAL is not set +# CONFIG_AT1700 is not set +# CONFIG_DEPCA is not set +# CONFIG_HP100 is not set +# CONFIG_NET_ISA is not set +# CONFIG_NET_PCI is not set +# CONFIG_NET_POCKET is not set +CONFIG_CS89x0=y + +# +# Ethernet (1000 Mbit) +# +# CONFIG_ACENIC is not set +# CONFIG_DL2K is not set +# CONFIG_MYRI_SBUS is not set +# CONFIG_NS83820 is not set +# CONFIG_HAMACHI is not set +# CONFIG_YELLOWFIN is not set +# CONFIG_SK98LIN is not set +# CONFIG_FDDI is not set +# CONFIG_HIPPI is not set +# CONFIG_PLIP is not set +# CONFIG_PPP is not set +# CONFIG_SLIP is not set + +# +# Wireless LAN (non-hamradio) +# +# CONFIG_NET_RADIO is not set + +# +# Token Ring devices +# +# CONFIG_TR is not set +# CONFIG_NET_FC is not set +# CONFIG_RCPCI is not set +# CONFIG_SHAPER is not set + +# +# Wan interfaces +# +# CONFIG_WAN is not set + +# +# PCMCIA network device support +# +CONFIG_NET_PCMCIA=y +# CONFIG_PCMCIA_3C589 is not set +# CONFIG_PCMCIA_3C574 is not set +# CONFIG_PCMCIA_FMVJ18X is not set +CONFIG_PCMCIA_PCNET=m +# CONFIG_PCMCIA_AXNET is not set +# CONFIG_PCMCIA_NMCLAN is not set +# CONFIG_PCMCIA_SMC91C92 is not set +# CONFIG_PCMCIA_XIRC2PS is not set +# CONFIG_ARCNET_COM20020_CS is not set +# CONFIG_PCMCIA_IBMTR is not set +# CONFIG_NET_PCMCIA_RADIO is not set + +# +# Amateur Radio support +# +# CONFIG_HAMRADIO is not set + +# +# IrDA (infrared) support +# +CONFIG_IRDA=m + +# +# IrDA protocols +# +CONFIG_IRLAN=m +# CONFIG_IRNET is not set +CONFIG_IRCOMM=m +# CONFIG_IRDA_ULTRA is not set + +# +# IrDA options +# +# CONFIG_IRDA_CACHE_LAST_LSAP is not set +# CONFIG_IRDA_FAST_RR is not set +# CONFIG_IRDA_DEBUG is not set + +# +# Infrared-port device drivers +# + +# +# SIR device drivers +# +CONFIG_IRTTY_SIR=m +CONFIG_IRPORT_SIR=m + +# +# Dongle support +# +# CONFIG_DONGLE is not set + +# +# FIR device drivers +# +# CONFIG_USB_IRDA is not set +# CONFIG_NSC_FIR is not set +# CONFIG_WINBOND_FIR is not set +# CONFIG_TOSHIBA_FIR is not set +# CONFIG_SMC_IRCC_FIR is not set +# CONFIG_ALI_FIR is not set +# CONFIG_VLSI_FIR is not set +CONFIG_SA1100_FIR=m + +# +# ATA/IDE/MFM/RLL support +# +# CONFIG_IDE is not set +# CONFIG_BLK_DEV_IDE_MODES is not set +# CONFIG_BLK_DEV_HD is not set + +# +# SCSI support +# +CONFIG_SCSI=y + +# +# SCSI support type (disk, tape, CD-ROM) +# +CONFIG_BLK_DEV_SD=m +CONFIG_SD_EXTRA_DEVS=40 +# CONFIG_CHR_DEV_ST is not set +# CONFIG_CHR_DEV_OSST is not set +# CONFIG_BLK_DEV_SR is not set +CONFIG_CHR_DEV_SG=m + +# +# Some SCSI devices (e.g. CD jukebox) support multiple LUNs +# +# CONFIG_SCSI_DEBUG_QUEUES is not set +# CONFIG_SCSI_MULTI_LUN is not set +# CONFIG_SCSI_CONSTANTS is not set +# CONFIG_SCSI_LOGGING is not set + +# +# SCSI low-level drivers +# +# CONFIG_SCSI_7000FASST is not set +# CONFIG_SCSI_ACARD is not set +# CONFIG_SCSI_AHA152X is not set +# CONFIG_SCSI_AHA1542 is not set +# CONFIG_SCSI_AHA1740 is not set +# CONFIG_SCSI_AACRAID is not set +# CONFIG_SCSI_AIC7XXX is not set +# CONFIG_SCSI_AIC7XXX_OLD is not set +# CONFIG_SCSI_DPT_I2O is not set +# CONFIG_SCSI_ADVANSYS is not set +# CONFIG_SCSI_IN2000 is not set +# CONFIG_SCSI_AM53C974 is not set +# CONFIG_SCSI_MEGARAID is not set +# CONFIG_SCSI_BUSLOGIC is not set +# CONFIG_SCSI_DMX3191D is not set +# CONFIG_SCSI_DTC3280 is not set +# CONFIG_SCSI_EATA is not set +# CONFIG_SCSI_EATA_DMA is not set +# CONFIG_SCSI_EATA_PIO is not set +# CONFIG_SCSI_FUTURE_DOMAIN is not set +# CONFIG_SCSI_GDTH is not set +# CONFIG_SCSI_GENERIC_NCR5380 is not set +# CONFIG_SCSI_INITIO is not set +# CONFIG_SCSI_INIA100 is not set +# CONFIG_SCSI_PPA is not set +# CONFIG_SCSI_IMM is not set +# CONFIG_SCSI_NCR53C406A is not set +# CONFIG_SCSI_NCR53C7xx is not set +# CONFIG_SCSI_PAS16 is not set +# CONFIG_SCSI_PCI2000 is not set +# CONFIG_SCSI_PCI2220I is not set +# CONFIG_SCSI_PSI240I is not set +# CONFIG_SCSI_QLOGIC_FAS is not set +# CONFIG_SCSI_SIM710 is not set +# CONFIG_SCSI_SYM53C416 is not set +# CONFIG_SCSI_T128 is not set +# CONFIG_SCSI_U14_34F is not set +# CONFIG_SCSI_DEBUG is not set + +# +# PCMCIA SCSI adapter support +# +# CONFIG_SCSI_PCMCIA is not set + +# +# I2O device support +# +# CONFIG_I2O is not set +# CONFIG_I2O_BLOCK is not set +# CONFIG_I2O_LAN is not set +# CONFIG_I2O_SCSI is not set +# CONFIG_I2O_PROC is not set + +# +# ISDN subsystem +# +# CONFIG_ISDN is not set + +# +# Input core support +# +CONFIG_INPUT=y +CONFIG_INPUT_KEYBDEV=y +CONFIG_INPUT_MOUSEDEV=y +CONFIG_INPUT_MOUSEDEV_SCREEN_X=1024 +CONFIG_INPUT_MOUSEDEV_SCREEN_Y=768 +# CONFIG_INPUT_JOYDEV is not set +CONFIG_INPUT_EVDEV=y + +# +# Character devices +# +CONFIG_VT=y +CONFIG_VT_CONSOLE=y +CONFIG_SERIAL=y +# CONFIG_SERIAL_CONSOLE is not set +# CONFIG_SERIAL_EXTENDED is not set +# CONFIG_SERIAL_NONSTANDARD is not set + +# +# Serial drivers +# +# CONFIG_SERIAL_ANAKIN is not set +# CONFIG_SERIAL_ANAKIN_CONSOLE is not set +# CONFIG_SERIAL_AMBA is not set +# CONFIG_SERIAL_AMBA_CONSOLE is not set +# CONFIG_SERIAL_CLPS711X is not set +# CONFIG_SERIAL_CLPS711X_CONSOLE is not set +# CONFIG_SERIAL_21285 is not set +# CONFIG_SERIAL_21285_OLD is not set +# CONFIG_SERIAL_21285_CONSOLE is not set +# CONFIG_SERIAL_UART00 is not set +# CONFIG_SERIAL_UART00_CONSOLE is not set +CONFIG_SERIAL_SA1100=y +CONFIG_SERIAL_SA1100_CONSOLE=y +CONFIG_SA1100_DEFAULT_BAUDRATE=115200 +# CONFIG_SERIAL_8250 is not set +# CONFIG_SERIAL_8250_CONSOLE is not set +# CONFIG_SERIAL_8250_EXTENDED is not set +# CONFIG_SERIAL_8250_MANY_PORTS is not set +# CONFIG_SERIAL_8250_SHARE_IRQ is not set +# CONFIG_SERIAL_8250_DETECT_IRQ is not set +# CONFIG_SERIAL_8250_MULTIPORT is not set +# CONFIG_SERIAL_8250_HUB6 is not set +CONFIG_SERIAL_CORE=y +CONFIG_SERIAL_CORE_CONSOLE=y +CONFIG_SA1111_PS2_KEYB=y +CONFIG_ASI_KEYBOARD=y +# CONFIG_RTC_ASI is not set +CONFIG_UNIX98_PTYS=y +CONFIG_UNIX98_PTY_COUNT=32 +CONFIG_PRINTER=m +# CONFIG_LP_CONSOLE is not set +# CONFIG_PPDEV is not set + +# +# I2C support +# +# CONFIG_I2C is not set + +# +# L3 serial bus support +# +CONFIG_L3=y +# CONFIG_L3_ALGOBIT is not set +# CONFIG_L3_BIT_SA1100_GPIO is not set + +# +# Other L3 adapters +# +CONFIG_L3_SA1111=y +# CONFIG_BIT_SA1100_GPIO is not set + +# +# Mice +# +# CONFIG_BUSMOUSE is not set +CONFIG_MOUSE=m +CONFIG_PSMOUSE=y +# CONFIG_82C710_MOUSE is not set +# CONFIG_PC110_PAD is not set + +# +# Joysticks +# +# CONFIG_INPUT_GAMEPORT is not set +# CONFIG_INPUT_NS558 is not set +# CONFIG_INPUT_LIGHTNING is not set +# CONFIG_INPUT_PCIGAME is not set +# CONFIG_INPUT_CS461X is not set +# CONFIG_INPUT_EMU10K1 is not set +# CONFIG_INPUT_SERIO is not set +# CONFIG_INPUT_SERPORT is not set + +# +# Joysticks +# +# CONFIG_INPUT_ANALOG is not set +# CONFIG_INPUT_A3D is not set +# CONFIG_INPUT_ADI is not set +# CONFIG_INPUT_COBRA is not set +# CONFIG_INPUT_GF2K is not set +# CONFIG_INPUT_GRIP is not set +# CONFIG_INPUT_INTERACT is not set +# CONFIG_INPUT_TMDC is not set +# CONFIG_INPUT_SIDEWINDER is not set +# CONFIG_INPUT_IFORCE_USB is not set +# CONFIG_INPUT_IFORCE_232 is not set +# CONFIG_INPUT_WARRIOR is not set +# CONFIG_INPUT_MAGELLAN is not set +# CONFIG_INPUT_SPACEORB is not set +# CONFIG_INPUT_SPACEBALL is not set +# CONFIG_INPUT_STINGER is not set +# CONFIG_INPUT_DB9 is not set +# CONFIG_INPUT_GAMECON is not set +# CONFIG_INPUT_TURBOGRAFX is not set +# CONFIG_QIC02_TAPE is not set + +# +# Watchdog Cards +# +CONFIG_WATCHDOG=y +# CONFIG_WATCHDOG_NOWAYOUT is not set +# CONFIG_SOFT_WATCHDOG is not set +# CONFIG_WDT is not set +# CONFIG_WDTPCI is not set +# CONFIG_PCWATCHDOG is not set +# CONFIG_ACQUIRE_WDT is not set +# CONFIG_ADVANTECH_WDT is not set +# CONFIG_21285_WATCHDOG is not set +# CONFIG_977_WATCHDOG is not set +CONFIG_SA1100_WATCHDOG=m +# CONFIG_EUROTECH_WDT is not set +# CONFIG_IB700_WDT is not set +# CONFIG_I810_TCO is not set +# CONFIG_MIXCOMWD is not set +# CONFIG_60XX_WDT is not set +# CONFIG_W83877F_WDT is not set +# CONFIG_MACHZ_WDT is not set +# CONFIG_INTEL_RNG is not set +# CONFIG_NVRAM is not set +# CONFIG_RTC is not set +CONFIG_SA1100_RTC=m +# CONFIG_DTLK is not set +# CONFIG_R3964 is not set +# CONFIG_APPLICOM is not set + +# +# Ftape, the floppy tape device driver +# +# CONFIG_FTAPE is not set +# CONFIG_AGP is not set +# CONFIG_DRM is not set + +# +# PCMCIA character devices +# +# CONFIG_PCMCIA_SERIAL_CS is not set + +# +# Multimedia devices +# +# CONFIG_VIDEO_DEV is not set + +# +# File systems +# +# CONFIG_QUOTA is not set +# CONFIG_AUTOFS_FS is not set +# CONFIG_AUTOFS4_FS is not set +# CONFIG_REISERFS_FS is not set +# CONFIG_REISERFS_CHECK is not set +# CONFIG_REISERFS_PROC_INFO is not set +# CONFIG_ADFS_FS is not set +# CONFIG_ADFS_FS_RW is not set +# CONFIG_AFFS_FS is not set +# CONFIG_HFS_FS is not set +# CONFIG_BFS_FS is not set +CONFIG_EXT3_FS=m +CONFIG_JBD=m +# CONFIG_JBD_DEBUG is not set +CONFIG_FAT_FS=m +CONFIG_MSDOS_FS=m +# CONFIG_UMSDOS_FS is not set +CONFIG_VFAT_FS=m +# CONFIG_EFS_FS is not set +# CONFIG_JFFS_FS is not set +CONFIG_JFFS2_FS=y +CONFIG_JFFS2_FS_DEBUG=0 +CONFIG_CRAMFS=m +# CONFIG_TMPFS is not set +CONFIG_RAMFS=y +# CONFIG_ISO9660_FS is not set +# CONFIG_JOLIET is not set +# CONFIG_ZISOFS is not set +# CONFIG_MINIX_FS is not set +# CONFIG_VXFS_FS is not set +# CONFIG_NTFS_FS is not set +# CONFIG_NTFS_RW is not set +# CONFIG_HPFS_FS is not set +CONFIG_PROC_FS=y +CONFIG_DEVFS_FS=y +CONFIG_DEVFS_MOUNT=y +# CONFIG_DEVFS_DEBUG is not set +# CONFIG_DEVPTS_FS is not set +# CONFIG_QNX4FS_FS is not set +# CONFIG_QNX4FS_RW is not set +# CONFIG_ROMFS_FS is not set +CONFIG_EXT2_FS=m +# CONFIG_SYSV_FS is not set +CONFIG_UDF_FS=m +# CONFIG_UDF_RW is not set +# CONFIG_UFS_FS is not set +# CONFIG_UFS_FS_WRITE is not set + +# +# Network File Systems +# +# CONFIG_CODA_FS is not set +# CONFIG_INTERMEZZO_FS is not set +CONFIG_NFS_FS=m +CONFIG_NFS_V3=y +# CONFIG_ROOT_NFS is not set +CONFIG_NFSD=m +CONFIG_NFSD_V3=y +CONFIG_SUNRPC=m +CONFIG_LOCKD=m +CONFIG_LOCKD_V4=y +CONFIG_SMB_FS=m +CONFIG_SMB_NLS_DEFAULT=y +CONFIG_SMB_NLS_REMOTE="cp437" +# CONFIG_NCP_FS is not set +# CONFIG_NCPFS_PACKET_SIGNING is not set +# CONFIG_NCPFS_IOCTL_LOCKING is not set +# CONFIG_NCPFS_STRONG is not set +# CONFIG_NCPFS_NFS_NS is not set +# CONFIG_NCPFS_OS2_NS is not set +# CONFIG_NCPFS_SMALLDOS is not set +# CONFIG_NCPFS_NLS is not set +# CONFIG_NCPFS_EXTRAS is not set +# CONFIG_ZISOFS_FS is not set +CONFIG_ZLIB_FS_INFLATE=m + +# +# Partition Types +# +# CONFIG_PARTITION_ADVANCED is not set +CONFIG_MSDOS_PARTITION=y +CONFIG_SMB_NLS=y +CONFIG_NLS=y + +# +# Native Language Support +# +CONFIG_NLS_DEFAULT="iso8859-1" +CONFIG_NLS_CODEPAGE_437=m +# CONFIG_NLS_CODEPAGE_737 is not set +# CONFIG_NLS_CODEPAGE_775 is not set +# CONFIG_NLS_CODEPAGE_850 is not set +# CONFIG_NLS_CODEPAGE_852 is not set +# CONFIG_NLS_CODEPAGE_855 is not set +# CONFIG_NLS_CODEPAGE_857 is not set +# CONFIG_NLS_CODEPAGE_860 is not set +# CONFIG_NLS_CODEPAGE_861 is not set +# CONFIG_NLS_CODEPAGE_862 is not set +# CONFIG_NLS_CODEPAGE_863 is not set +# CONFIG_NLS_CODEPAGE_864 is not set +# CONFIG_NLS_CODEPAGE_865 is not set +# CONFIG_NLS_CODEPAGE_866 is not set +# CONFIG_NLS_CODEPAGE_869 is not set +# CONFIG_NLS_CODEPAGE_936 is not set +# CONFIG_NLS_CODEPAGE_950 is not set +# CONFIG_NLS_CODEPAGE_932 is not set +# CONFIG_NLS_CODEPAGE_949 is not set +# CONFIG_NLS_CODEPAGE_874 is not set +# CONFIG_NLS_ISO8859_8 is not set +# CONFIG_NLS_CODEPAGE_1250 is not set +# CONFIG_NLS_CODEPAGE_1251 is not set +# CONFIG_NLS_ISO8859_1 is not set +# CONFIG_NLS_ISO8859_2 is not set +# CONFIG_NLS_ISO8859_3 is not set +# CONFIG_NLS_ISO8859_4 is not set +# CONFIG_NLS_ISO8859_5 is not set +# CONFIG_NLS_ISO8859_6 is not set +# CONFIG_NLS_ISO8859_7 is not set +# CONFIG_NLS_ISO8859_9 is not set +# CONFIG_NLS_ISO8859_13 is not set +# CONFIG_NLS_ISO8859_14 is not set +# CONFIG_NLS_ISO8859_15 is not set +# CONFIG_NLS_KOI8_R is not set +# CONFIG_NLS_KOI8_U is not set +# CONFIG_NLS_UTF8 is not set + +# +# Console drivers +# +CONFIG_PC_KEYMAP=y +# CONFIG_VGA_CONSOLE is not set + +# +# Frame-buffer support +# +CONFIG_FB=y +CONFIG_DUMMY_CONSOLE=y +# CONFIG_FB_ACORN is not set +# CONFIG_FB_ANAKIN is not set +# CONFIG_FB_CLPS711X is not set +CONFIG_FB_SA1100=y +# CONFIG_FB_CYBER2000 is not set +# CONFIG_FB_VIRTUAL is not set +# CONFIG_FBCON_ADVANCED is not set +CONFIG_FBCON_CFB2=y +CONFIG_FBCON_CFB4=y +CONFIG_FBCON_CFB8=y +CONFIG_FBCON_CFB16=y +# CONFIG_FBCON_FONTWIDTH8_ONLY is not set +# CONFIG_FBCON_FONTS is not set +CONFIG_FONT_8x8=y +CONFIG_FONT_8x16=y + +# +# Sound +# +CONFIG_SOUND=y +# CONFIG_SOUND_BT878 is not set +# CONFIG_SOUND_CMPCI is not set +# CONFIG_SOUND_EMU10K1 is not set +# CONFIG_MIDI_EMU10K1 is not set +# CONFIG_SOUND_FUSION is not set +# CONFIG_SOUND_CS4281 is not set +# CONFIG_SOUND_ES1370 is not set +# CONFIG_SOUND_ES1371 is not set +# CONFIG_SOUND_ESSSOLO1 is not set +# CONFIG_SOUND_MAESTRO is not set +# CONFIG_SOUND_MAESTRO3 is not set +# CONFIG_SOUND_ICH is not set +# CONFIG_SOUND_RME96XX is not set +# CONFIG_SOUND_SONICVIBES is not set +# CONFIG_SOUND_TRIDENT is not set +# CONFIG_SOUND_MSNDCLAS is not set +# CONFIG_SOUND_MSNDPIN is not set +# CONFIG_SOUND_VIA82CXXX is not set +# CONFIG_MIDI_VIA82CXXX is not set +CONFIG_SOUND_SA1100=y +CONFIG_SOUND_UDA1341=y +# CONFIG_SOUND_ASSABET_UDA1341 is not set +# CONFIG_SOUND_H3600_UDA1341 is not set +# CONFIG_SOUND_PANGOLIN_UDA1341 is not set +CONFIG_SOUND_SA1111_UDA1341=y +# CONFIG_SOUND_SA1100SSP is not set +# CONFIG_SOUND_SA1111_AC97 is not set +# CONFIG_SOUND_SA1111_AC97_RATE_CONVERSION is not set +# CONFIG_SOUND_OSS is not set +# CONFIG_SOUND_WAVEARTIST is not set +# CONFIG_SOUND_TVMIXER is not set + +# +# Multimedia Capabilities Port drivers +# +CONFIG_MCP=y +CONFIG_MCP_SA1100=y +CONFIG_MCP_UCB1200=y +CONFIG_MCP_UCB1200_AUDIO=m +CONFIG_MCP_UCB1200_TS=m + +# +# USB support +# +CONFIG_USB=y +# CONFIG_USB_DEBUG is not set + +# +# Miscellaneous USB options +# +CONFIG_USB_DEVICEFS=y +# CONFIG_USB_BANDWIDTH is not set +# CONFIG_USB_LONG_TIMEOUT is not set + +# +# USB Controllers +# +# CONFIG_USB_UHCI is not set +# CONFIG_USB_UHCI_ALT is not set +# CONFIG_USB_OHCI is not set +CONFIG_USB_OHCI_SA1111=m +CONFIG_USB_SL811HS=m + +# +# USB Device Class drivers +# +# CONFIG_USB_AUDIO is not set +# CONFIG_USB_BLUETOOTH is not set +CONFIG_USB_STORAGE=m +# CONFIG_USB_STORAGE_DEBUG is not set +# CONFIG_USB_STORAGE_DATAFAB is not set +# CONFIG_USB_STORAGE_FREECOM is not set +# CONFIG_USB_STORAGE_ISD200 is not set +# CONFIG_USB_STORAGE_DPCM is not set +# CONFIG_USB_STORAGE_HP8200e is not set +# CONFIG_USB_STORAGE_SDDR09 is not set +# CONFIG_USB_STORAGE_JUMPSHOT is not set +# CONFIG_USB_ACM is not set +# CONFIG_USB_PRINTER is not set + +# +# USB Human Interface Devices (HID) +# +CONFIG_USB_HID=m +CONFIG_USB_HIDDEV=y +# CONFIG_USB_KBD is not set +# CONFIG_USB_MOUSE is not set +# CONFIG_USB_WACOM is not set + +# +# USB Imaging devices +# +# CONFIG_USB_DC2XX is not set +# CONFIG_USB_MDC800 is not set +# CONFIG_USB_SCANNER is not set +# CONFIG_USB_MICROTEK is not set +# CONFIG_USB_HPUSBSCSI is not set + +# +# USB Multimedia devices +# + +# +# Video4Linux support is needed for USB Multimedia device support +# + +# +# USB Network adaptors +# +# CONFIG_USB_PEGASUS is not set +# CONFIG_USB_KAWETH is not set +# CONFIG_USB_CATC is not set +# CONFIG_USB_CDCETHER is not set +CONFIG_USB_USBNET=m + +# +# USB port drivers +# +# CONFIG_USB_USS720 is not set + +# +# USB Serial Converter support +# +CONFIG_USB_SERIAL=m +CONFIG_USB_SERIAL_GENERIC=y +# CONFIG_USB_SERIAL_BELKIN is not set +# CONFIG_USB_SERIAL_WHITEHEAT is not set +# CONFIG_USB_SERIAL_DIGI_ACCELEPORT is not set +# CONFIG_USB_SERIAL_EMPEG is not set +# CONFIG_USB_SERIAL_FTDI_SIO is not set +# CONFIG_USB_SERIAL_VISOR is not set +# CONFIG_USB_SERIAL_IPAQ is not set +# CONFIG_USB_SERIAL_IR is not set +# CONFIG_USB_SERIAL_EDGEPORT is not set +# CONFIG_USB_SERIAL_KEYSPAN_PDA is not set +# CONFIG_USB_SERIAL_KEYSPAN is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XA is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XB is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA18X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19W is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA49W is not set +# CONFIG_USB_SERIAL_MCT_U232 is not set +# CONFIG_USB_SERIAL_KLSI is not set +# CONFIG_USB_SERIAL_PL2303 is not set +# CONFIG_USB_SERIAL_CYBERJACK is not set +# CONFIG_USB_SERIAL_XIRCOM is not set +# CONFIG_USB_SERIAL_OMNINET is not set + +# +# USB Miscellaneous drivers +# +# CONFIG_USB_RIO500 is not set + +# +# Bluetooth support +# +# CONFIG_BLUEZ is not set + +# +# Kernel hacking +# +CONFIG_FRAME_POINTER=y +CONFIG_DEBUG_USER=y +# CONFIG_DEBUG_INFO is not set +# CONFIG_NO_PGT_CACHE is not set +CONFIG_DEBUG_KERNEL=y +CONFIG_DEBUG_SLAB=y +# CONFIG_MAGIC_SYSRQ is not set +CONFIG_DEBUG_SPINLOCK=y +CONFIG_DEBUG_WAITQ=y +CONFIG_DEBUG_BUGVERBOSE=y +CONFIG_DEBUG_ERRORS=y +CONFIG_DEBUG_LL=y +# CONFIG_DEBUG_DC21285_PORT is not set +# CONFIG_DEBUG_CLPS711X_UART2 is not set diff -urN orig/arch/arm/def-configs/adsagc linux/arch/arm/def-configs/adsagc --- orig/arch/arm/def-configs/adsagc Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/def-configs/adsagc Fri Feb 21 15:17:34 2003 @@ -0,0 +1,1101 @@ +# +# Automatically generated make config: don't edit +# +CONFIG_ARM=y +# CONFIG_EISA is not set +# CONFIG_SBUS is not set +# CONFIG_MCA is not set +CONFIG_UID16=y +CONFIG_RWSEM_GENERIC_SPINLOCK=y +# CONFIG_RWSEM_XCHGADD_ALGORITHM is not set +# CONFIG_GENERIC_BUST_SPINLOCK is not set +# CONFIG_GENERIC_ISA_DMA is not set + +# +# Code maturity level options +# +CONFIG_EXPERIMENTAL=y +# CONFIG_OBSOLETE is not set + +# +# Loadable module support +# +CONFIG_MODULES=y +# CONFIG_MODVERSIONS is not set +# CONFIG_KMOD is not set + +# +# System Type +# +# CONFIG_ARCH_ANAKIN is not set +# CONFIG_ARCH_ARCA5K is not set +# CONFIG_ARCH_CLPS7500 is not set +# CONFIG_ARCH_CLPS711X is not set +# CONFIG_ARCH_CO285 is not set +# CONFIG_ARCH_EBSA110 is not set +# CONFIG_ARCH_CAMELOT is not set +# CONFIG_ARCH_FOOTBRIDGE is not set +# CONFIG_ARCH_INTEGRATOR is not set +# CONFIG_ARCH_OMAHA is not set +# CONFIG_ARCH_L7200 is not set +# CONFIG_ARCH_MX1ADS is not set +# CONFIG_ARCH_RPC is not set +# CONFIG_ARCH_RISCSTATION is not set +CONFIG_ARCH_SA1100=y +# CONFIG_ARCH_SHARK is not set +# CONFIG_ARCH_AT91RM9200DK is not set + +# +# Archimedes/A5000 Implementations +# + +# +# Archimedes/A5000 Implementations (select only ONE) +# +# CONFIG_ARCH_ARC is not set +# CONFIG_ARCH_A5K is not set + +# +# Footbridge Implementations +# +# CONFIG_ARCH_CATS is not set +# CONFIG_ARCH_PERSONAL_SERVER is not set +# CONFIG_ARCH_EBSA285_ADDIN is not set +# CONFIG_ARCH_EBSA285_HOST is not set +# CONFIG_ARCH_NETWINDER is not set + +# +# SA11x0 Implementations +# +# CONFIG_SA1100_ACCELENT is not set +# CONFIG_SA1100_ASSABET is not set +# CONFIG_ASSABET_NEPONSET is not set +CONFIG_SA1100_ADSAGC=y +# CONFIG_SA1100_ADSBITSY is not set +# CONFIG_SA1100_ADSBITSYPLUS is not set +# CONFIG_SA1100_BRUTUS is not set +# CONFIG_SA1100_CEP is not set +# CONFIG_SA1100_CERF is not set +# CONFIG_SA1100_H3100 is not set +# CONFIG_SA1100_H3600 is not set +# CONFIG_SA1100_H3800 is not set +# CONFIG_SA1100_H3XXX is not set +# CONFIG_SA1100_EXTENEX1 is not set +# CONFIG_SA1100_FLEXANET is not set +# CONFIG_SA1100_FREEBIRD is not set +# CONFIG_SA1100_FRODO is not set +# CONFIG_SA1100_GRAPHICSCLIENT is not set +# CONFIG_SA1100_GRAPHICSMASTER is not set +# CONFIG_SA1100_HACKKIT is not set +# CONFIG_SA1100_BADGE4 is not set +# CONFIG_SA1100_JORNADA720 is not set +# CONFIG_SA1100_HUW_WEBPANEL is not set +# CONFIG_SA1100_ITSY is not set +# CONFIG_SA1100_LART is not set +# CONFIG_SA1100_NANOENGINE is not set +# CONFIG_SA1100_OMNIMETER is not set +# CONFIG_SA1100_PANGOLIN is not set +# CONFIG_SA1100_PLEB is not set +# CONFIG_SA1100_PT_SYSTEM3 is not set +# CONFIG_SA1100_SHANNON is not set +# CONFIG_SA1100_SHERMAN is not set +# CONFIG_SA1100_SIMPAD is not set +# CONFIG_SA1100_SIMPUTER is not set +# CONFIG_SA1100_PFS168 is not set +# CONFIG_SA1100_VICTOR is not set +# CONFIG_SA1100_XP860 is not set +# CONFIG_SA1100_YOPY is not set +# CONFIG_SA1100_USB is not set +# CONFIG_SA1100_USB_NETLINK is not set +# CONFIG_SA1100_USB_CHAR is not set +# CONFIG_H3600_SLEEVE is not set + +# +# CLPS711X/EP721X Implementations +# +# CONFIG_ARCH_AUTCPU12 is not set +# CONFIG_ARCH_CDB89712 is not set +# CONFIG_ARCH_CLEP7312 is not set +# CONFIG_ARCH_EDB7211 is not set +# CONFIG_ARCH_P720T is not set +# CONFIG_ARCH_FORTUNET is not set +# CONFIG_ARCH_EP7211 is not set +# CONFIG_ARCH_EP7212 is not set +# CONFIG_ARCH_ACORN is not set +# CONFIG_FOOTBRIDGE is not set +# CONFIG_FOOTBRIDGE_HOST is not set +# CONFIG_FOOTBRIDGE_ADDIN is not set +CONFIG_SA1111=y +CONFIG_FORCE_MAX_ZONEORDER=9 + +# +# Processor Type +# +CONFIG_CPU_32=y +# CONFIG_CPU_26 is not set +# CONFIG_CPU_ARM610 is not set +# CONFIG_CPU_ARM710 is not set +# CONFIG_CPU_ARM720T is not set +# CONFIG_CPU_ARM920T is not set +# CONFIG_CPU_ARM922T is not set +# CONFIG_PLD is not set +# CONFIG_CPU_ARM926T is not set +# CONFIG_CPU_ARM1020 is not set +# CONFIG_CPU_ARM1026 is not set +# CONFIG_CPU_SA110 is not set +CONFIG_CPU_SA1100=y +# CONFIG_CPU_32v3 is not set +CONFIG_CPU_32v4=y + +# +# Processor Features +# +CONFIG_DISCONTIGMEM=y + +# +# General setup +# +# CONFIG_PCI is not set +# CONFIG_ISA is not set +# CONFIG_ISA_DMA is not set +# CONFIG_ZBOOT_ROM is not set +CONFIG_ZBOOT_ROM_TEXT=0 +CONFIG_ZBOOT_ROM_BSS=0 +# CONFIG_CPU_FREQ is not set +CONFIG_HOTPLUG=y + +# +# PCMCIA/CardBus support +# +CONFIG_PCMCIA=y +CONFIG_PCMCIA_PROBE=y +# CONFIG_I82092 is not set +# CONFIG_I82365 is not set +# CONFIG_TCIC is not set +# CONFIG_PCMCIA_CLPS6700 is not set +CONFIG_PCMCIA_SA1100=y +CONFIG_NET=y +CONFIG_SYSVIPC=y +# CONFIG_BSD_PROCESS_ACCT is not set +CONFIG_SYSCTL=y + +# +# At least one math emulation must be selected +# +CONFIG_FPE_NWFPE=y +# CONFIG_FPE_FASTFPE is not set +CONFIG_KCORE_ELF=y +# CONFIG_KCORE_AOUT is not set +# CONFIG_BINFMT_AOUT is not set +CONFIG_BINFMT_ELF=y +# CONFIG_BINFMT_MISC is not set +CONFIG_PM=y +# CONFIG_ARTHUR is not set +CONFIG_CMDLINE="console=ttySA0 mem=16m@0xC0000000 mem=16m@0xC8000000 rw ramdisk_size=8192 initrd=0xC0800000,3m root=/dev/ram" +CONFIG_LEDS=y +CONFIG_LEDS_TIMER=y +CONFIG_LEDS_CPU=y +CONFIG_ALIGNMENT_TRAP=y + +# +# Parallel port support +# +# CONFIG_PARPORT is not set + +# +# Memory Technology Devices (MTD) +# +CONFIG_MTD=y +# CONFIG_MTD_DEBUG is not set +CONFIG_MTD_PARTITIONS=y +# CONFIG_MTD_CONCAT is not set +# CONFIG_MTD_REDBOOT_PARTS is not set +CONFIG_MTD_CMDLINE_PARTS=y +# CONFIG_MTD_AFS_PARTS is not set + +# +# User Modules And Translation Layers +# +CONFIG_MTD_CHAR=y +CONFIG_MTD_BLOCK=y +# CONFIG_FTL is not set +# CONFIG_NFTL is not set + +# +# RAM/ROM/Flash chip drivers +# +CONFIG_MTD_CFI=y +# CONFIG_MTD_JEDECPROBE is not set +CONFIG_MTD_GEN_PROBE=y +# CONFIG_MTD_CFI_ADV_OPTIONS is not set +CONFIG_MTD_CFI_INTELEXT=y +# CONFIG_MTD_CFI_AMDSTD is not set +# CONFIG_MTD_RAM is not set +# CONFIG_MTD_ROM is not set +# CONFIG_MTD_ABSENT is not set +# CONFIG_MTD_OBSOLETE_CHIPS is not set +# CONFIG_MTD_AMDSTD is not set +# CONFIG_MTD_SHARP is not set +# CONFIG_MTD_JEDEC is not set + +# +# Mapping drivers for chip access +# +CONFIG_MTD_PHYSMAP=y +CONFIG_MTD_PHYSMAP_START=8000000 +CONFIG_MTD_PHYSMAP_LEN=4000000 +CONFIG_MTD_PHYSMAP_BUSWIDTH=4 +# CONFIG_MTD_NORA is not set +# CONFIG_MTD_ARM_INTEGRATOR is not set +# CONFIG_MTD_CDB89712 is not set +CONFIG_MTD_SA1100=y +# CONFIG_MTD_DC21285 is not set +# CONFIG_MTD_IQ80310 is not set +# CONFIG_MTD_FORTUNET is not set +# CONFIG_MTD_EPXA10DB is not set +# CONFIG_MTD_AUTCPU12 is not set +# CONFIG_MTD_EDB7312 is not set +# CONFIG_MTD_IMPA7 is not set +# CONFIG_MTD_PCI is not set + +# +# Self-contained MTD device drivers +# +# CONFIG_MTD_PMC551 is not set +# CONFIG_MTD_SLRAM is not set +# CONFIG_MTD_MTDRAM is not set +# CONFIG_MTD_BLKMTD is not set + +# +# Disk-On-Chip Device Drivers +# +# CONFIG_MTD_DOC1000 is not set +# CONFIG_MTD_DOC2000 is not set +# CONFIG_MTD_DOC2001 is not set +# CONFIG_MTD_DOCPROBE is not set + +# +# NAND Flash Device Drivers +# +# CONFIG_MTD_NAND is not set + +# +# Plug and Play configuration +# +# CONFIG_PNP is not set +# CONFIG_ISAPNP is not set + +# +# Block devices +# +# CONFIG_BLK_DEV_FD is not set +# CONFIG_BLK_DEV_XD is not set +# CONFIG_PARIDE is not set +# CONFIG_BLK_CPQ_DA is not set +# CONFIG_BLK_CPQ_CISS_DA is not set +# CONFIG_CISS_SCSI_TAPE is not set +# CONFIG_BLK_DEV_DAC960 is not set +# CONFIG_BLK_DEV_UMEM is not set +CONFIG_BLK_DEV_LOOP=y +# CONFIG_BLK_DEV_NBD is not set +CONFIG_BLK_DEV_RAM=y +CONFIG_BLK_DEV_RAM_SIZE=8192 +CONFIG_BLK_DEV_INITRD=y + +# +# Multi-device support (RAID and LVM) +# +# CONFIG_MD is not set +# CONFIG_BLK_DEV_MD is not set +# CONFIG_MD_LINEAR is not set +# CONFIG_MD_RAID0 is not set +# CONFIG_MD_RAID1 is not set +# CONFIG_MD_RAID5 is not set +# CONFIG_MD_MULTIPATH is not set +# CONFIG_BLK_DEV_LVM is not set + +# +# Networking options +# +CONFIG_PACKET=y +# CONFIG_PACKET_MMAP is not set +# CONFIG_NETLINK_DEV is not set +# CONFIG_NETFILTER is not set +# CONFIG_FILTER is not set +CONFIG_UNIX=y +CONFIG_INET=y +# CONFIG_IP_MULTICAST is not set +# CONFIG_IP_ADVANCED_ROUTER is not set +CONFIG_IP_PNP=y +# CONFIG_IP_PNP_DHCP is not set +CONFIG_IP_PNP_BOOTP=y +# CONFIG_IP_PNP_RARP is not set +# CONFIG_NET_IPIP is not set +# CONFIG_NET_IPGRE is not set +# CONFIG_ARPD is not set +# CONFIG_INET_ECN is not set +# CONFIG_SYN_COOKIES is not set +# CONFIG_IPV6 is not set +# CONFIG_KHTTPD is not set +# CONFIG_ATM is not set +# CONFIG_VLAN_8021Q is not set + +# +# +# +# CONFIG_IPX is not set +# CONFIG_ATALK is not set + +# +# Appletalk devices +# +# CONFIG_DEV_APPLETALK is not set +# CONFIG_DECNET is not set +# CONFIG_BRIDGE is not set +# CONFIG_X25 is not set +# CONFIG_LAPB is not set +# CONFIG_LLC is not set +# CONFIG_NET_DIVERT is not set +# CONFIG_ECONET is not set +# CONFIG_WAN_ROUTER is not set +# CONFIG_NET_FASTROUTE is not set +# CONFIG_NET_HW_FLOWCONTROL is not set + +# +# QoS and/or fair queueing +# +# CONFIG_NET_SCHED is not set + +# +# Network testing +# +# CONFIG_NET_PKTGEN is not set + +# +# Network device support +# +CONFIG_NETDEVICES=y + +# +# ARCnet devices +# +# CONFIG_ARCNET is not set +# CONFIG_DUMMY is not set +# CONFIG_BONDING is not set +# CONFIG_EQUALIZER is not set +# CONFIG_TUN is not set +# CONFIG_ETHERTAP is not set + +# +# Ethernet (10 or 100Mbit) +# +CONFIG_NET_ETHERNET=y +# CONFIG_ARM_AM79C961A is not set +# CONFIG_ARM_CIRRUS is not set +# CONFIG_SUNLANCE is not set +# CONFIG_SUNBMAC is not set +# CONFIG_SUNQE is not set +# CONFIG_SUNGEM is not set +# CONFIG_NET_VENDOR_3COM is not set +# CONFIG_LANCE is not set +CONFIG_NET_VENDOR_SMC=y +# CONFIG_WD80x3 is not set +# CONFIG_ULTRAMCA is not set +# CONFIG_ULTRA is not set +# CONFIG_ULTRA32 is not set +# CONFIG_SMC9194 is not set +CONFIG_SMC91111=y +# CONFIG_NET_VENDOR_RACAL is not set +# CONFIG_NET_ISA is not set +# CONFIG_NET_PCI is not set +# CONFIG_NET_POCKET is not set + +# +# Ethernet (1000 Mbit) +# +# CONFIG_ACENIC is not set +# CONFIG_DL2K is not set +# CONFIG_MYRI_SBUS is not set +# CONFIG_NS83820 is not set +# CONFIG_HAMACHI is not set +# CONFIG_YELLOWFIN is not set +# CONFIG_SK98LIN is not set +# CONFIG_TIGON3 is not set +# CONFIG_FDDI is not set +# CONFIG_HIPPI is not set +# CONFIG_PLIP is not set +CONFIG_PPP=y +# CONFIG_PPP_MULTILINK is not set +# CONFIG_PPP_FILTER is not set +CONFIG_PPP_ASYNC=y +# CONFIG_PPP_SYNC_TTY is not set +# CONFIG_PPP_DEFLATE is not set +# CONFIG_PPP_BSDCOMP is not set +# CONFIG_PPPOE is not set +# CONFIG_SLIP is not set + +# +# Wireless LAN (non-hamradio) +# +CONFIG_NET_RADIO=y +# CONFIG_STRIP is not set +# CONFIG_WAVELAN is not set +# CONFIG_ARLAN is not set +# CONFIG_AIRONET4500 is not set +# CONFIG_AIRONET4500_NONCS is not set +# CONFIG_AIRONET4500_PROC is not set +# CONFIG_HERMES is not set + +# +# Wireless Pcmcia cards support +# +# CONFIG_PCMCIA_HERMES is not set +# CONFIG_AIRO_CS is not set +CONFIG_NET_WIRELESS=y + +# +# Token Ring devices +# +# CONFIG_TR is not set +# CONFIG_NET_FC is not set +# CONFIG_RCPCI is not set +# CONFIG_SHAPER is not set + +# +# Wan interfaces +# +# CONFIG_WAN is not set + +# +# PCMCIA network device support +# +CONFIG_NET_PCMCIA=y +# CONFIG_PCMCIA_3C589 is not set +# CONFIG_PCMCIA_3C574 is not set +# CONFIG_PCMCIA_FMVJ18X is not set +CONFIG_PCMCIA_PCNET=y +# CONFIG_PCMCIA_AXNET is not set +# CONFIG_PCMCIA_NMCLAN is not set +# CONFIG_PCMCIA_SMC91C92 is not set +# CONFIG_PCMCIA_XIRC2PS is not set +# CONFIG_ARCNET_COM20020_CS is not set +# CONFIG_PCMCIA_IBMTR is not set +# CONFIG_NET_PCMCIA_RADIO is not set + +# +# Amateur Radio support +# +# CONFIG_HAMRADIO is not set + +# +# IrDA (infrared) support +# +# CONFIG_IRDA is not set + +# +# ATA/ATAPI/MFM/RLL support +# +CONFIG_IDE=y + +# +# IDE, ATA and ATAPI Block devices +# +CONFIG_BLK_DEV_IDE=y + +# +# Please see Documentation/ide.txt for help/info on IDE drives +# +# CONFIG_BLK_DEV_HD_IDE is not set +# CONFIG_BLK_DEV_HD is not set +CONFIG_BLK_DEV_IDEDISK=y +# CONFIG_IDEDISK_MULTI_MODE is not set +# CONFIG_IDEDISK_STROKE is not set +# CONFIG_BLK_DEV_IDEDISK_VENDOR is not set +# CONFIG_BLK_DEV_IDEDISK_FUJITSU is not set +# CONFIG_BLK_DEV_IDEDISK_IBM is not set +# CONFIG_BLK_DEV_IDEDISK_MAXTOR is not set +# CONFIG_BLK_DEV_IDEDISK_QUANTUM is not set +# CONFIG_BLK_DEV_IDEDISK_SEAGATE is not set +# CONFIG_BLK_DEV_IDEDISK_WD is not set +# CONFIG_BLK_DEV_COMMERIAL is not set +# CONFIG_BLK_DEV_TIVO is not set +CONFIG_BLK_DEV_IDECS=y +# CONFIG_BLK_DEV_IDECD is not set +# CONFIG_BLK_DEV_IDETAPE is not set +# CONFIG_BLK_DEV_IDEFLOPPY is not set +# CONFIG_BLK_DEV_IDESCSI is not set +# CONFIG_IDE_TASK_IOCTL is not set + +# +# IDE chipset support/bugfixes +# +# CONFIG_BLK_DEV_CMD640 is not set +# CONFIG_BLK_DEV_CMD640_ENHANCED is not set +# CONFIG_BLK_DEV_ISAPNP is not set +# CONFIG_IDE_CHIPSETS is not set +# CONFIG_IDEDMA_AUTO is not set +# CONFIG_DMA_NONPCI is not set +# CONFIG_BLK_DEV_IDE_MODES is not set +# CONFIG_BLK_DEV_ATARAID is not set +# CONFIG_BLK_DEV_ATARAID_PDC is not set +# CONFIG_BLK_DEV_ATARAID_HPT is not set + +# +# SCSI support +# +# CONFIG_SCSI is not set + +# +# I2O device support +# +# CONFIG_I2O is not set +# CONFIG_I2O_BLOCK is not set +# CONFIG_I2O_LAN is not set +# CONFIG_I2O_SCSI is not set +# CONFIG_I2O_PROC is not set + +# +# ISDN subsystem +# +# CONFIG_ISDN is not set + +# +# Input core support +# +CONFIG_INPUT=y +# CONFIG_INPUT_KEYBDEV is not set +CONFIG_INPUT_MOUSEDEV=y +CONFIG_INPUT_MOUSEDEV_SCREEN_X=640 +CONFIG_INPUT_MOUSEDEV_SCREEN_Y=480 +# CONFIG_INPUT_JOYDEV is not set +# CONFIG_INPUT_EVDEV is not set + +# +# Character devices +# +CONFIG_VT=y +CONFIG_VT_CONSOLE=y +# CONFIG_SERIAL is not set +# CONFIG_SERIAL_EXTENDED is not set +# CONFIG_SERIAL_NONSTANDARD is not set + +# +# Serial drivers +# +# CONFIG_SERIAL_ANAKIN is not set +# CONFIG_SERIAL_ANAKIN_CONSOLE is not set +# CONFIG_SERIAL_AMBA is not set +# CONFIG_SERIAL_AMBA_CONSOLE is not set +# CONFIG_SERIAL_CLPS711X is not set +# CONFIG_SERIAL_CLPS711X_CONSOLE is not set +# CONFIG_SERIAL_21285 is not set +# CONFIG_SERIAL_21285_OLD is not set +# CONFIG_SERIAL_21285_CONSOLE is not set +# CONFIG_SERIAL_UART00 is not set +# CONFIG_SERIAL_UART00_CONSOLE is not set +CONFIG_SERIAL_SA1100=y +CONFIG_SERIAL_SA1100_CONSOLE=y +CONFIG_SA1100_DEFAULT_BAUDRATE=38400 +# CONFIG_SERIAL_OMAHA is not set +# CONFIG_SERIAL_OMAHA_CONSOLE is not set +# CONFIG_SERIAL_8250 is not set +# CONFIG_SERIAL_8250_CONSOLE is not set +# CONFIG_SERIAL_8250_EXTENDED is not set +# CONFIG_SERIAL_8250_MANY_PORTS is not set +# CONFIG_SERIAL_8250_SHARE_IRQ is not set +# CONFIG_SERIAL_8250_DETECT_IRQ is not set +# CONFIG_SERIAL_8250_MULTIPORT is not set +# CONFIG_SERIAL_8250_HUB6 is not set +CONFIG_SERIAL_CORE=y +CONFIG_SERIAL_CORE_CONSOLE=y +CONFIG_TOUCHSCREEN_ADS784X=y +# CONFIG_UCB1200 is not set +# CONFIG_TOUCHSCREEN_UCB1200 is not set +# CONFIG_AUDIO_UCB1200 is not set +# CONFIG_ADC_UCB1200 is not set +CONFIG_UNIX98_PTYS=y +CONFIG_UNIX98_PTY_COUNT=32 + +# +# I2C support +# +CONFIG_I2C=y +# CONFIG_I2C_ALGOBIT is not set +# CONFIG_I2C_ALGOPCF is not set +CONFIG_ADSSIO_I2C_ALGO=y +# CONFIG_I2C_CHARDEV is not set +# CONFIG_I2C_PROC is not set +CONFIG_I2C_DS1307=y +CONFIG_I2C_DS1307_ADS=y + +# +# L3 serial bus support +# +# CONFIG_L3 is not set +# CONFIG_L3_ALGOBIT is not set +# CONFIG_L3_BIT_SA1100_GPIO is not set + +# +# Other L3 adapters +# +# CONFIG_L3_SA1111 is not set +# CONFIG_BIT_SA1100_GPIO is not set + +# +# Mice +# +# CONFIG_BUSMOUSE is not set +# CONFIG_MOUSE is not set + +# +# Joysticks +# +# CONFIG_INPUT_GAMEPORT is not set +# CONFIG_INPUT_NS558 is not set +# CONFIG_INPUT_LIGHTNING is not set +# CONFIG_INPUT_PCIGAME is not set +# CONFIG_INPUT_CS461X is not set +# CONFIG_INPUT_EMU10K1 is not set +# CONFIG_INPUT_SERIO is not set +# CONFIG_INPUT_SERPORT is not set + +# +# Joysticks +# +# CONFIG_INPUT_ANALOG is not set +# CONFIG_INPUT_A3D is not set +# CONFIG_INPUT_ADI is not set +# CONFIG_INPUT_COBRA is not set +# CONFIG_INPUT_GF2K is not set +# CONFIG_INPUT_GRIP is not set +# CONFIG_INPUT_INTERACT is not set +# CONFIG_INPUT_TMDC is not set +# CONFIG_INPUT_SIDEWINDER is not set +# CONFIG_INPUT_IFORCE_USB is not set +# CONFIG_INPUT_IFORCE_232 is not set +# CONFIG_INPUT_WARRIOR is not set +# CONFIG_INPUT_MAGELLAN is not set +# CONFIG_INPUT_SPACEORB is not set +# CONFIG_INPUT_SPACEBALL is not set +# CONFIG_INPUT_STINGER is not set +# CONFIG_INPUT_DB9 is not set +# CONFIG_INPUT_GAMECON is not set +# CONFIG_INPUT_TURBOGRAFX is not set +# CONFIG_QIC02_TAPE is not set + +# +# Watchdog Cards +# +# CONFIG_WATCHDOG is not set +# CONFIG_NVRAM is not set +# CONFIG_RTC is not set +# CONFIG_SA1100_RTC is not set +# CONFIG_DTLK is not set +# CONFIG_R3964 is not set +# CONFIG_APPLICOM is not set + +# +# Ftape, the floppy tape device driver +# +# CONFIG_FTAPE is not set +# CONFIG_AGP is not set +# CONFIG_DRM is not set + +# +# PCMCIA character devices +# + +# +# Multimedia devices +# +# CONFIG_VIDEO_DEV is not set + +# +# File systems +# +# CONFIG_QUOTA is not set +# CONFIG_AUTOFS_FS is not set +# CONFIG_AUTOFS4_FS is not set +# CONFIG_REISERFS_FS is not set +# CONFIG_REISERFS_CHECK is not set +# CONFIG_REISERFS_PROC_INFO is not set +# CONFIG_ADFS_FS is not set +# CONFIG_ADFS_FS_RW is not set +# CONFIG_AFFS_FS is not set +# CONFIG_HFS_FS is not set +# CONFIG_BFS_FS is not set +# CONFIG_EXT3_FS is not set +# CONFIG_JBD is not set +# CONFIG_JBD_DEBUG is not set +CONFIG_FAT_FS=y +# CONFIG_MSDOS_FS is not set +# CONFIG_UMSDOS_FS is not set +CONFIG_VFAT_FS=y +# CONFIG_EFS_FS is not set +# CONFIG_JFFS_FS is not set +CONFIG_JFFS2_FS=y +CONFIG_JFFS2_FS_DEBUG=0 +CONFIG_CRAMFS=y +# CONFIG_TMPFS is not set +CONFIG_RAMFS=y +# CONFIG_ISO9660_FS is not set +# CONFIG_JOLIET is not set +# CONFIG_ZISOFS is not set +# CONFIG_MINIX_FS is not set +# CONFIG_VXFS_FS is not set +# CONFIG_NTFS_FS is not set +# CONFIG_NTFS_RW is not set +# CONFIG_HPFS_FS is not set +CONFIG_PROC_FS=y +# CONFIG_DEVFS_FS is not set +# CONFIG_DEVFS_MOUNT is not set +# CONFIG_DEVFS_DEBUG is not set +CONFIG_DEVPTS_FS=y +# CONFIG_QNX4FS_FS is not set +# CONFIG_QNX4FS_RW is not set +# CONFIG_ROMFS_FS is not set +CONFIG_EXT2_FS=y +# CONFIG_SYSV_FS is not set +# CONFIG_UDF_FS is not set +# CONFIG_UDF_RW is not set +# CONFIG_UFS_FS is not set +# CONFIG_UFS_FS_WRITE is not set + +# +# Network File Systems +# +# CONFIG_CODA_FS is not set +# CONFIG_INTERMEZZO_FS is not set +CONFIG_NFS_FS=y +# CONFIG_NFS_V3 is not set +CONFIG_ROOT_NFS=y +# CONFIG_NFSD is not set +# CONFIG_NFSD_V3 is not set +CONFIG_SUNRPC=y +CONFIG_LOCKD=y +# CONFIG_SMB_FS is not set +# CONFIG_NCP_FS is not set +# CONFIG_NCPFS_PACKET_SIGNING is not set +# CONFIG_NCPFS_IOCTL_LOCKING is not set +# CONFIG_NCPFS_STRONG is not set +# CONFIG_NCPFS_NFS_NS is not set +# CONFIG_NCPFS_OS2_NS is not set +# CONFIG_NCPFS_SMALLDOS is not set +# CONFIG_NCPFS_NLS is not set +# CONFIG_NCPFS_EXTRAS is not set +# CONFIG_ZISOFS_FS is not set +CONFIG_ZLIB_FS_INFLATE=y + +# +# Partition Types +# +# CONFIG_PARTITION_ADVANCED is not set +CONFIG_MSDOS_PARTITION=y +# CONFIG_SMB_NLS is not set +CONFIG_NLS=y + +# +# Native Language Support +# +CONFIG_NLS_DEFAULT="iso8859-1" +# CONFIG_NLS_CODEPAGE_437 is not set +# CONFIG_NLS_CODEPAGE_737 is not set +# CONFIG_NLS_CODEPAGE_775 is not set +# CONFIG_NLS_CODEPAGE_850 is not set +# CONFIG_NLS_CODEPAGE_852 is not set +# CONFIG_NLS_CODEPAGE_855 is not set +# CONFIG_NLS_CODEPAGE_857 is not set +# CONFIG_NLS_CODEPAGE_860 is not set +# CONFIG_NLS_CODEPAGE_861 is not set +# CONFIG_NLS_CODEPAGE_862 is not set +# CONFIG_NLS_CODEPAGE_863 is not set +# CONFIG_NLS_CODEPAGE_864 is not set +# CONFIG_NLS_CODEPAGE_865 is not set +# CONFIG_NLS_CODEPAGE_866 is not set +# CONFIG_NLS_CODEPAGE_869 is not set +# CONFIG_NLS_CODEPAGE_936 is not set +# CONFIG_NLS_CODEPAGE_950 is not set +# CONFIG_NLS_CODEPAGE_932 is not set +# CONFIG_NLS_CODEPAGE_949 is not set +# CONFIG_NLS_CODEPAGE_874 is not set +# CONFIG_NLS_ISO8859_8 is not set +# CONFIG_NLS_CODEPAGE_1250 is not set +# CONFIG_NLS_CODEPAGE_1251 is not set +# CONFIG_NLS_ISO8859_1 is not set +# CONFIG_NLS_ISO8859_2 is not set +# CONFIG_NLS_ISO8859_3 is not set +# CONFIG_NLS_ISO8859_4 is not set +# CONFIG_NLS_ISO8859_5 is not set +# CONFIG_NLS_ISO8859_6 is not set +# CONFIG_NLS_ISO8859_7 is not set +# CONFIG_NLS_ISO8859_9 is not set +# CONFIG_NLS_ISO8859_13 is not set +# CONFIG_NLS_ISO8859_14 is not set +# CONFIG_NLS_ISO8859_15 is not set +# CONFIG_NLS_KOI8_R is not set +# CONFIG_NLS_KOI8_U is not set +# CONFIG_NLS_UTF8 is not set + +# +# Console drivers +# +CONFIG_PC_KEYMAP=y +# CONFIG_VGA_CONSOLE is not set + +# +# Frame-buffer support +# +CONFIG_FB=y +CONFIG_DUMMY_CONSOLE=y +# CONFIG_FB_ACORN is not set +# CONFIG_FB_ANAKIN is not set +# CONFIG_FB_CLPS711X is not set +# CONFIG_FB_SA1100 is not set +# CONFIG_FB_CYBER2000 is not set +CONFIG_FB_E13806=y +CONFIG_E13806_REG_BASE=48000000 +CONFIG_E13806_FB_BASE=48200000 +# CONFIG_FB_VIRTUAL is not set +CONFIG_FBCON_ADVANCED=y +# CONFIG_FBCON_MFB is not set +# CONFIG_FBCON_CFB2 is not set +CONFIG_FBCON_CFB4=y +CONFIG_FBCON_CFB8=y +CONFIG_FBCON_CFB16=y +# CONFIG_FBCON_CFB24 is not set +# CONFIG_FBCON_CFB32 is not set +# CONFIG_FBCON_AFB is not set +# CONFIG_FBCON_ILBM is not set +# CONFIG_FBCON_IPLAN2P2 is not set +# CONFIG_FBCON_IPLAN2P4 is not set +# CONFIG_FBCON_IPLAN2P8 is not set +# CONFIG_FBCON_MAC is not set +# CONFIG_FBCON_VGA_PLANES is not set +# CONFIG_FBCON_VGA is not set +# CONFIG_FBCON_HGA is not set +CONFIG_FBCON_FONTWIDTH8_ONLY=y +CONFIG_FBCON_FONTS=y +# CONFIG_FONT_8x8 is not set +CONFIG_FONT_8x16=y +# CONFIG_FONT_SUN8x16 is not set +# CONFIG_FONT_PEARL_8x8 is not set +# CONFIG_FONT_ACORN_8x8 is not set + +# +# Sound +# +CONFIG_SOUND=y +# CONFIG_SOUND_BT878 is not set +# CONFIG_SOUND_CMPCI is not set +# CONFIG_SOUND_EMU10K1 is not set +# CONFIG_MIDI_EMU10K1 is not set +# CONFIG_SOUND_FUSION is not set +# CONFIG_SOUND_CS4281 is not set +# CONFIG_SOUND_ES1370 is not set +# CONFIG_SOUND_ES1371 is not set +# CONFIG_SOUND_ESSSOLO1 is not set +# CONFIG_SOUND_MAESTRO is not set +# CONFIG_SOUND_MAESTRO3 is not set +# CONFIG_SOUND_ICH is not set +# CONFIG_SOUND_RME96XX is not set +# CONFIG_SOUND_SONICVIBES is not set +# CONFIG_SOUND_TRIDENT is not set +# CONFIG_SOUND_MSNDCLAS is not set +# CONFIG_SOUND_MSNDPIN is not set +# CONFIG_SOUND_VIA82CXXX is not set +# CONFIG_MIDI_VIA82CXXX is not set +CONFIG_SOUND_SA1100=y +# CONFIG_SOUND_UDA1341 is not set +# CONFIG_SOUND_ASSABET_UDA1341 is not set +# CONFIG_SOUND_H3600_UDA1341 is not set +# CONFIG_SOUND_PANGOLIN_UDA1341 is not set +# CONFIG_SOUND_SA1111_UDA1341 is not set +CONFIG_SOUND_SA1111_AC97=y +# CONFIG_SOUND_SA1100SSP is not set +CONFIG_SOUND_OSS=y +# CONFIG_SOUND_TRACEINIT is not set +# CONFIG_SOUND_DMAP is not set +# CONFIG_SOUND_AD1816 is not set +# CONFIG_SOUND_SGALAXY is not set +# CONFIG_SOUND_ADLIB is not set +# CONFIG_SOUND_ACI_MIXER is not set +# CONFIG_SOUND_CS4232 is not set +# CONFIG_SOUND_SSCAPE is not set +# CONFIG_SOUND_GUS is not set +# CONFIG_SOUND_VMIDI is not set +# CONFIG_SOUND_TRIX is not set +# CONFIG_SOUND_MSS is not set +# CONFIG_SOUND_MPU401 is not set +# CONFIG_SOUND_NM256 is not set +# CONFIG_SOUND_MAD16 is not set +# CONFIG_SOUND_PAS is not set +# CONFIG_PAS_JOYSTICK is not set +# CONFIG_SOUND_PSS is not set +# CONFIG_SOUND_SB is not set +# CONFIG_SOUND_AWE32_SYNTH is not set +# CONFIG_SOUND_WAVEFRONT is not set +# CONFIG_SOUND_MAUI is not set +# CONFIG_SOUND_YM3812 is not set +# CONFIG_SOUND_OPL3SA1 is not set +# CONFIG_SOUND_OPL3SA2 is not set +# CONFIG_SOUND_YMFPCI is not set +# CONFIG_SOUND_YMFPCI_LEGACY is not set +# CONFIG_SOUND_UART6850 is not set +# CONFIG_SOUND_AEDSP16 is not set +# CONFIG_SOUND_VIDC is not set +# CONFIG_SOUND_WAVEARTIST is not set +# CONFIG_SOUND_TVMIXER is not set + +# +# Multimedia Capabilities Port drivers +# +# CONFIG_MCP is not set +# CONFIG_MCP_SA1100 is not set +# CONFIG_MCP_UCB1200 is not set +# CONFIG_MCP_UCB1200_AUDIO is not set +# CONFIG_MCP_UCB1200_TS is not set + +# +# USB support +# +CONFIG_USB=y +# CONFIG_USB_DEBUG is not set + +# +# Miscellaneous USB options +# +CONFIG_USB_DEVICEFS=y +# CONFIG_USB_BANDWIDTH is not set +# CONFIG_USB_LONG_TIMEOUT is not set + +# +# USB Host Controller Drivers +# +# CONFIG_USB_EHCI_HCD is not set +# CONFIG_USB_UHCI is not set +# CONFIG_USB_UHCI_ALT is not set +# CONFIG_USB_OHCI is not set +CONFIG_USB_OHCI_SA1111=y + +# +# USB Device Class drivers +# +# CONFIG_USB_AUDIO is not set +# CONFIG_USB_EMI26 is not set +# CONFIG_USB_BLUETOOTH is not set + +# +# SCSI support is needed for USB Storage +# +# CONFIG_USB_STORAGE is not set +# CONFIG_USB_STORAGE_DEBUG is not set +# CONFIG_USB_STORAGE_DATAFAB is not set +# CONFIG_USB_STORAGE_FREECOM is not set +# CONFIG_USB_STORAGE_ISD200 is not set +# CONFIG_USB_STORAGE_DPCM is not set +# CONFIG_USB_STORAGE_HP8200e is not set +# CONFIG_USB_STORAGE_SDDR09 is not set +# CONFIG_USB_STORAGE_JUMPSHOT is not set +# CONFIG_USB_ACM is not set +# CONFIG_USB_PRINTER is not set + +# +# USB Human Interface Devices (HID) +# +CONFIG_USB_HID=y +CONFIG_USB_HIDINPUT=y +# CONFIG_USB_HIDDEV is not set +# CONFIG_USB_WACOM is not set + +# +# USB Imaging devices +# +# CONFIG_USB_DC2XX is not set +# CONFIG_USB_MDC800 is not set +# CONFIG_USB_SCANNER is not set +# CONFIG_USB_MICROTEK is not set +# CONFIG_USB_HPUSBSCSI is not set + +# +# USB Multimedia devices +# + +# +# Video4Linux support is needed for USB Multimedia device support +# + +# +# USB Network adaptors +# +# CONFIG_USB_PEGASUS is not set +# CONFIG_USB_RTL8150 is not set +# CONFIG_USB_KAWETH is not set +# CONFIG_USB_CATC is not set +# CONFIG_USB_CDCETHER is not set +# CONFIG_USB_USBNET is not set + +# +# USB port drivers +# +# CONFIG_USB_USS720 is not set + +# +# USB Serial Converter support +# +# CONFIG_USB_SERIAL is not set +# CONFIG_USB_SERIAL_GENERIC is not set +# CONFIG_USB_SERIAL_BELKIN is not set +# CONFIG_USB_SERIAL_WHITEHEAT is not set +# CONFIG_USB_SERIAL_DIGI_ACCELEPORT is not set +# CONFIG_USB_SERIAL_EMPEG is not set +# CONFIG_USB_SERIAL_FTDI_SIO is not set +# CONFIG_USB_SERIAL_VISOR is not set +# CONFIG_USB_SERIAL_IPAQ is not set +# CONFIG_USB_SERIAL_IR is not set +# CONFIG_USB_SERIAL_EDGEPORT is not set +# CONFIG_USB_SERIAL_KEYSPAN_PDA is not set +# CONFIG_USB_SERIAL_KEYSPAN is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XA is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XB is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA18X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19W is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA49W is not set +# CONFIG_USB_SERIAL_MCT_U232 is not set +# CONFIG_USB_SERIAL_KLSI is not set +# CONFIG_USB_SERIAL_PL2303 is not set +# CONFIG_USB_SERIAL_CYBERJACK is not set +# CONFIG_USB_SERIAL_XIRCOM is not set +# CONFIG_USB_SERIAL_OMNINET is not set + +# +# USB Miscellaneous drivers +# +# CONFIG_USB_RIO500 is not set +# CONFIG_USB_AUERSWALD is not set +# CONFIG_USB_BRLVGER is not set + +# +# Bluetooth support +# +# CONFIG_BLUEZ is not set + +# +# Kernel hacking +# +CONFIG_FRAME_POINTER=y +# CONFIG_DEBUG_USER is not set +# CONFIG_DEBUG_INFO is not set +# CONFIG_NO_PGT_CACHE is not set +# CONFIG_DEBUG_KERNEL is not set +# CONFIG_DEBUG_SLAB is not set +# CONFIG_MAGIC_SYSRQ is not set +# CONFIG_DEBUG_SPINLOCK is not set +# CONFIG_DEBUG_WAITQ is not set +# CONFIG_DEBUG_BUGVERBOSE is not set +# CONFIG_DEBUG_ERRORS is not set +# CONFIG_DEBUG_LL is not set +# CONFIG_DEBUG_DC21285_PORT is not set +# CONFIG_DEBUG_CLPS711X_UART2 is not set diff -urN orig/arch/arm/def-configs/adsbitsy linux/arch/arm/def-configs/adsbitsy --- orig/arch/arm/def-configs/adsbitsy Mon Aug 5 13:29:42 2002 +++ linux/arch/arm/def-configs/adsbitsy Thu Feb 27 23:20:10 2003 @@ -1,5 +1,5 @@ # -# Automatically generated by make menuconfig: don't edit +# Automatically generated make config: don't edit # CONFIG_ARM=y # CONFIG_EISA is not set @@ -8,6 +8,8 @@ CONFIG_UID16=y CONFIG_RWSEM_GENERIC_SPINLOCK=y # CONFIG_RWSEM_XCHGADD_ALGORITHM is not set +# CONFIG_GENERIC_BUST_SPINLOCK is not set +# CONFIG_GENERIC_ISA_DMA is not set # # Code maturity level options @@ -31,16 +33,25 @@ # CONFIG_ARCH_CLPS711X is not set # CONFIG_ARCH_CO285 is not set # CONFIG_ARCH_EBSA110 is not set +# CONFIG_ARCH_CAMELOT is not set # CONFIG_ARCH_FOOTBRIDGE is not set # CONFIG_ARCH_INTEGRATOR is not set +# CONFIG_ARCH_OMAHA is not set # CONFIG_ARCH_L7200 is not set +# CONFIG_ARCH_MX1ADS is not set # CONFIG_ARCH_RPC is not set +# CONFIG_ARCH_RISCSTATION is not set CONFIG_ARCH_SA1100=y # CONFIG_ARCH_SHARK is not set +# CONFIG_ARCH_AT91RM9200DK is not set # # Archimedes/A5000 Implementations # + +# +# Archimedes/A5000 Implementations (select only ONE) +# # CONFIG_ARCH_ARC is not set # CONFIG_ARCH_A5K is not set @@ -56,15 +67,27 @@ # # SA11x0 Implementations # +# CONFIG_SA1100_ACCELENT is not set # CONFIG_SA1100_ASSABET is not set # CONFIG_ASSABET_NEPONSET is not set +# CONFIG_SA1100_ADSAGC is not set +CONFIG_SA1100_ADSBITSY=y +# CONFIG_SA1100_ADSBITSYPLUS is not set # CONFIG_SA1100_BRUTUS is not set +# CONFIG_SA1100_CEP is not set # CONFIG_SA1100_CERF is not set -# CONFIG_SA1100_BITSY is not set +# CONFIG_SA1100_H3100 is not set +# CONFIG_SA1100_H3600 is not set +# CONFIG_SA1100_H3800 is not set +# CONFIG_SA1100_H3XXX is not set # CONFIG_SA1100_EXTENEX1 is not set # CONFIG_SA1100_FLEXANET is not set # CONFIG_SA1100_FREEBIRD is not set +# CONFIG_SA1100_FRODO is not set # CONFIG_SA1100_GRAPHICSCLIENT is not set +# CONFIG_SA1100_GRAPHICSMASTER is not set +# CONFIG_SA1100_HACKKIT is not set +# CONFIG_SA1100_BADGE4 is not set # CONFIG_SA1100_JORNADA720 is not set # CONFIG_SA1100_HUW_WEBPANEL is not set # CONFIG_SA1100_ITSY is not set @@ -73,40 +96,61 @@ # CONFIG_SA1100_OMNIMETER is not set # CONFIG_SA1100_PANGOLIN is not set # CONFIG_SA1100_PLEB is not set +# CONFIG_SA1100_PT_SYSTEM3 is not set +# CONFIG_SA1100_SHANNON is not set # CONFIG_SA1100_SHERMAN is not set # CONFIG_SA1100_SIMPAD is not set +# CONFIG_SA1100_SIMPUTER is not set # CONFIG_SA1100_PFS168 is not set # CONFIG_SA1100_VICTOR is not set # CONFIG_SA1100_XP860 is not set # CONFIG_SA1100_YOPY is not set -# CONFIG_SA1100_GRAPHICSMASTER is not set -CONFIG_SA1100_ADSBITSY=y -CONFIG_SA1111=y # CONFIG_SA1100_USB is not set # CONFIG_SA1100_USB_NETLINK is not set # CONFIG_SA1100_USB_CHAR is not set +# CONFIG_H3600_SLEEVE is not set # # CLPS711X/EP721X Implementations # +# CONFIG_ARCH_AUTCPU12 is not set +# CONFIG_ARCH_CDB89712 is not set +# CONFIG_ARCH_CLEP7312 is not set +# CONFIG_ARCH_EDB7211 is not set # CONFIG_ARCH_P720T is not set +# CONFIG_ARCH_FORTUNET is not set +# CONFIG_ARCH_EP7211 is not set +# CONFIG_ARCH_EP7212 is not set # CONFIG_ARCH_ACORN is not set # CONFIG_FOOTBRIDGE is not set # CONFIG_FOOTBRIDGE_HOST is not set # CONFIG_FOOTBRIDGE_ADDIN is not set +CONFIG_SA1111=y +CONFIG_FORCE_MAX_ZONEORDER=9 + +# +# Processor Type +# CONFIG_CPU_32=y # CONFIG_CPU_26 is not set -# CONFIG_CPU_32v3 is not set -CONFIG_CPU_32v4=y # CONFIG_CPU_ARM610 is not set # CONFIG_CPU_ARM710 is not set # CONFIG_CPU_ARM720T is not set # CONFIG_CPU_ARM920T is not set +# CONFIG_CPU_ARM922T is not set +# CONFIG_PLD is not set +# CONFIG_CPU_ARM926T is not set # CONFIG_CPU_ARM1020 is not set +# CONFIG_CPU_ARM1026 is not set # CONFIG_CPU_SA110 is not set CONFIG_CPU_SA1100=y +# CONFIG_CPU_32v3 is not set +CONFIG_CPU_32v4=y + +# +# Processor Features +# CONFIG_DISCONTIGMEM=y -# CONFIG_CPU_BIG_ENDIAN is not set # # General setup @@ -114,6 +158,9 @@ # CONFIG_PCI is not set # CONFIG_ISA is not set # CONFIG_ISA_DMA is not set +# CONFIG_ZBOOT_ROM is not set +CONFIG_ZBOOT_ROM_TEXT=0 +CONFIG_ZBOOT_ROM_BSS=0 # CONFIG_CPU_FREQ is not set CONFIG_HOTPLUG=y @@ -121,6 +168,8 @@ # PCMCIA/CardBus support # CONFIG_PCMCIA=y +CONFIG_PCMCIA_PROBE=y +# CONFIG_I82092 is not set # CONFIG_I82365 is not set # CONFIG_TCIC is not set # CONFIG_PCMCIA_CLPS6700 is not set @@ -129,6 +178,10 @@ CONFIG_SYSVIPC=y # CONFIG_BSD_PROCESS_ACCT is not set CONFIG_SYSCTL=y + +# +# At least one math emulation must be selected +# CONFIG_FPE_NWFPE=y # CONFIG_FPE_FASTFPE is not set CONFIG_KCORE_ELF=y @@ -136,12 +189,12 @@ # CONFIG_BINFMT_AOUT is not set CONFIG_BINFMT_ELF=y # CONFIG_BINFMT_MISC is not set -# CONFIG_PM is not set -# CONFIG_APM is not set +CONFIG_PM=y # CONFIG_ARTHUR is not set -CONFIG_CMDLINE="ip=off" -# CONFIG_PFS168_CMDLINE is not set -# CONFIG_LEDS is not set +CONFIG_CMDLINE="console=ttySA0 mem=32m@0xC0000000 rw ramdisk_size=8192 initrd=0xC0800000,3m root=/dev/ram" +CONFIG_LEDS=y +CONFIG_LEDS_TIMER=y +# CONFIG_LEDS_CPU is not set CONFIG_ALIGNMENT_TRAP=y # @@ -152,7 +205,79 @@ # # Memory Technology Devices (MTD) # -# CONFIG_MTD is not set +CONFIG_MTD=y +# CONFIG_MTD_DEBUG is not set +CONFIG_MTD_PARTITIONS=y +# CONFIG_MTD_CONCAT is not set +# CONFIG_MTD_REDBOOT_PARTS is not set +CONFIG_MTD_CMDLINE_PARTS=y +# CONFIG_MTD_AFS_PARTS is not set + +# +# User Modules And Translation Layers +# +CONFIG_MTD_CHAR=y +CONFIG_MTD_BLOCK=y +# CONFIG_FTL is not set +# CONFIG_NFTL is not set + +# +# RAM/ROM/Flash chip drivers +# +CONFIG_MTD_CFI=y +# CONFIG_MTD_JEDECPROBE is not set +CONFIG_MTD_GEN_PROBE=y +# CONFIG_MTD_CFI_ADV_OPTIONS is not set +CONFIG_MTD_CFI_INTELEXT=y +# CONFIG_MTD_CFI_AMDSTD is not set +# CONFIG_MTD_RAM is not set +# CONFIG_MTD_ROM is not set +# CONFIG_MTD_ABSENT is not set +# CONFIG_MTD_OBSOLETE_CHIPS is not set +# CONFIG_MTD_AMDSTD is not set +# CONFIG_MTD_SHARP is not set +# CONFIG_MTD_JEDEC is not set + +# +# Mapping drivers for chip access +# +CONFIG_MTD_PHYSMAP=y +CONFIG_MTD_PHYSMAP_START=8000000 +CONFIG_MTD_PHYSMAP_LEN=4000000 +CONFIG_MTD_PHYSMAP_BUSWIDTH=4 +# CONFIG_MTD_NORA is not set +# CONFIG_MTD_ARM_INTEGRATOR is not set +# CONFIG_MTD_CDB89712 is not set +CONFIG_MTD_SA1100=y +# CONFIG_MTD_DC21285 is not set +# CONFIG_MTD_IQ80310 is not set +# CONFIG_MTD_FORTUNET is not set +# CONFIG_MTD_EPXA10DB is not set +# CONFIG_MTD_AUTCPU12 is not set +# CONFIG_MTD_EDB7312 is not set +# CONFIG_MTD_IMPA7 is not set +# CONFIG_MTD_PCI is not set + +# +# Self-contained MTD device drivers +# +# CONFIG_MTD_PMC551 is not set +# CONFIG_MTD_SLRAM is not set +# CONFIG_MTD_MTDRAM is not set +# CONFIG_MTD_BLKMTD is not set + +# +# Disk-On-Chip Device Drivers +# +# CONFIG_MTD_DOC1000 is not set +# CONFIG_MTD_DOC2000 is not set +# CONFIG_MTD_DOC2001 is not set +# CONFIG_MTD_DOCPROBE is not set + +# +# NAND Flash Device Drivers +# +# CONFIG_MTD_NAND is not set # # Plug and Play configuration @@ -168,7 +293,9 @@ # CONFIG_PARIDE is not set # CONFIG_BLK_CPQ_DA is not set # CONFIG_BLK_CPQ_CISS_DA is not set +# CONFIG_CISS_SCSI_TAPE is not set # CONFIG_BLK_DEV_DAC960 is not set +# CONFIG_BLK_DEV_UMEM is not set CONFIG_BLK_DEV_LOOP=y # CONFIG_BLK_DEV_NBD is not set CONFIG_BLK_DEV_RAM=y @@ -184,6 +311,7 @@ # CONFIG_MD_RAID0 is not set # CONFIG_MD_RAID1 is not set # CONFIG_MD_RAID5 is not set +# CONFIG_MD_MULTIPATH is not set # CONFIG_BLK_DEV_LVM is not set # @@ -191,7 +319,7 @@ # CONFIG_PACKET=y # CONFIG_PACKET_MMAP is not set -# CONFIG_NETLINK is not set +# CONFIG_NETLINK_DEV is not set # CONFIG_NETFILTER is not set # CONFIG_FILTER is not set CONFIG_UNIX=y @@ -204,13 +332,24 @@ # CONFIG_IP_PNP_RARP is not set # CONFIG_NET_IPIP is not set # CONFIG_NET_IPGRE is not set +# CONFIG_ARPD is not set # CONFIG_INET_ECN is not set # CONFIG_SYN_COOKIES is not set # CONFIG_IPV6 is not set # CONFIG_KHTTPD is not set # CONFIG_ATM is not set +# CONFIG_VLAN_8021Q is not set + +# +# +# # CONFIG_IPX is not set # CONFIG_ATALK is not set + +# +# Appletalk devices +# +# CONFIG_DEV_APPLETALK is not set # CONFIG_DECNET is not set # CONFIG_BRIDGE is not set # CONFIG_X25 is not set @@ -228,6 +367,11 @@ # CONFIG_NET_SCHED is not set # +# Network testing +# +# CONFIG_NET_PKTGEN is not set + +# # Network device support # CONFIG_NETDEVICES=y @@ -240,19 +384,27 @@ # CONFIG_BONDING is not set # CONFIG_EQUALIZER is not set # CONFIG_TUN is not set +# CONFIG_ETHERTAP is not set # # Ethernet (10 or 100Mbit) # CONFIG_NET_ETHERNET=y +# CONFIG_ARM_AM79C961A is not set +# CONFIG_ARM_CIRRUS is not set # CONFIG_SUNLANCE is not set # CONFIG_SUNBMAC is not set # CONFIG_SUNQE is not set -# CONFIG_SUNLANCE is not set # CONFIG_SUNGEM is not set # CONFIG_NET_VENDOR_3COM is not set # CONFIG_LANCE is not set -# CONFIG_NET_VENDOR_SMC is not set +CONFIG_NET_VENDOR_SMC=y +# CONFIG_WD80x3 is not set +# CONFIG_ULTRAMCA is not set +# CONFIG_ULTRA is not set +# CONFIG_ULTRA32 is not set +# CONFIG_SMC9194 is not set +CONFIG_SMC91111=y # CONFIG_NET_VENDOR_RACAL is not set # CONFIG_NET_ISA is not set # CONFIG_NET_PCI is not set @@ -262,21 +414,44 @@ # Ethernet (1000 Mbit) # # CONFIG_ACENIC is not set -# CONFIG_ACENIC_OMIT_TIGON_I is not set +# CONFIG_DL2K is not set # CONFIG_MYRI_SBUS is not set +# CONFIG_NS83820 is not set # CONFIG_HAMACHI is not set # CONFIG_YELLOWFIN is not set # CONFIG_SK98LIN is not set +# CONFIG_TIGON3 is not set # CONFIG_FDDI is not set # CONFIG_HIPPI is not set # CONFIG_PLIP is not set -# CONFIG_PPP is not set +CONFIG_PPP=y +# CONFIG_PPP_MULTILINK is not set +# CONFIG_PPP_FILTER is not set +CONFIG_PPP_ASYNC=y +# CONFIG_PPP_SYNC_TTY is not set +# CONFIG_PPP_DEFLATE is not set +# CONFIG_PPP_BSDCOMP is not set +# CONFIG_PPPOE is not set # CONFIG_SLIP is not set # # Wireless LAN (non-hamradio) # -# CONFIG_NET_RADIO is not set +CONFIG_NET_RADIO=y +# CONFIG_STRIP is not set +# CONFIG_WAVELAN is not set +# CONFIG_ARLAN is not set +# CONFIG_AIRONET4500 is not set +# CONFIG_AIRONET4500_NONCS is not set +# CONFIG_AIRONET4500_PROC is not set +# CONFIG_HERMES is not set + +# +# Wireless Pcmcia cards support +# +# CONFIG_PCMCIA_HERMES is not set +# CONFIG_AIRO_CS is not set +CONFIG_NET_WIRELESS=y # # Token Ring devices @@ -299,6 +474,7 @@ # CONFIG_PCMCIA_3C574 is not set # CONFIG_PCMCIA_FMVJ18X is not set CONFIG_PCMCIA_PCNET=y +# CONFIG_PCMCIA_AXNET is not set # CONFIG_PCMCIA_NMCLAN is not set # CONFIG_PCMCIA_SMC91C92 is not set # CONFIG_PCMCIA_XIRC2PS is not set @@ -317,7 +493,7 @@ # CONFIG_IRDA is not set # -# ATA/IDE/MFM/RLL support +# ATA/ATAPI/MFM/RLL support # CONFIG_IDE=y @@ -325,10 +501,15 @@ # IDE, ATA and ATAPI Block devices # CONFIG_BLK_DEV_IDE=y + +# +# Please see Documentation/ide.txt for help/info on IDE drives +# # CONFIG_BLK_DEV_HD_IDE is not set # CONFIG_BLK_DEV_HD is not set CONFIG_BLK_DEV_IDEDISK=y # CONFIG_IDEDISK_MULTI_MODE is not set +# CONFIG_IDEDISK_STROKE is not set # CONFIG_BLK_DEV_IDEDISK_VENDOR is not set # CONFIG_BLK_DEV_IDEDISK_FUJITSU is not set # CONFIG_BLK_DEV_IDEDISK_IBM is not set @@ -343,6 +524,11 @@ # CONFIG_BLK_DEV_IDETAPE is not set # CONFIG_BLK_DEV_IDEFLOPPY is not set # CONFIG_BLK_DEV_IDESCSI is not set +# CONFIG_IDE_TASK_IOCTL is not set + +# +# IDE chipset support/bugfixes +# # CONFIG_BLK_DEV_CMD640 is not set # CONFIG_BLK_DEV_CMD640_ENHANCED is not set # CONFIG_BLK_DEV_ISAPNP is not set @@ -350,6 +536,9 @@ # CONFIG_IDEDMA_AUTO is not set # CONFIG_DMA_NONPCI is not set # CONFIG_BLK_DEV_IDE_MODES is not set +# CONFIG_BLK_DEV_ATARAID is not set +# CONFIG_BLK_DEV_ATARAID_PDC is not set +# CONFIG_BLK_DEV_ATARAID_HPT is not set # # SCSI support @@ -385,35 +574,47 @@ # Character devices # CONFIG_VT=y -# CONFIG_VT_CONSOLE is not set +CONFIG_VT_CONSOLE=y # CONFIG_SERIAL is not set # CONFIG_SERIAL_EXTENDED is not set # CONFIG_SERIAL_NONSTANDARD is not set -# CONFIG_SERIAL_21285 is not set -# CONFIG_SERIAL_21285_OLD is not set -# CONFIG_SERIAL_21285_CONSOLE is not set + +# +# Serial drivers +# +# CONFIG_SERIAL_ANAKIN is not set +# CONFIG_SERIAL_ANAKIN_CONSOLE is not set # CONFIG_SERIAL_AMBA is not set # CONFIG_SERIAL_AMBA_CONSOLE is not set # CONFIG_SERIAL_CLPS711X is not set # CONFIG_SERIAL_CLPS711X_CONSOLE is not set +# CONFIG_SERIAL_21285 is not set +# CONFIG_SERIAL_21285_OLD is not set +# CONFIG_SERIAL_21285_CONSOLE is not set +# CONFIG_SERIAL_UART00 is not set +# CONFIG_SERIAL_UART00_CONSOLE is not set CONFIG_SERIAL_SA1100=y CONFIG_SERIAL_SA1100_CONSOLE=y CONFIG_SA1100_DEFAULT_BAUDRATE=38400 +# CONFIG_SERIAL_OMAHA is not set +# CONFIG_SERIAL_OMAHA_CONSOLE is not set # CONFIG_SERIAL_8250 is not set # CONFIG_SERIAL_8250_CONSOLE is not set +# CONFIG_SERIAL_8250_EXTENDED is not set +# CONFIG_SERIAL_8250_MANY_PORTS is not set +# CONFIG_SERIAL_8250_SHARE_IRQ is not set +# CONFIG_SERIAL_8250_DETECT_IRQ is not set +# CONFIG_SERIAL_8250_MULTIPORT is not set +# CONFIG_SERIAL_8250_HUB6 is not set CONFIG_SERIAL_CORE=y CONFIG_SERIAL_CORE_CONSOLE=y -CONFIG_UNIX98_PTYS=y -CONFIG_UNIX98_PTY_COUNT=32 +# CONFIG_TOUCHSCREEN_ADS784X is not set CONFIG_UCB1200=y CONFIG_TOUCHSCREEN_UCB1200=y CONFIG_AUDIO_UCB1200=y CONFIG_ADC_UCB1200=y -# CONFIG_TOUCHSCREEN_BITSY is not set -# CONFIG_PROFILER is not set -# CONFIG_PFS168_SPI is not set -# CONFIG_PFS168_DTMF is not set -# CONFIG_PFS168_MISC is not set +CONFIG_UNIX98_PTYS=y +CONFIG_UNIX98_PTY_COUNT=32 # # I2C support @@ -421,6 +622,19 @@ # CONFIG_I2C is not set # +# L3 serial bus support +# +CONFIG_L3=y +# CONFIG_L3_ALGOBIT is not set +# CONFIG_L3_BIT_SA1100_GPIO is not set + +# +# Other L3 adapters +# +CONFIG_L3_SA1111=y +# CONFIG_BIT_SA1100_GPIO is not set + +# # Mice # # CONFIG_BUSMOUSE is not set @@ -429,14 +643,43 @@ # # Joysticks # -# CONFIG_JOYSTICK is not set +# CONFIG_INPUT_GAMEPORT is not set +# CONFIG_INPUT_NS558 is not set +# CONFIG_INPUT_LIGHTNING is not set +# CONFIG_INPUT_PCIGAME is not set +# CONFIG_INPUT_CS461X is not set +# CONFIG_INPUT_EMU10K1 is not set +# CONFIG_INPUT_SERIO is not set +# CONFIG_INPUT_SERPORT is not set + +# +# Joysticks +# +# CONFIG_INPUT_ANALOG is not set +# CONFIG_INPUT_A3D is not set +# CONFIG_INPUT_ADI is not set +# CONFIG_INPUT_COBRA is not set +# CONFIG_INPUT_GF2K is not set +# CONFIG_INPUT_GRIP is not set +# CONFIG_INPUT_INTERACT is not set +# CONFIG_INPUT_TMDC is not set +# CONFIG_INPUT_SIDEWINDER is not set +# CONFIG_INPUT_IFORCE_USB is not set +# CONFIG_INPUT_IFORCE_232 is not set +# CONFIG_INPUT_WARRIOR is not set +# CONFIG_INPUT_MAGELLAN is not set +# CONFIG_INPUT_SPACEORB is not set +# CONFIG_INPUT_SPACEBALL is not set +# CONFIG_INPUT_STINGER is not set +# CONFIG_INPUT_DB9 is not set +# CONFIG_INPUT_GAMECON is not set +# CONFIG_INPUT_TURBOGRAFX is not set # CONFIG_QIC02_TAPE is not set # # Watchdog Cards # # CONFIG_WATCHDOG is not set -# CONFIG_INTEL_RNG is not set # CONFIG_NVRAM is not set # CONFIG_RTC is not set # CONFIG_SA1100_RTC is not set @@ -468,23 +711,29 @@ # CONFIG_AUTOFS4_FS is not set # CONFIG_REISERFS_FS is not set # CONFIG_REISERFS_CHECK is not set +# CONFIG_REISERFS_PROC_INFO is not set # CONFIG_ADFS_FS is not set # CONFIG_ADFS_FS_RW is not set # CONFIG_AFFS_FS is not set # CONFIG_HFS_FS is not set # CONFIG_BFS_FS is not set +# CONFIG_EXT3_FS is not set +# CONFIG_JBD is not set +# CONFIG_JBD_DEBUG is not set CONFIG_FAT_FS=y # CONFIG_MSDOS_FS is not set # CONFIG_UMSDOS_FS is not set CONFIG_VFAT_FS=y # CONFIG_EFS_FS is not set # CONFIG_JFFS_FS is not set -# CONFIG_JFFS2_FS is not set +CONFIG_JFFS2_FS=y +CONFIG_JFFS2_FS_DEBUG=0 CONFIG_CRAMFS=y # CONFIG_TMPFS is not set CONFIG_RAMFS=y # CONFIG_ISO9660_FS is not set # CONFIG_JOLIET is not set +# CONFIG_ZISOFS is not set # CONFIG_MINIX_FS is not set # CONFIG_VXFS_FS is not set # CONFIG_NTFS_FS is not set @@ -509,6 +758,7 @@ # Network File Systems # # CONFIG_CODA_FS is not set +# CONFIG_INTERMEZZO_FS is not set CONFIG_NFS_FS=y # CONFIG_NFS_V3 is not set CONFIG_ROOT_NFS=y @@ -526,6 +776,8 @@ # CONFIG_NCPFS_SMALLDOS is not set # CONFIG_NCPFS_NLS is not set # CONFIG_NCPFS_EXTRAS is not set +# CONFIG_ZISOFS_FS is not set +CONFIG_ZLIB_FS_INFLATE=y # # Partition Types @@ -560,6 +812,7 @@ # CONFIG_NLS_CODEPAGE_949 is not set # CONFIG_NLS_CODEPAGE_874 is not set # CONFIG_NLS_ISO8859_8 is not set +# CONFIG_NLS_CODEPAGE_1250 is not set # CONFIG_NLS_CODEPAGE_1251 is not set # CONFIG_NLS_ISO8859_1 is not set # CONFIG_NLS_ISO8859_2 is not set @@ -588,11 +841,11 @@ CONFIG_FB=y CONFIG_DUMMY_CONSOLE=y # CONFIG_FB_ACORN is not set +# CONFIG_FB_ANAKIN is not set # CONFIG_FB_CLPS711X is not set -# CONFIG_FB_CYBER2000 is not set CONFIG_FB_SA1100=y -# CONFIG_FB_ANAKIN is not set -# CONFIG_FB_E1355 is not set +# CONFIG_FB_CYBER2000 is not set +# CONFIG_FB_E13806 is not set # CONFIG_FB_VIRTUAL is not set # CONFIG_FBCON_ADVANCED is not set CONFIG_FBCON_CFB2=y @@ -610,50 +863,169 @@ # # Sound # -# CONFIG_SOUND is not set +CONFIG_SOUND=y +# CONFIG_SOUND_BT878 is not set +# CONFIG_SOUND_CMPCI is not set +# CONFIG_SOUND_EMU10K1 is not set +# CONFIG_MIDI_EMU10K1 is not set +# CONFIG_SOUND_FUSION is not set +# CONFIG_SOUND_CS4281 is not set +# CONFIG_SOUND_ES1370 is not set +# CONFIG_SOUND_ES1371 is not set +# CONFIG_SOUND_ESSSOLO1 is not set +# CONFIG_SOUND_MAESTRO is not set +# CONFIG_SOUND_MAESTRO3 is not set +# CONFIG_SOUND_ICH is not set +# CONFIG_SOUND_RME96XX is not set +# CONFIG_SOUND_SONICVIBES is not set +# CONFIG_SOUND_TRIDENT is not set +# CONFIG_SOUND_MSNDCLAS is not set +# CONFIG_SOUND_MSNDPIN is not set +# CONFIG_SOUND_VIA82CXXX is not set +# CONFIG_MIDI_VIA82CXXX is not set +CONFIG_SOUND_SA1100=y +CONFIG_SOUND_UDA1341=y +# CONFIG_SOUND_ASSABET_UDA1341 is not set +# CONFIG_SOUND_H3600_UDA1341 is not set +# CONFIG_SOUND_PANGOLIN_UDA1341 is not set +CONFIG_SOUND_SA1111_UDA1341=y +# CONFIG_SOUND_SA1111_AC97 is not set +# CONFIG_SOUND_SA1100SSP is not set +# CONFIG_SOUND_OSS is not set +# CONFIG_SOUND_VIDC is not set +# CONFIG_SOUND_WAVEARTIST is not set +# CONFIG_SOUND_TVMIXER is not set + +# +# Multimedia Capabilities Port drivers +# +# CONFIG_MCP is not set +# CONFIG_MCP_SA1100 is not set +# CONFIG_MCP_UCB1200 is not set +# CONFIG_MCP_UCB1200_AUDIO is not set +# CONFIG_MCP_UCB1200_TS is not set # # USB support # CONFIG_USB=y # CONFIG_USB_DEBUG is not set + +# +# Miscellaneous USB options +# CONFIG_USB_DEVICEFS=y # CONFIG_USB_BANDWIDTH is not set +# CONFIG_USB_LONG_TIMEOUT is not set + +# +# USB Host Controller Drivers +# +# CONFIG_USB_EHCI_HCD is not set # CONFIG_USB_UHCI is not set # CONFIG_USB_UHCI_ALT is not set -CONFIG_USB_OHCI=y -CONFIG_USB_OHCI_NOPCI=y +# CONFIG_USB_OHCI is not set +CONFIG_USB_OHCI_SA1111=y + +# +# USB Device Class drivers +# # CONFIG_USB_AUDIO is not set +# CONFIG_USB_EMI26 is not set # CONFIG_USB_BLUETOOTH is not set + +# +# SCSI support is needed for USB Storage +# # CONFIG_USB_STORAGE is not set +# CONFIG_USB_STORAGE_DEBUG is not set +# CONFIG_USB_STORAGE_DATAFAB is not set +# CONFIG_USB_STORAGE_FREECOM is not set +# CONFIG_USB_STORAGE_ISD200 is not set +# CONFIG_USB_STORAGE_DPCM is not set +# CONFIG_USB_STORAGE_HP8200e is not set +# CONFIG_USB_STORAGE_SDDR09 is not set +# CONFIG_USB_STORAGE_JUMPSHOT is not set # CONFIG_USB_ACM is not set # CONFIG_USB_PRINTER is not set -# CONFIG_USB_HID is not set -# CONFIG_USB_KBD is not set -CONFIG_USB_MOUSE=y + +# +# USB Human Interface Devices (HID) +# +CONFIG_USB_HID=y +CONFIG_USB_HIDINPUT=y +# CONFIG_USB_HIDDEV is not set # CONFIG_USB_WACOM is not set + +# +# USB Imaging devices +# # CONFIG_USB_DC2XX is not set # CONFIG_USB_MDC800 is not set # CONFIG_USB_SCANNER is not set # CONFIG_USB_MICROTEK is not set -# CONFIG_USB_IBMCAM is not set -# CONFIG_USB_OV511 is not set -# CONFIG_USB_PWC is not set -# CONFIG_USB_SE401 is not set -# CONFIG_USB_DSBR is not set -# CONFIG_USB_DABUSB is not set -# CONFIG_USB_PLUSB is not set +# CONFIG_USB_HPUSBSCSI is not set + +# +# USB Multimedia devices +# + +# +# Video4Linux support is needed for USB Multimedia device support +# + +# +# USB Network adaptors +# # CONFIG_USB_PEGASUS is not set +# CONFIG_USB_RTL8150 is not set +# CONFIG_USB_KAWETH is not set # CONFIG_USB_CATC is not set -# CONFIG_USB_NET1080 is not set +# CONFIG_USB_CDCETHER is not set # CONFIG_USB_USBNET is not set + +# +# USB port drivers +# # CONFIG_USB_USS720 is not set # # USB Serial Converter support # # CONFIG_USB_SERIAL is not set +# CONFIG_USB_SERIAL_GENERIC is not set +# CONFIG_USB_SERIAL_BELKIN is not set +# CONFIG_USB_SERIAL_WHITEHEAT is not set +# CONFIG_USB_SERIAL_DIGI_ACCELEPORT is not set +# CONFIG_USB_SERIAL_EMPEG is not set +# CONFIG_USB_SERIAL_FTDI_SIO is not set +# CONFIG_USB_SERIAL_VISOR is not set +# CONFIG_USB_SERIAL_IPAQ is not set +# CONFIG_USB_SERIAL_IR is not set +# CONFIG_USB_SERIAL_EDGEPORT is not set +# CONFIG_USB_SERIAL_KEYSPAN_PDA is not set +# CONFIG_USB_SERIAL_KEYSPAN is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XA is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XB is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA18X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19W is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA49W is not set +# CONFIG_USB_SERIAL_MCT_U232 is not set +# CONFIG_USB_SERIAL_KLSI is not set +# CONFIG_USB_SERIAL_PL2303 is not set +# CONFIG_USB_SERIAL_CYBERJACK is not set +# CONFIG_USB_SERIAL_XIRCOM is not set +# CONFIG_USB_SERIAL_OMNINET is not set + +# +# USB Miscellaneous drivers +# # CONFIG_USB_RIO500 is not set +# CONFIG_USB_AUERSWALD is not set +# CONFIG_USB_BRLVGER is not set # # Bluetooth support @@ -664,11 +1036,16 @@ # Kernel hacking # CONFIG_FRAME_POINTER=y -CONFIG_DEBUG_ERRORS=y # CONFIG_DEBUG_USER is not set # CONFIG_DEBUG_INFO is not set -# CONFIG_MAGIC_SYSRQ is not set # CONFIG_NO_PGT_CACHE is not set +# CONFIG_DEBUG_KERNEL is not set +# CONFIG_DEBUG_SLAB is not set +# CONFIG_MAGIC_SYSRQ is not set +# CONFIG_DEBUG_SPINLOCK is not set +# CONFIG_DEBUG_WAITQ is not set +# CONFIG_DEBUG_BUGVERBOSE is not set +# CONFIG_DEBUG_ERRORS is not set # CONFIG_DEBUG_LL is not set # CONFIG_DEBUG_DC21285_PORT is not set # CONFIG_DEBUG_CLPS711X_UART2 is not set diff -urN orig/arch/arm/def-configs/adsbitsyplus linux/arch/arm/def-configs/adsbitsyplus --- orig/arch/arm/def-configs/adsbitsyplus Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/def-configs/adsbitsyplus Fri Feb 21 15:17:33 2003 @@ -0,0 +1,1087 @@ +# +# Automatically generated make config: don't edit +# +CONFIG_ARM=y +# CONFIG_EISA is not set +# CONFIG_SBUS is not set +# CONFIG_MCA is not set +CONFIG_UID16=y +CONFIG_RWSEM_GENERIC_SPINLOCK=y +# CONFIG_RWSEM_XCHGADD_ALGORITHM is not set +# CONFIG_GENERIC_BUST_SPINLOCK is not set +# CONFIG_GENERIC_ISA_DMA is not set + +# +# Code maturity level options +# +CONFIG_EXPERIMENTAL=y +# CONFIG_OBSOLETE is not set + +# +# Loadable module support +# +CONFIG_MODULES=y +# CONFIG_MODVERSIONS is not set +# CONFIG_KMOD is not set + +# +# System Type +# +# CONFIG_ARCH_ANAKIN is not set +# CONFIG_ARCH_ARCA5K is not set +# CONFIG_ARCH_CLPS7500 is not set +# CONFIG_ARCH_CLPS711X is not set +# CONFIG_ARCH_CO285 is not set +# CONFIG_ARCH_EBSA110 is not set +# CONFIG_ARCH_CAMELOT is not set +# CONFIG_ARCH_FOOTBRIDGE is not set +# CONFIG_ARCH_INTEGRATOR is not set +# CONFIG_ARCH_OMAHA is not set +# CONFIG_ARCH_L7200 is not set +# CONFIG_ARCH_MX1ADS is not set +# CONFIG_ARCH_RPC is not set +# CONFIG_ARCH_RISCSTATION is not set +CONFIG_ARCH_SA1100=y +# CONFIG_ARCH_SHARK is not set +# CONFIG_ARCH_AT91RM9200DK is not set + +# +# Archimedes/A5000 Implementations +# + +# +# Archimedes/A5000 Implementations (select only ONE) +# +# CONFIG_ARCH_ARC is not set +# CONFIG_ARCH_A5K is not set + +# +# Footbridge Implementations +# +# CONFIG_ARCH_CATS is not set +# CONFIG_ARCH_PERSONAL_SERVER is not set +# CONFIG_ARCH_EBSA285_ADDIN is not set +# CONFIG_ARCH_EBSA285_HOST is not set +# CONFIG_ARCH_NETWINDER is not set + +# +# SA11x0 Implementations +# +# CONFIG_SA1100_ACCELENT is not set +# CONFIG_SA1100_ASSABET is not set +# CONFIG_ASSABET_NEPONSET is not set +# CONFIG_SA1100_ADSAGC is not set +# CONFIG_SA1100_ADSBITSY is not set +CONFIG_SA1100_ADSBITSYPLUS=y +# CONFIG_SA1100_BRUTUS is not set +# CONFIG_SA1100_CEP is not set +# CONFIG_SA1100_CERF is not set +# CONFIG_SA1100_H3100 is not set +# CONFIG_SA1100_H3600 is not set +# CONFIG_SA1100_H3800 is not set +# CONFIG_SA1100_H3XXX is not set +# CONFIG_SA1100_EXTENEX1 is not set +# CONFIG_SA1100_FLEXANET is not set +# CONFIG_SA1100_FREEBIRD is not set +# CONFIG_SA1100_FRODO is not set +# CONFIG_SA1100_GRAPHICSCLIENT is not set +# CONFIG_SA1100_GRAPHICSMASTER is not set +# CONFIG_SA1100_HACKKIT is not set +# CONFIG_SA1100_BADGE4 is not set +# CONFIG_SA1100_JORNADA720 is not set +# CONFIG_SA1100_HUW_WEBPANEL is not set +# CONFIG_SA1100_ITSY is not set +# CONFIG_SA1100_LART is not set +# CONFIG_SA1100_NANOENGINE is not set +# CONFIG_SA1100_OMNIMETER is not set +# CONFIG_SA1100_PANGOLIN is not set +# CONFIG_SA1100_PLEB is not set +# CONFIG_SA1100_PT_SYSTEM3 is not set +# CONFIG_SA1100_SHANNON is not set +# CONFIG_SA1100_SHERMAN is not set +# CONFIG_SA1100_SIMPAD is not set +# CONFIG_SA1100_SIMPUTER is not set +# CONFIG_SA1100_PFS168 is not set +# CONFIG_SA1100_VICTOR is not set +# CONFIG_SA1100_XP860 is not set +# CONFIG_SA1100_YOPY is not set +# CONFIG_SA1100_USB is not set +# CONFIG_SA1100_USB_NETLINK is not set +# CONFIG_SA1100_USB_CHAR is not set +# CONFIG_H3600_SLEEVE is not set + +# +# CLPS711X/EP721X Implementations +# +# CONFIG_ARCH_AUTCPU12 is not set +# CONFIG_ARCH_CDB89712 is not set +# CONFIG_ARCH_CLEP7312 is not set +# CONFIG_ARCH_EDB7211 is not set +# CONFIG_ARCH_P720T is not set +# CONFIG_ARCH_FORTUNET is not set +# CONFIG_ARCH_EP7211 is not set +# CONFIG_ARCH_EP7212 is not set +# CONFIG_ARCH_ACORN is not set +# CONFIG_FOOTBRIDGE is not set +# CONFIG_FOOTBRIDGE_HOST is not set +# CONFIG_FOOTBRIDGE_ADDIN is not set +CONFIG_SA1111=y +CONFIG_FORCE_MAX_ZONEORDER=9 + +# +# Processor Type +# +CONFIG_CPU_32=y +# CONFIG_CPU_26 is not set +# CONFIG_CPU_ARM610 is not set +# CONFIG_CPU_ARM710 is not set +# CONFIG_CPU_ARM720T is not set +# CONFIG_CPU_ARM920T is not set +# CONFIG_CPU_ARM922T is not set +# CONFIG_PLD is not set +# CONFIG_CPU_ARM926T is not set +# CONFIG_CPU_ARM1020 is not set +# CONFIG_CPU_ARM1026 is not set +# CONFIG_CPU_SA110 is not set +CONFIG_CPU_SA1100=y +# CONFIG_CPU_32v3 is not set +CONFIG_CPU_32v4=y + +# +# Processor Features +# +CONFIG_DISCONTIGMEM=y + +# +# General setup +# +# CONFIG_PCI is not set +# CONFIG_ISA is not set +# CONFIG_ISA_DMA is not set +# CONFIG_ZBOOT_ROM is not set +CONFIG_ZBOOT_ROM_TEXT=0 +CONFIG_ZBOOT_ROM_BSS=0 +# CONFIG_CPU_FREQ is not set +CONFIG_HOTPLUG=y + +# +# PCMCIA/CardBus support +# +CONFIG_PCMCIA=y +CONFIG_PCMCIA_PROBE=y +# CONFIG_I82092 is not set +# CONFIG_I82365 is not set +# CONFIG_TCIC is not set +# CONFIG_PCMCIA_CLPS6700 is not set +CONFIG_PCMCIA_SA1100=y +CONFIG_NET=y +CONFIG_SYSVIPC=y +# CONFIG_BSD_PROCESS_ACCT is not set +CONFIG_SYSCTL=y + +# +# At least one math emulation must be selected +# +CONFIG_FPE_NWFPE=y +# CONFIG_FPE_FASTFPE is not set +CONFIG_KCORE_ELF=y +# CONFIG_KCORE_AOUT is not set +# CONFIG_BINFMT_AOUT is not set +CONFIG_BINFMT_ELF=y +# CONFIG_BINFMT_MISC is not set +CONFIG_PM=y +# CONFIG_ARTHUR is not set +CONFIG_CMDLINE="console=ttySA0 mem=32m@0xC0000000 rw ramdisk_size=8192 initrd=0xC0800000,3m root=/dev/ram" +CONFIG_LEDS=y +CONFIG_LEDS_TIMER=y +# CONFIG_LEDS_CPU is not set +CONFIG_ALIGNMENT_TRAP=y + +# +# Parallel port support +# +# CONFIG_PARPORT is not set + +# +# Memory Technology Devices (MTD) +# +CONFIG_MTD=y +# CONFIG_MTD_DEBUG is not set +CONFIG_MTD_PARTITIONS=y +# CONFIG_MTD_CONCAT is not set +# CONFIG_MTD_REDBOOT_PARTS is not set +CONFIG_MTD_CMDLINE_PARTS=y +# CONFIG_MTD_AFS_PARTS is not set + +# +# User Modules And Translation Layers +# +CONFIG_MTD_CHAR=y +CONFIG_MTD_BLOCK=y +# CONFIG_FTL is not set +# CONFIG_NFTL is not set + +# +# RAM/ROM/Flash chip drivers +# +CONFIG_MTD_CFI=y +# CONFIG_MTD_JEDECPROBE is not set +CONFIG_MTD_GEN_PROBE=y +# CONFIG_MTD_CFI_ADV_OPTIONS is not set +CONFIG_MTD_CFI_INTELEXT=y +# CONFIG_MTD_CFI_AMDSTD is not set +# CONFIG_MTD_RAM is not set +# CONFIG_MTD_ROM is not set +# CONFIG_MTD_ABSENT is not set +# CONFIG_MTD_OBSOLETE_CHIPS is not set +# CONFIG_MTD_AMDSTD is not set +# CONFIG_MTD_SHARP is not set +# CONFIG_MTD_JEDEC is not set + +# +# Mapping drivers for chip access +# +CONFIG_MTD_PHYSMAP=y +CONFIG_MTD_PHYSMAP_START=8000000 +CONFIG_MTD_PHYSMAP_LEN=4000000 +CONFIG_MTD_PHYSMAP_BUSWIDTH=4 +# CONFIG_MTD_NORA is not set +# CONFIG_MTD_ARM_INTEGRATOR is not set +# CONFIG_MTD_CDB89712 is not set +CONFIG_MTD_SA1100=y +# CONFIG_MTD_DC21285 is not set +# CONFIG_MTD_IQ80310 is not set +# CONFIG_MTD_FORTUNET is not set +# CONFIG_MTD_EPXA10DB is not set +# CONFIG_MTD_AUTCPU12 is not set +# CONFIG_MTD_EDB7312 is not set +# CONFIG_MTD_IMPA7 is not set +# CONFIG_MTD_PCI is not set + +# +# Self-contained MTD device drivers +# +# CONFIG_MTD_PMC551 is not set +# CONFIG_MTD_SLRAM is not set +# CONFIG_MTD_MTDRAM is not set +# CONFIG_MTD_BLKMTD is not set + +# +# Disk-On-Chip Device Drivers +# +# CONFIG_MTD_DOC1000 is not set +# CONFIG_MTD_DOC2000 is not set +# CONFIG_MTD_DOC2001 is not set +# CONFIG_MTD_DOCPROBE is not set + +# +# NAND Flash Device Drivers +# +# CONFIG_MTD_NAND is not set + +# +# Plug and Play configuration +# +# CONFIG_PNP is not set +# CONFIG_ISAPNP is not set + +# +# Block devices +# +# CONFIG_BLK_DEV_FD is not set +# CONFIG_BLK_DEV_XD is not set +# CONFIG_PARIDE is not set +# CONFIG_BLK_CPQ_DA is not set +# CONFIG_BLK_CPQ_CISS_DA is not set +# CONFIG_CISS_SCSI_TAPE is not set +# CONFIG_BLK_DEV_DAC960 is not set +# CONFIG_BLK_DEV_UMEM is not set +CONFIG_BLK_DEV_LOOP=y +# CONFIG_BLK_DEV_NBD is not set +CONFIG_BLK_DEV_RAM=y +CONFIG_BLK_DEV_RAM_SIZE=8192 +CONFIG_BLK_DEV_INITRD=y + +# +# Multi-device support (RAID and LVM) +# +# CONFIG_MD is not set +# CONFIG_BLK_DEV_MD is not set +# CONFIG_MD_LINEAR is not set +# CONFIG_MD_RAID0 is not set +# CONFIG_MD_RAID1 is not set +# CONFIG_MD_RAID5 is not set +# CONFIG_MD_MULTIPATH is not set +# CONFIG_BLK_DEV_LVM is not set + +# +# Networking options +# +CONFIG_PACKET=y +# CONFIG_PACKET_MMAP is not set +# CONFIG_NETLINK_DEV is not set +# CONFIG_NETFILTER is not set +# CONFIG_FILTER is not set +CONFIG_UNIX=y +CONFIG_INET=y +# CONFIG_IP_MULTICAST is not set +# CONFIG_IP_ADVANCED_ROUTER is not set +CONFIG_IP_PNP=y +# CONFIG_IP_PNP_DHCP is not set +CONFIG_IP_PNP_BOOTP=y +# CONFIG_IP_PNP_RARP is not set +# CONFIG_NET_IPIP is not set +# CONFIG_NET_IPGRE is not set +# CONFIG_ARPD is not set +# CONFIG_INET_ECN is not set +# CONFIG_SYN_COOKIES is not set +# CONFIG_IPV6 is not set +# CONFIG_KHTTPD is not set +# CONFIG_ATM is not set +# CONFIG_VLAN_8021Q is not set + +# +# +# +# CONFIG_IPX is not set +# CONFIG_ATALK is not set + +# +# Appletalk devices +# +# CONFIG_DEV_APPLETALK is not set +# CONFIG_DECNET is not set +# CONFIG_BRIDGE is not set +# CONFIG_X25 is not set +# CONFIG_LAPB is not set +# CONFIG_LLC is not set +# CONFIG_NET_DIVERT is not set +# CONFIG_ECONET is not set +# CONFIG_WAN_ROUTER is not set +# CONFIG_NET_FASTROUTE is not set +# CONFIG_NET_HW_FLOWCONTROL is not set + +# +# QoS and/or fair queueing +# +# CONFIG_NET_SCHED is not set + +# +# Network testing +# +# CONFIG_NET_PKTGEN is not set + +# +# Network device support +# +CONFIG_NETDEVICES=y + +# +# ARCnet devices +# +# CONFIG_ARCNET is not set +# CONFIG_DUMMY is not set +# CONFIG_BONDING is not set +# CONFIG_EQUALIZER is not set +# CONFIG_TUN is not set +# CONFIG_ETHERTAP is not set + +# +# Ethernet (10 or 100Mbit) +# +CONFIG_NET_ETHERNET=y +# CONFIG_ARM_AM79C961A is not set +# CONFIG_ARM_CIRRUS is not set +# CONFIG_SUNLANCE is not set +# CONFIG_SUNBMAC is not set +# CONFIG_SUNQE is not set +# CONFIG_SUNGEM is not set +# CONFIG_NET_VENDOR_3COM is not set +# CONFIG_LANCE is not set +CONFIG_NET_VENDOR_SMC=y +# CONFIG_WD80x3 is not set +# CONFIG_ULTRAMCA is not set +# CONFIG_ULTRA is not set +# CONFIG_ULTRA32 is not set +# CONFIG_SMC9194 is not set +CONFIG_SMC91111=y +# CONFIG_NET_VENDOR_RACAL is not set +# CONFIG_NET_ISA is not set +# CONFIG_NET_PCI is not set +# CONFIG_NET_POCKET is not set + +# +# Ethernet (1000 Mbit) +# +# CONFIG_ACENIC is not set +# CONFIG_DL2K is not set +# CONFIG_MYRI_SBUS is not set +# CONFIG_NS83820 is not set +# CONFIG_HAMACHI is not set +# CONFIG_YELLOWFIN is not set +# CONFIG_SK98LIN is not set +# CONFIG_TIGON3 is not set +# CONFIG_FDDI is not set +# CONFIG_HIPPI is not set +# CONFIG_PLIP is not set +CONFIG_PPP=y +# CONFIG_PPP_MULTILINK is not set +# CONFIG_PPP_FILTER is not set +CONFIG_PPP_ASYNC=y +# CONFIG_PPP_SYNC_TTY is not set +# CONFIG_PPP_DEFLATE is not set +# CONFIG_PPP_BSDCOMP is not set +# CONFIG_PPPOE is not set +# CONFIG_SLIP is not set + +# +# Wireless LAN (non-hamradio) +# +CONFIG_NET_RADIO=y +# CONFIG_STRIP is not set +# CONFIG_WAVELAN is not set +# CONFIG_ARLAN is not set +# CONFIG_AIRONET4500 is not set +# CONFIG_AIRONET4500_NONCS is not set +# CONFIG_AIRONET4500_PROC is not set +# CONFIG_HERMES is not set + +# +# Wireless Pcmcia cards support +# +# CONFIG_PCMCIA_HERMES is not set +# CONFIG_AIRO_CS is not set +CONFIG_NET_WIRELESS=y + +# +# Token Ring devices +# +# CONFIG_TR is not set +# CONFIG_NET_FC is not set +# CONFIG_RCPCI is not set +# CONFIG_SHAPER is not set + +# +# Wan interfaces +# +# CONFIG_WAN is not set + +# +# PCMCIA network device support +# +CONFIG_NET_PCMCIA=y +# CONFIG_PCMCIA_3C589 is not set +# CONFIG_PCMCIA_3C574 is not set +# CONFIG_PCMCIA_FMVJ18X is not set +CONFIG_PCMCIA_PCNET=y +# CONFIG_PCMCIA_AXNET is not set +# CONFIG_PCMCIA_NMCLAN is not set +# CONFIG_PCMCIA_SMC91C92 is not set +# CONFIG_PCMCIA_XIRC2PS is not set +# CONFIG_ARCNET_COM20020_CS is not set +# CONFIG_PCMCIA_IBMTR is not set +# CONFIG_NET_PCMCIA_RADIO is not set + +# +# Amateur Radio support +# +# CONFIG_HAMRADIO is not set + +# +# IrDA (infrared) support +# +# CONFIG_IRDA is not set + +# +# ATA/ATAPI/MFM/RLL support +# +CONFIG_IDE=y + +# +# IDE, ATA and ATAPI Block devices +# +CONFIG_BLK_DEV_IDE=y + +# +# Please see Documentation/ide.txt for help/info on IDE drives +# +# CONFIG_BLK_DEV_HD_IDE is not set +# CONFIG_BLK_DEV_HD is not set +CONFIG_BLK_DEV_IDEDISK=y +# CONFIG_IDEDISK_MULTI_MODE is not set +# CONFIG_IDEDISK_STROKE is not set +# CONFIG_BLK_DEV_IDEDISK_VENDOR is not set +# CONFIG_BLK_DEV_IDEDISK_FUJITSU is not set +# CONFIG_BLK_DEV_IDEDISK_IBM is not set +# CONFIG_BLK_DEV_IDEDISK_MAXTOR is not set +# CONFIG_BLK_DEV_IDEDISK_QUANTUM is not set +# CONFIG_BLK_DEV_IDEDISK_SEAGATE is not set +# CONFIG_BLK_DEV_IDEDISK_WD is not set +# CONFIG_BLK_DEV_COMMERIAL is not set +# CONFIG_BLK_DEV_TIVO is not set +CONFIG_BLK_DEV_IDECS=y +# CONFIG_BLK_DEV_IDECD is not set +# CONFIG_BLK_DEV_IDETAPE is not set +# CONFIG_BLK_DEV_IDEFLOPPY is not set +# CONFIG_BLK_DEV_IDESCSI is not set +# CONFIG_IDE_TASK_IOCTL is not set + +# +# IDE chipset support/bugfixes +# +# CONFIG_BLK_DEV_CMD640 is not set +# CONFIG_BLK_DEV_CMD640_ENHANCED is not set +# CONFIG_BLK_DEV_ISAPNP is not set +# CONFIG_IDE_CHIPSETS is not set +# CONFIG_IDEDMA_AUTO is not set +# CONFIG_DMA_NONPCI is not set +# CONFIG_BLK_DEV_IDE_MODES is not set +# CONFIG_BLK_DEV_ATARAID is not set +# CONFIG_BLK_DEV_ATARAID_PDC is not set +# CONFIG_BLK_DEV_ATARAID_HPT is not set + +# +# SCSI support +# +# CONFIG_SCSI is not set + +# +# I2O device support +# +# CONFIG_I2O is not set +# CONFIG_I2O_BLOCK is not set +# CONFIG_I2O_LAN is not set +# CONFIG_I2O_SCSI is not set +# CONFIG_I2O_PROC is not set + +# +# ISDN subsystem +# +# CONFIG_ISDN is not set + +# +# Input core support +# +CONFIG_INPUT=y +# CONFIG_INPUT_KEYBDEV is not set +CONFIG_INPUT_MOUSEDEV=y +CONFIG_INPUT_MOUSEDEV_SCREEN_X=640 +CONFIG_INPUT_MOUSEDEV_SCREEN_Y=480 +# CONFIG_INPUT_JOYDEV is not set +# CONFIG_INPUT_EVDEV is not set + +# +# Character devices +# +CONFIG_VT=y +CONFIG_VT_CONSOLE=y +# CONFIG_SERIAL is not set +# CONFIG_SERIAL_EXTENDED is not set +# CONFIG_SERIAL_NONSTANDARD is not set + +# +# Serial drivers +# +# CONFIG_SERIAL_ANAKIN is not set +# CONFIG_SERIAL_ANAKIN_CONSOLE is not set +# CONFIG_SERIAL_AMBA is not set +# CONFIG_SERIAL_AMBA_CONSOLE is not set +# CONFIG_SERIAL_CLPS711X is not set +# CONFIG_SERIAL_CLPS711X_CONSOLE is not set +# CONFIG_SERIAL_21285 is not set +# CONFIG_SERIAL_21285_OLD is not set +# CONFIG_SERIAL_21285_CONSOLE is not set +# CONFIG_SERIAL_UART00 is not set +# CONFIG_SERIAL_UART00_CONSOLE is not set +CONFIG_SERIAL_SA1100=y +CONFIG_SERIAL_SA1100_CONSOLE=y +CONFIG_SA1100_DEFAULT_BAUDRATE=38400 +# CONFIG_SERIAL_OMAHA is not set +# CONFIG_SERIAL_OMAHA_CONSOLE is not set +# CONFIG_SERIAL_8250 is not set +# CONFIG_SERIAL_8250_CONSOLE is not set +# CONFIG_SERIAL_8250_EXTENDED is not set +# CONFIG_SERIAL_8250_MANY_PORTS is not set +# CONFIG_SERIAL_8250_SHARE_IRQ is not set +# CONFIG_SERIAL_8250_DETECT_IRQ is not set +# CONFIG_SERIAL_8250_MULTIPORT is not set +# CONFIG_SERIAL_8250_HUB6 is not set +CONFIG_SERIAL_CORE=y +CONFIG_SERIAL_CORE_CONSOLE=y +CONFIG_TOUCHSCREEN_ADS784X=y +# CONFIG_UCB1200 is not set +# CONFIG_TOUCHSCREEN_UCB1200 is not set +# CONFIG_AUDIO_UCB1200 is not set +# CONFIG_ADC_UCB1200 is not set +CONFIG_UNIX98_PTYS=y +CONFIG_UNIX98_PTY_COUNT=32 + +# +# I2C support +# +CONFIG_I2C=y +# CONFIG_I2C_ALGOBIT is not set +# CONFIG_I2C_ALGOPCF is not set +CONFIG_ADSSIO_I2C_ALGO=y +# CONFIG_I2C_CHARDEV is not set +# CONFIG_I2C_PROC is not set +CONFIG_I2C_DS1307=y +CONFIG_I2C_DS1307_ADS=y + +# +# L3 serial bus support +# +CONFIG_L3=y +# CONFIG_L3_ALGOBIT is not set +# CONFIG_L3_BIT_SA1100_GPIO is not set + +# +# Other L3 adapters +# +CONFIG_L3_SA1111=y +# CONFIG_BIT_SA1100_GPIO is not set + +# +# Mice +# +# CONFIG_BUSMOUSE is not set +# CONFIG_MOUSE is not set + +# +# Joysticks +# +# CONFIG_INPUT_GAMEPORT is not set +# CONFIG_INPUT_NS558 is not set +# CONFIG_INPUT_LIGHTNING is not set +# CONFIG_INPUT_PCIGAME is not set +# CONFIG_INPUT_CS461X is not set +# CONFIG_INPUT_EMU10K1 is not set +# CONFIG_INPUT_SERIO is not set +# CONFIG_INPUT_SERPORT is not set + +# +# Joysticks +# +# CONFIG_INPUT_ANALOG is not set +# CONFIG_INPUT_A3D is not set +# CONFIG_INPUT_ADI is not set +# CONFIG_INPUT_COBRA is not set +# CONFIG_INPUT_GF2K is not set +# CONFIG_INPUT_GRIP is not set +# CONFIG_INPUT_INTERACT is not set +# CONFIG_INPUT_TMDC is not set +# CONFIG_INPUT_SIDEWINDER is not set +# CONFIG_INPUT_IFORCE_USB is not set +# CONFIG_INPUT_IFORCE_232 is not set +# CONFIG_INPUT_WARRIOR is not set +# CONFIG_INPUT_MAGELLAN is not set +# CONFIG_INPUT_SPACEORB is not set +# CONFIG_INPUT_SPACEBALL is not set +# CONFIG_INPUT_STINGER is not set +# CONFIG_INPUT_DB9 is not set +# CONFIG_INPUT_GAMECON is not set +# CONFIG_INPUT_TURBOGRAFX is not set +# CONFIG_QIC02_TAPE is not set + +# +# Watchdog Cards +# +# CONFIG_WATCHDOG is not set +# CONFIG_NVRAM is not set +# CONFIG_RTC is not set +# CONFIG_SA1100_RTC is not set +# CONFIG_DTLK is not set +# CONFIG_R3964 is not set +# CONFIG_APPLICOM is not set + +# +# Ftape, the floppy tape device driver +# +# CONFIG_FTAPE is not set +# CONFIG_AGP is not set +# CONFIG_DRM is not set + +# +# PCMCIA character devices +# + +# +# Multimedia devices +# +# CONFIG_VIDEO_DEV is not set + +# +# File systems +# +# CONFIG_QUOTA is not set +# CONFIG_AUTOFS_FS is not set +# CONFIG_AUTOFS4_FS is not set +# CONFIG_REISERFS_FS is not set +# CONFIG_REISERFS_CHECK is not set +# CONFIG_REISERFS_PROC_INFO is not set +# CONFIG_ADFS_FS is not set +# CONFIG_ADFS_FS_RW is not set +# CONFIG_AFFS_FS is not set +# CONFIG_HFS_FS is not set +# CONFIG_BFS_FS is not set +# CONFIG_EXT3_FS is not set +# CONFIG_JBD is not set +# CONFIG_JBD_DEBUG is not set +CONFIG_FAT_FS=y +# CONFIG_MSDOS_FS is not set +# CONFIG_UMSDOS_FS is not set +CONFIG_VFAT_FS=y +# CONFIG_EFS_FS is not set +# CONFIG_JFFS_FS is not set +CONFIG_JFFS2_FS=y +CONFIG_JFFS2_FS_DEBUG=0 +CONFIG_CRAMFS=y +# CONFIG_TMPFS is not set +CONFIG_RAMFS=y +# CONFIG_ISO9660_FS is not set +# CONFIG_JOLIET is not set +# CONFIG_ZISOFS is not set +# CONFIG_MINIX_FS is not set +# CONFIG_VXFS_FS is not set +# CONFIG_NTFS_FS is not set +# CONFIG_NTFS_RW is not set +# CONFIG_HPFS_FS is not set +CONFIG_PROC_FS=y +# CONFIG_DEVFS_FS is not set +# CONFIG_DEVFS_MOUNT is not set +# CONFIG_DEVFS_DEBUG is not set +CONFIG_DEVPTS_FS=y +# CONFIG_QNX4FS_FS is not set +# CONFIG_QNX4FS_RW is not set +# CONFIG_ROMFS_FS is not set +CONFIG_EXT2_FS=y +# CONFIG_SYSV_FS is not set +# CONFIG_UDF_FS is not set +# CONFIG_UDF_RW is not set +# CONFIG_UFS_FS is not set +# CONFIG_UFS_FS_WRITE is not set + +# +# Network File Systems +# +# CONFIG_CODA_FS is not set +# CONFIG_INTERMEZZO_FS is not set +CONFIG_NFS_FS=y +# CONFIG_NFS_V3 is not set +CONFIG_ROOT_NFS=y +# CONFIG_NFSD is not set +# CONFIG_NFSD_V3 is not set +CONFIG_SUNRPC=y +CONFIG_LOCKD=y +# CONFIG_SMB_FS is not set +# CONFIG_NCP_FS is not set +# CONFIG_NCPFS_PACKET_SIGNING is not set +# CONFIG_NCPFS_IOCTL_LOCKING is not set +# CONFIG_NCPFS_STRONG is not set +# CONFIG_NCPFS_NFS_NS is not set +# CONFIG_NCPFS_OS2_NS is not set +# CONFIG_NCPFS_SMALLDOS is not set +# CONFIG_NCPFS_NLS is not set +# CONFIG_NCPFS_EXTRAS is not set +# CONFIG_ZISOFS_FS is not set +CONFIG_ZLIB_FS_INFLATE=y + +# +# Partition Types +# +# CONFIG_PARTITION_ADVANCED is not set +CONFIG_MSDOS_PARTITION=y +# CONFIG_SMB_NLS is not set +CONFIG_NLS=y + +# +# Native Language Support +# +CONFIG_NLS_DEFAULT="iso8859-1" +# CONFIG_NLS_CODEPAGE_437 is not set +# CONFIG_NLS_CODEPAGE_737 is not set +# CONFIG_NLS_CODEPAGE_775 is not set +# CONFIG_NLS_CODEPAGE_850 is not set +# CONFIG_NLS_CODEPAGE_852 is not set +# CONFIG_NLS_CODEPAGE_855 is not set +# CONFIG_NLS_CODEPAGE_857 is not set +# CONFIG_NLS_CODEPAGE_860 is not set +# CONFIG_NLS_CODEPAGE_861 is not set +# CONFIG_NLS_CODEPAGE_862 is not set +# CONFIG_NLS_CODEPAGE_863 is not set +# CONFIG_NLS_CODEPAGE_864 is not set +# CONFIG_NLS_CODEPAGE_865 is not set +# CONFIG_NLS_CODEPAGE_866 is not set +# CONFIG_NLS_CODEPAGE_869 is not set +# CONFIG_NLS_CODEPAGE_936 is not set +# CONFIG_NLS_CODEPAGE_950 is not set +# CONFIG_NLS_CODEPAGE_932 is not set +# CONFIG_NLS_CODEPAGE_949 is not set +# CONFIG_NLS_CODEPAGE_874 is not set +# CONFIG_NLS_ISO8859_8 is not set +# CONFIG_NLS_CODEPAGE_1250 is not set +# CONFIG_NLS_CODEPAGE_1251 is not set +# CONFIG_NLS_ISO8859_1 is not set +# CONFIG_NLS_ISO8859_2 is not set +# CONFIG_NLS_ISO8859_3 is not set +# CONFIG_NLS_ISO8859_4 is not set +# CONFIG_NLS_ISO8859_5 is not set +# CONFIG_NLS_ISO8859_6 is not set +# CONFIG_NLS_ISO8859_7 is not set +# CONFIG_NLS_ISO8859_9 is not set +# CONFIG_NLS_ISO8859_13 is not set +# CONFIG_NLS_ISO8859_14 is not set +# CONFIG_NLS_ISO8859_15 is not set +# CONFIG_NLS_KOI8_R is not set +# CONFIG_NLS_KOI8_U is not set +# CONFIG_NLS_UTF8 is not set + +# +# Console drivers +# +CONFIG_PC_KEYMAP=y +# CONFIG_VGA_CONSOLE is not set + +# +# Frame-buffer support +# +CONFIG_FB=y +CONFIG_DUMMY_CONSOLE=y +# CONFIG_FB_ACORN is not set +# CONFIG_FB_ANAKIN is not set +# CONFIG_FB_CLPS711X is not set +CONFIG_FB_SA1100=y +# CONFIG_FB_CYBER2000 is not set +# CONFIG_FB_E13806 is not set +# CONFIG_FB_VIRTUAL is not set +# CONFIG_FBCON_ADVANCED is not set +CONFIG_FBCON_CFB2=y +CONFIG_FBCON_CFB4=y +CONFIG_FBCON_CFB8=y +CONFIG_FBCON_CFB16=y +CONFIG_FBCON_FONTWIDTH8_ONLY=y +CONFIG_FBCON_FONTS=y +# CONFIG_FONT_8x8 is not set +CONFIG_FONT_8x16=y +# CONFIG_FONT_SUN8x16 is not set +# CONFIG_FONT_PEARL_8x8 is not set +# CONFIG_FONT_ACORN_8x8 is not set + +# +# Sound +# +CONFIG_SOUND=y +# CONFIG_SOUND_BT878 is not set +# CONFIG_SOUND_CMPCI is not set +# CONFIG_SOUND_EMU10K1 is not set +# CONFIG_MIDI_EMU10K1 is not set +# CONFIG_SOUND_FUSION is not set +# CONFIG_SOUND_CS4281 is not set +# CONFIG_SOUND_ES1370 is not set +# CONFIG_SOUND_ES1371 is not set +# CONFIG_SOUND_ESSSOLO1 is not set +# CONFIG_SOUND_MAESTRO is not set +# CONFIG_SOUND_MAESTRO3 is not set +# CONFIG_SOUND_ICH is not set +# CONFIG_SOUND_RME96XX is not set +# CONFIG_SOUND_SONICVIBES is not set +# CONFIG_SOUND_TRIDENT is not set +# CONFIG_SOUND_MSNDCLAS is not set +# CONFIG_SOUND_MSNDPIN is not set +# CONFIG_SOUND_VIA82CXXX is not set +# CONFIG_MIDI_VIA82CXXX is not set +CONFIG_SOUND_SA1100=y +# CONFIG_SOUND_UDA1341 is not set +# CONFIG_SOUND_ASSABET_UDA1341 is not set +# CONFIG_SOUND_H3600_UDA1341 is not set +# CONFIG_SOUND_PANGOLIN_UDA1341 is not set +# CONFIG_SOUND_SA1111_UDA1341 is not set +CONFIG_SOUND_SA1111_AC97=y +# CONFIG_SOUND_SA1100SSP is not set +CONFIG_SOUND_OSS=y +# CONFIG_SOUND_TRACEINIT is not set +# CONFIG_SOUND_DMAP is not set +# CONFIG_SOUND_AD1816 is not set +# CONFIG_SOUND_SGALAXY is not set +# CONFIG_SOUND_ADLIB is not set +# CONFIG_SOUND_ACI_MIXER is not set +# CONFIG_SOUND_CS4232 is not set +# CONFIG_SOUND_SSCAPE is not set +# CONFIG_SOUND_GUS is not set +# CONFIG_SOUND_VMIDI is not set +# CONFIG_SOUND_TRIX is not set +# CONFIG_SOUND_MSS is not set +# CONFIG_SOUND_MPU401 is not set +# CONFIG_SOUND_NM256 is not set +# CONFIG_SOUND_MAD16 is not set +# CONFIG_SOUND_PAS is not set +# CONFIG_PAS_JOYSTICK is not set +# CONFIG_SOUND_PSS is not set +# CONFIG_SOUND_SB is not set +# CONFIG_SOUND_AWE32_SYNTH is not set +# CONFIG_SOUND_WAVEFRONT is not set +# CONFIG_SOUND_MAUI is not set +# CONFIG_SOUND_YM3812 is not set +# CONFIG_SOUND_OPL3SA1 is not set +# CONFIG_SOUND_OPL3SA2 is not set +# CONFIG_SOUND_YMFPCI is not set +# CONFIG_SOUND_YMFPCI_LEGACY is not set +# CONFIG_SOUND_UART6850 is not set +# CONFIG_SOUND_AEDSP16 is not set +# CONFIG_SOUND_VIDC is not set +# CONFIG_SOUND_WAVEARTIST is not set +# CONFIG_SOUND_TVMIXER is not set + +# +# Multimedia Capabilities Port drivers +# +# CONFIG_MCP is not set +# CONFIG_MCP_SA1100 is not set +# CONFIG_MCP_UCB1200 is not set +# CONFIG_MCP_UCB1200_AUDIO is not set +# CONFIG_MCP_UCB1200_TS is not set + +# +# USB support +# +CONFIG_USB=y +# CONFIG_USB_DEBUG is not set + +# +# Miscellaneous USB options +# +CONFIG_USB_DEVICEFS=y +# CONFIG_USB_BANDWIDTH is not set +# CONFIG_USB_LONG_TIMEOUT is not set + +# +# USB Host Controller Drivers +# +# CONFIG_USB_EHCI_HCD is not set +# CONFIG_USB_UHCI is not set +# CONFIG_USB_UHCI_ALT is not set +# CONFIG_USB_OHCI is not set +CONFIG_USB_OHCI_SA1111=y + +# +# USB Device Class drivers +# +# CONFIG_USB_AUDIO is not set +# CONFIG_USB_EMI26 is not set +# CONFIG_USB_BLUETOOTH is not set + +# +# SCSI support is needed for USB Storage +# +# CONFIG_USB_STORAGE is not set +# CONFIG_USB_STORAGE_DEBUG is not set +# CONFIG_USB_STORAGE_DATAFAB is not set +# CONFIG_USB_STORAGE_FREECOM is not set +# CONFIG_USB_STORAGE_ISD200 is not set +# CONFIG_USB_STORAGE_DPCM is not set +# CONFIG_USB_STORAGE_HP8200e is not set +# CONFIG_USB_STORAGE_SDDR09 is not set +# CONFIG_USB_STORAGE_JUMPSHOT is not set +# CONFIG_USB_ACM is not set +# CONFIG_USB_PRINTER is not set + +# +# USB Human Interface Devices (HID) +# +CONFIG_USB_HID=y +CONFIG_USB_HIDINPUT=y +# CONFIG_USB_HIDDEV is not set +# CONFIG_USB_WACOM is not set + +# +# USB Imaging devices +# +# CONFIG_USB_DC2XX is not set +# CONFIG_USB_MDC800 is not set +# CONFIG_USB_SCANNER is not set +# CONFIG_USB_MICROTEK is not set +# CONFIG_USB_HPUSBSCSI is not set + +# +# USB Multimedia devices +# + +# +# Video4Linux support is needed for USB Multimedia device support +# + +# +# USB Network adaptors +# +# CONFIG_USB_PEGASUS is not set +# CONFIG_USB_RTL8150 is not set +# CONFIG_USB_KAWETH is not set +# CONFIG_USB_CATC is not set +# CONFIG_USB_CDCETHER is not set +# CONFIG_USB_USBNET is not set + +# +# USB port drivers +# +# CONFIG_USB_USS720 is not set + +# +# USB Serial Converter support +# +# CONFIG_USB_SERIAL is not set +# CONFIG_USB_SERIAL_GENERIC is not set +# CONFIG_USB_SERIAL_BELKIN is not set +# CONFIG_USB_SERIAL_WHITEHEAT is not set +# CONFIG_USB_SERIAL_DIGI_ACCELEPORT is not set +# CONFIG_USB_SERIAL_EMPEG is not set +# CONFIG_USB_SERIAL_FTDI_SIO is not set +# CONFIG_USB_SERIAL_VISOR is not set +# CONFIG_USB_SERIAL_IPAQ is not set +# CONFIG_USB_SERIAL_IR is not set +# CONFIG_USB_SERIAL_EDGEPORT is not set +# CONFIG_USB_SERIAL_KEYSPAN_PDA is not set +# CONFIG_USB_SERIAL_KEYSPAN is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XA is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XB is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA18X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19W is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA49W is not set +# CONFIG_USB_SERIAL_MCT_U232 is not set +# CONFIG_USB_SERIAL_KLSI is not set +# CONFIG_USB_SERIAL_PL2303 is not set +# CONFIG_USB_SERIAL_CYBERJACK is not set +# CONFIG_USB_SERIAL_XIRCOM is not set +# CONFIG_USB_SERIAL_OMNINET is not set + +# +# USB Miscellaneous drivers +# +# CONFIG_USB_RIO500 is not set +# CONFIG_USB_AUERSWALD is not set +# CONFIG_USB_BRLVGER is not set + +# +# Bluetooth support +# +# CONFIG_BLUEZ is not set + +# +# Kernel hacking +# +CONFIG_FRAME_POINTER=y +# CONFIG_DEBUG_USER is not set +# CONFIG_DEBUG_INFO is not set +# CONFIG_NO_PGT_CACHE is not set +# CONFIG_DEBUG_KERNEL is not set +# CONFIG_DEBUG_SLAB is not set +# CONFIG_MAGIC_SYSRQ is not set +# CONFIG_DEBUG_SPINLOCK is not set +# CONFIG_DEBUG_WAITQ is not set +# CONFIG_DEBUG_BUGVERBOSE is not set +# CONFIG_DEBUG_ERRORS is not set +# CONFIG_DEBUG_LL is not set +# CONFIG_DEBUG_DC21285_PORT is not set +# CONFIG_DEBUG_CLPS711X_UART2 is not set diff -urN orig/arch/arm/def-configs/at91rm9200dk linux/arch/arm/def-configs/at91rm9200dk --- orig/arch/arm/def-configs/at91rm9200dk Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/def-configs/at91rm9200dk Thu Feb 27 23:20:11 2003 @@ -0,0 +1,482 @@ +# +# Automatically generated make config: don't edit +# +CONFIG_ARM=y +# CONFIG_EISA is not set +# CONFIG_SBUS is not set +# CONFIG_MCA is not set +CONFIG_UID16=y +CONFIG_RWSEM_GENERIC_SPINLOCK=y +# CONFIG_RWSEM_XCHGADD_ALGORITHM is not set +# CONFIG_GENERIC_BUST_SPINLOCK is not set +# CONFIG_GENERIC_ISA_DMA is not set + +# +# Code maturity level options +# +CONFIG_EXPERIMENTAL=y +# CONFIG_OBSOLETE is not set + +# +# Loadable module support +# +CONFIG_MODULES=y +CONFIG_MODVERSIONS=y +CONFIG_KMOD=y + +# +# System Type +# +# CONFIG_ARCH_ANAKIN is not set +# CONFIG_ARCH_ARCA5K is not set +# CONFIG_ARCH_CLPS7500 is not set +# CONFIG_ARCH_CLPS711X is not set +# CONFIG_ARCH_CO285 is not set +# CONFIG_ARCH_EBSA110 is not set +# CONFIG_ARCH_CAMELOT is not set +# CONFIG_ARCH_FOOTBRIDGE is not set +# CONFIG_ARCH_INTEGRATOR is not set +# CONFIG_ARCH_OMAHA is not set +# CONFIG_ARCH_L7200 is not set +# CONFIG_ARCH_MX1ADS is not set +# CONFIG_ARCH_RPC is not set +# CONFIG_ARCH_RISCSTATION is not set +# CONFIG_ARCH_SA1100 is not set +# CONFIG_ARCH_SHARK is not set +CONFIG_ARCH_AT91RM9200DK=y + +# +# Archimedes/A5000 Implementations +# + +# +# Archimedes/A5000 Implementations (select only ONE) +# + +# +# Footbridge Implementations +# + +# +# SA11x0 Implementations +# +# CONFIG_SA1100_H3XXX is not set + +# +# CLPS711X/EP721X Implementations +# +# CONFIG_ARCH_EP7211 is not set +# CONFIG_ARCH_EP7212 is not set +# CONFIG_ARCH_ACORN is not set +# CONFIG_FOOTBRIDGE is not set +# CONFIG_FOOTBRIDGE_HOST is not set +# CONFIG_FOOTBRIDGE_ADDIN is not set + +# +# Processor Type +# +CONFIG_CPU_32=y +# CONFIG_CPU_26 is not set +# CONFIG_CPU_ARM610 is not set +# CONFIG_CPU_ARM710 is not set +# CONFIG_CPU_ARM720T is not set +CONFIG_CPU_ARM920T=y +# CONFIG_CPU_ARM922T is not set +# CONFIG_PLD is not set +# CONFIG_CPU_ARM926T is not set +# CONFIG_CPU_ARM1020 is not set +# CONFIG_CPU_ARM1026 is not set +# CONFIG_CPU_SA110 is not set +# CONFIG_CPU_SA1100 is not set +# CONFIG_CPU_32v3 is not set +CONFIG_CPU_32v4=y + +# +# Processor Features +# +# CONFIG_ARM_THUMB is not set +# CONFIG_CPU_ICACHE_DISABLE is not set +# CONFIG_CPU_DCACHE_DISABLE is not set +# CONFIG_DISCONTIGMEM is not set + +# +# General setup +# +# CONFIG_PCI is not set +# CONFIG_ISA is not set +# CONFIG_ISA_DMA is not set +# CONFIG_ZBOOT_ROM is not set +CONFIG_ZBOOT_ROM_TEXT=0 +CONFIG_ZBOOT_ROM_BSS=0 +# CONFIG_HOTPLUG is not set +# CONFIG_PCMCIA is not set +CONFIG_NET=y +CONFIG_SYSVIPC=y +# CONFIG_BSD_PROCESS_ACCT is not set +CONFIG_SYSCTL=y + +# +# At least one math emulation must be selected +# +CONFIG_FPE_NWFPE=y +# CONFIG_FPE_FASTFPE is not set +CONFIG_KCORE_ELF=y +# CONFIG_KCORE_AOUT is not set +CONFIG_BINFMT_AOUT=y +CONFIG_BINFMT_ELF=y +# CONFIG_BINFMT_MISC is not set +# CONFIG_PM is not set +# CONFIG_ARTHUR is not set +CONFIG_CMDLINE="console=ttyS4,115200 mem=32M root=/dev/ram rw" +CONFIG_ALIGNMENT_TRAP=y + +# +# Parallel port support +# +# CONFIG_PARPORT is not set + +# +# Memory Technology Devices (MTD) +# +CONFIG_MTD=y +# CONFIG_MTD_DEBUG is not set +# CONFIG_MTD_PARTITIONS is not set +# CONFIG_MTD_CONCAT is not set + +# +# User Modules And Translation Layers +# +CONFIG_MTD_CHAR=y +CONFIG_MTD_BLOCK=y +# CONFIG_FTL is not set +# CONFIG_NFTL is not set + +# +# RAM/ROM/Flash chip drivers +# +CONFIG_MTD_CFI=y +# CONFIG_MTD_JEDECPROBE is not set +CONFIG_MTD_GEN_PROBE=y +# CONFIG_MTD_CFI_ADV_OPTIONS is not set +CONFIG_MTD_CFI_INTELEXT=y +# CONFIG_MTD_CFI_AMDSTD is not set +# CONFIG_MTD_RAM is not set +# CONFIG_MTD_ROM is not set +# CONFIG_MTD_ABSENT is not set +# CONFIG_MTD_OBSOLETE_CHIPS is not set + +# +# Mapping drivers for chip access +# +# CONFIG_MTD_PHYSMAP is not set +# CONFIG_MTD_NORA is not set +# CONFIG_MTD_ARM_INTEGRATOR is not set +# CONFIG_MTD_IQ80310 is not set +# CONFIG_MTD_EDB7312 is not set + +# +# Self-contained MTD device drivers +# +# CONFIG_MTD_SLRAM is not set +# CONFIG_MTD_MTDRAM is not set +# CONFIG_MTD_BLKMTD is not set + +# +# Disk-On-Chip Device Drivers +# +# CONFIG_MTD_DOC1000 is not set +# CONFIG_MTD_DOC2000 is not set +# CONFIG_MTD_DOC2001 is not set +# CONFIG_MTD_DOCPROBE is not set + +# +# NAND Flash Device Drivers +# +# CONFIG_MTD_NAND is not set + +# +# Plug and Play configuration +# +# CONFIG_PNP is not set + +# +# Block devices +# +# CONFIG_BLK_DEV_FD is not set +# CONFIG_BLK_DEV_LOOP is not set +# CONFIG_BLK_DEV_NBD is not set +CONFIG_BLK_DEV_RAM=y +CONFIG_BLK_DEV_RAM_SIZE=8192 +CONFIG_BLK_DEV_INITRD=y + +# +# Multi-device support (RAID and LVM) +# +# CONFIG_MD is not set + +# +# Networking options +# +# CONFIG_PACKET is not set +# CONFIG_NETLINK_DEV is not set +# CONFIG_NETFILTER is not set +# CONFIG_FILTER is not set +CONFIG_UNIX=y +CONFIG_INET=y +# CONFIG_IP_MULTICAST is not set +# CONFIG_IP_ADVANCED_ROUTER is not set +CONFIG_IP_PNP=y +# CONFIG_IP_PNP_DHCP is not set +CONFIG_IP_PNP_BOOTP=y +# CONFIG_IP_PNP_RARP is not set +# CONFIG_NET_IPIP is not set +# CONFIG_NET_IPGRE is not set +# CONFIG_ARPD is not set +# CONFIG_INET_ECN is not set +# CONFIG_SYN_COOKIES is not set +# CONFIG_IPV6 is not set +# CONFIG_KHTTPD is not set +# CONFIG_ATM is not set +# CONFIG_VLAN_8021Q is not set + +# +# +# +# CONFIG_IPX is not set +# CONFIG_ATALK is not set + +# +# Appletalk devices +# +# CONFIG_DECNET is not set +# CONFIG_BRIDGE is not set +# CONFIG_X25 is not set +# CONFIG_LAPB is not set +# CONFIG_LLC is not set +# CONFIG_NET_DIVERT is not set +# CONFIG_ECONET is not set +# CONFIG_WAN_ROUTER is not set +# CONFIG_NET_FASTROUTE is not set +# CONFIG_NET_HW_FLOWCONTROL is not set + +# +# QoS and/or fair queueing +# +# CONFIG_NET_SCHED is not set + +# +# Network testing +# +# CONFIG_NET_PKTGEN is not set + +# +# Network device support +# +# CONFIG_NETDEVICES is not set + +# +# Amateur Radio support +# +# CONFIG_HAMRADIO is not set + +# +# IrDA (infrared) support +# +# CONFIG_IRDA is not set + +# +# ATA/ATAPI/MFM/RLL support +# +# CONFIG_IDE is not set +# CONFIG_BLK_DEV_IDE_MODES is not set +# CONFIG_BLK_DEV_HD is not set + +# +# SCSI support +# +# CONFIG_SCSI is not set + +# +# I2O device support +# +# CONFIG_I2O is not set + +# +# ISDN subsystem +# +# CONFIG_ISDN is not set + +# +# Input core support +# +# CONFIG_INPUT is not set +CONFIG_INPUT_MOUSEDEV_SCREEN_X=1024 +CONFIG_INPUT_MOUSEDEV_SCREEN_Y=768 + +# +# Character devices +# +# CONFIG_VT is not set +# CONFIG_SERIAL is not set +# CONFIG_SERIAL_NONSTANDARD is not set + +# +# Serial drivers +# +# CONFIG_SERIAL_8250 is not set +CONFIG_UNIX98_PTYS=y +CONFIG_UNIX98_PTY_COUNT=256 + +# +# I2C support +# +# CONFIG_I2C is not set + +# +# L3 serial bus support +# +# CONFIG_L3 is not set + +# +# Other L3 adapters +# +# CONFIG_BIT_SA1100_GPIO is not set + +# +# Mice +# +# CONFIG_BUSMOUSE is not set +CONFIG_MOUSE=y +CONFIG_PSMOUSE=y +# CONFIG_82C710_MOUSE is not set +# CONFIG_PC110_PAD is not set +# CONFIG_MK712_MOUSE is not set + +# +# Joysticks +# +# CONFIG_INPUT_GAMEPORT is not set + +# +# Input core support is needed for gameports +# + +# +# Input core support is needed for joysticks +# +# CONFIG_QIC02_TAPE is not set + +# +# Watchdog Cards +# +# CONFIG_WATCHDOG is not set +# CONFIG_NVRAM is not set +# CONFIG_RTC is not set +# CONFIG_DTLK is not set +# CONFIG_R3964 is not set +# CONFIG_APPLICOM is not set + +# +# Ftape, the floppy tape device driver +# +# CONFIG_FTAPE is not set +# CONFIG_AGP is not set +# CONFIG_DRM is not set + +# +# Multimedia devices +# +# CONFIG_VIDEO_DEV is not set + +# +# File systems +# +# CONFIG_QUOTA is not set +# CONFIG_AUTOFS_FS is not set +# CONFIG_AUTOFS4_FS is not set +# CONFIG_REISERFS_FS is not set +# CONFIG_ADFS_FS is not set +# CONFIG_AFFS_FS is not set +# CONFIG_HFS_FS is not set +# CONFIG_BFS_FS is not set +# CONFIG_EXT3_FS is not set +# CONFIG_JBD is not set +# CONFIG_FAT_FS is not set +# CONFIG_EFS_FS is not set +# CONFIG_JFFS_FS is not set +# CONFIG_JFFS2_FS is not set +# CONFIG_CRAMFS is not set +# CONFIG_TMPFS is not set +CONFIG_RAMFS=y +# CONFIG_ISO9660_FS is not set +# CONFIG_MINIX_FS is not set +# CONFIG_VXFS_FS is not set +# CONFIG_NTFS_FS is not set +# CONFIG_HPFS_FS is not set +CONFIG_PROC_FS=y +CONFIG_DEVFS_FS=y +CONFIG_DEVFS_MOUNT=y +CONFIG_DEVFS_DEBUG=y +CONFIG_DEVPTS_FS=y +# CONFIG_QNX4FS_FS is not set +CONFIG_ROMFS_FS=y +CONFIG_EXT2_FS=y +# CONFIG_SYSV_FS is not set +# CONFIG_UDF_FS is not set +# CONFIG_UFS_FS is not set + +# +# Network File Systems +# +# CONFIG_CODA_FS is not set +# CONFIG_INTERMEZZO_FS is not set +CONFIG_NFS_FS=y +# CONFIG_NFS_V3 is not set +CONFIG_ROOT_NFS=y +# CONFIG_NFSD is not set +CONFIG_SUNRPC=y +CONFIG_LOCKD=y +# CONFIG_SMB_FS is not set +# CONFIG_NCP_FS is not set +# CONFIG_ZISOFS_FS is not set +# CONFIG_ZLIB_FS_INFLATE is not set + +# +# Partition Types +# +CONFIG_PARTITION_ADVANCED=y +# CONFIG_ACORN_PARTITION is not set +# CONFIG_OSF_PARTITION is not set +# CONFIG_AMIGA_PARTITION is not set +# CONFIG_ATARI_PARTITION is not set +# CONFIG_MAC_PARTITION is not set +# CONFIG_MSDOS_PARTITION is not set +# CONFIG_LDM_PARTITION is not set +# CONFIG_SGI_PARTITION is not set +# CONFIG_ULTRIX_PARTITION is not set +# CONFIG_SUN_PARTITION is not set +# CONFIG_SMB_NLS is not set +# CONFIG_NLS is not set + +# +# Multimedia Capabilities Port drivers +# +# CONFIG_MCP is not set + +# +# USB support +# +# CONFIG_USB is not set + +# +# Bluetooth support +# +# CONFIG_BLUEZ is not set + +# +# Kernel hacking +# +CONFIG_FRAME_POINTER=y +CONFIG_DEBUG_USER=y +# CONFIG_DEBUG_INFO is not set +# CONFIG_DEBUG_KERNEL is not set diff -urN orig/arch/arm/def-configs/cep linux/arch/arm/def-configs/cep --- orig/arch/arm/def-configs/cep Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/def-configs/cep Sun Jun 16 12:54:24 2002 @@ -0,0 +1,615 @@ +# +# Automatically generated by make menuconfig: don't edit +# +CONFIG_ARM=y +# CONFIG_EISA is not set +# CONFIG_SBUS is not set +# CONFIG_MCA is not set +CONFIG_UID16=y +CONFIG_RWSEM_GENERIC_SPINLOCK=y +# CONFIG_RWSEM_XCHGADD_ALGORITHM is not set +# CONFIG_GENERIC_BUST_SPINLOCK is not set +# CONFIG_GENERIC_ISA_DMA is not set + +# +# Code maturity level options +# +# CONFIG_EXPERIMENTAL is not set +# CONFIG_OBSOLETE is not set + +# +# Loadable module support +# +CONFIG_MODULES=y +# CONFIG_MODVERSIONS is not set +# CONFIG_KMOD is not set + +# +# System Type +# +# CONFIG_ARCH_ANAKIN is not set +# CONFIG_ARCH_ARCA5K is not set +# CONFIG_ARCH_CLPS7500 is not set +# CONFIG_ARCH_CLPS711X is not set +# CONFIG_ARCH_CO285 is not set +# CONFIG_ARCH_EBSA110 is not set +# CONFIG_ARCH_CAMELOT is not set +# CONFIG_ARCH_FOOTBRIDGE is not set +# CONFIG_ARCH_INTEGRATOR is not set +# CONFIG_ARCH_L7200 is not set +# CONFIG_ARCH_MX1ADS is not set +# CONFIG_ARCH_RPC is not set +CONFIG_ARCH_SA1100=y +# CONFIG_ARCH_SHARK is not set + +# +# Archimedes/A5000 Implementations +# +# CONFIG_ARCH_ARC is not set +# CONFIG_ARCH_A5K is not set + +# +# Footbridge Implementations +# +# CONFIG_ARCH_CATS is not set +# CONFIG_ARCH_PERSONAL_SERVER is not set +# CONFIG_ARCH_EBSA285_ADDIN is not set +# CONFIG_ARCH_EBSA285_HOST is not set +# CONFIG_ARCH_NETWINDER is not set + +# +# SA11x0 Implementations +# +# CONFIG_SA1100_ASSABET is not set +# CONFIG_ASSABET_NEPONSET is not set +# CONFIG_SA1100_ADSBITSY is not set +# CONFIG_SA1100_BRUTUS is not set +CONFIG_SA1100_CEP=y +# CONFIG_SA1100_CERF is not set +# CONFIG_SA1100_H3100 is not set +# CONFIG_SA1100_H3600 is not set +# CONFIG_SA1100_H3800 is not set +# CONFIG_SA1100_H3XXX is not set +# CONFIG_SA1100_EXTENEX1 is not set +# CONFIG_SA1100_FLEXANET is not set +# CONFIG_SA1100_FREEBIRD is not set +# CONFIG_SA1100_FRODO is not set +# CONFIG_SA1100_GRAPHICSCLIENT is not set +# CONFIG_SA1100_GRAPHICSMASTER is not set +# CONFIG_SA1100_BADGE4 is not set +# CONFIG_SA1100_JORNADA720 is not set +# CONFIG_SA1100_HUW_WEBPANEL is not set +# CONFIG_SA1100_ITSY is not set +# CONFIG_SA1100_LART is not set +# CONFIG_SA1100_NANOENGINE is not set +# CONFIG_SA1100_OMNIMETER is not set +# CONFIG_SA1100_PANGOLIN is not set +# CONFIG_SA1100_PLEB is not set +# CONFIG_SA1100_PT_SYSTEM3 is not set +# CONFIG_SA1100_SHANNON is not set +# CONFIG_SA1100_SHERMAN is not set +# CONFIG_SA1100_SIMPAD is not set +# CONFIG_SA1100_PFS168 is not set +# CONFIG_SA1100_VICTOR is not set +# CONFIG_SA1100_XP860 is not set +# CONFIG_SA1100_YOPY is not set +# CONFIG_SA1100_USB is not set +# CONFIG_SA1100_USB_NETLINK is not set +# CONFIG_SA1100_USB_CHAR is not set +# CONFIG_H3600_SLEEVE is not set + +# +# CLPS711X/EP721X Implementations +# +# CONFIG_ARCH_AUTCPU12 is not set +# CONFIG_ARCH_CDB89712 is not set +# CONFIG_ARCH_CLEP7312 is not set +# CONFIG_ARCH_EDB7211 is not set +# CONFIG_ARCH_P720T is not set +# CONFIG_ARCH_FORTUNET is not set +# CONFIG_ARCH_EP7211 is not set +# CONFIG_ARCH_EP7212 is not set +# CONFIG_ARCH_ACORN is not set +# CONFIG_FOOTBRIDGE is not set +# CONFIG_FOOTBRIDGE_HOST is not set +# CONFIG_FOOTBRIDGE_ADDIN is not set +CONFIG_CPU_32=y +# CONFIG_CPU_26 is not set +# CONFIG_CPU_32v3 is not set +CONFIG_CPU_32v4=y +# CONFIG_CPU_ARM610 is not set +# CONFIG_CPU_ARM710 is not set +# CONFIG_CPU_ARM720T is not set +# CONFIG_CPU_ARM920T is not set +# CONFIG_CPU_ARM922T is not set +# CONFIG_CPU_ARM926T is not set +# CONFIG_CPU_ARM1020 is not set +# CONFIG_CPU_SA110 is not set +CONFIG_CPU_SA1100=y +# CONFIG_ARM_THUMB is not set +CONFIG_DISCONTIGMEM=y + +# +# General setup +# +# CONFIG_PCI is not set +CONFIG_ISA=y +# CONFIG_ISA_DMA is not set +# CONFIG_CPU_FREQ is not set +# CONFIG_HOTPLUG is not set +# CONFIG_PCMCIA is not set +CONFIG_NET=y +# CONFIG_SYSVIPC is not set +# CONFIG_BSD_PROCESS_ACCT is not set +# CONFIG_SYSCTL is not set +CONFIG_FPE_NWFPE=y +# CONFIG_FPE_FASTFPE is not set +CONFIG_KCORE_ELF=y +# CONFIG_KCORE_AOUT is not set +# CONFIG_BINFMT_AOUT is not set +CONFIG_BINFMT_ELF=y +# CONFIG_BINFMT_MISC is not set +# CONFIG_PM is not set +# CONFIG_ARTHUR is not set +CONFIG_CMDLINE="console=ttySA0,9600 root=/dev/ram" +# CONFIG_LEDS is not set +CONFIG_ALIGNMENT_TRAP=y + +# +# Parallel port support +# +# CONFIG_PARPORT is not set + +# +# Memory Technology Devices (MTD) +# +# CONFIG_MTD is not set + +# +# Plug and Play configuration +# +# CONFIG_PNP is not set +# CONFIG_ISAPNP is not set + +# +# Block devices +# +# CONFIG_BLK_DEV_FD is not set +# CONFIG_BLK_DEV_XD is not set +# CONFIG_PARIDE is not set +# CONFIG_BLK_CPQ_DA is not set +# CONFIG_BLK_CPQ_CISS_DA is not set +# CONFIG_BLK_DEV_DAC960 is not set +CONFIG_BLK_DEV_LOOP=y +# CONFIG_BLK_DEV_NBD is not set +CONFIG_BLK_DEV_RAM=y +CONFIG_BLK_DEV_RAM_SIZE=4096 +CONFIG_BLK_DEV_INITRD=y + +# +# Multi-device support (RAID and LVM) +# +# CONFIG_MD is not set +# CONFIG_BLK_DEV_MD is not set +# CONFIG_MD_LINEAR is not set +# CONFIG_MD_RAID0 is not set +# CONFIG_MD_RAID1 is not set +# CONFIG_MD_RAID5 is not set +# CONFIG_MD_MULTIPATH is not set +# CONFIG_BLK_DEV_LVM is not set + +# +# Networking options +# +# CONFIG_PACKET is not set +# CONFIG_NETLINK_DEV is not set +# CONFIG_NETFILTER is not set +# CONFIG_FILTER is not set +CONFIG_UNIX=y +CONFIG_INET=y +# CONFIG_IP_MULTICAST is not set +# CONFIG_IP_ADVANCED_ROUTER is not set +# CONFIG_IP_PNP is not set +# CONFIG_NET_IPIP is not set +# CONFIG_NET_IPGRE is not set +# CONFIG_INET_ECN is not set +# CONFIG_SYN_COOKIES is not set +# CONFIG_IPX is not set +# CONFIG_ATALK is not set +# CONFIG_DECNET is not set +# CONFIG_BRIDGE is not set + +# +# QoS and/or fair queueing +# +# CONFIG_NET_SCHED is not set + +# +# Network device support +# +CONFIG_NETDEVICES=y + +# +# ARCnet devices +# +# CONFIG_ARCNET is not set +# CONFIG_DUMMY is not set +# CONFIG_BONDING is not set +# CONFIG_EQUALIZER is not set +# CONFIG_TUN is not set + +# +# Ethernet (10 or 100Mbit) +# +CONFIG_NET_ETHERNET=y +# CONFIG_ARM_AM79C961A is not set +# CONFIG_SUNLANCE is not set +# CONFIG_SUNBMAC is not set +# CONFIG_SUNQE is not set +# CONFIG_SUNGEM is not set +# CONFIG_NET_VENDOR_3COM is not set +# CONFIG_LANCE is not set +# CONFIG_NET_VENDOR_SMC is not set +# CONFIG_NET_VENDOR_RACAL is not set +# CONFIG_AT1700 is not set +# CONFIG_DEPCA is not set +# CONFIG_HP100 is not set +# CONFIG_NET_ISA is not set +# CONFIG_NET_PCI is not set +# CONFIG_NET_POCKET is not set + +# +# Ethernet (1000 Mbit) +# +# CONFIG_ACENIC is not set +# CONFIG_DL2K is not set +# CONFIG_MYRI_SBUS is not set +# CONFIG_NS83820 is not set +# CONFIG_HAMACHI is not set +# CONFIG_YELLOWFIN is not set +# CONFIG_SK98LIN is not set +# CONFIG_FDDI is not set +# CONFIG_PLIP is not set +# CONFIG_PPP is not set +# CONFIG_SLIP is not set + +# +# Wireless LAN (non-hamradio) +# +# CONFIG_NET_RADIO is not set + +# +# Token Ring devices +# +# CONFIG_TR is not set +# CONFIG_NET_FC is not set + +# +# Wan interfaces +# +# CONFIG_WAN is not set + +# +# Amateur Radio support +# +# CONFIG_HAMRADIO is not set + +# +# IrDA (infrared) support +# +# CONFIG_IRDA is not set + +# +# ATA/IDE/MFM/RLL support +# +# CONFIG_IDE is not set +# CONFIG_BLK_DEV_IDE_MODES is not set +# CONFIG_BLK_DEV_HD is not set + +# +# SCSI support +# +# CONFIG_SCSI is not set + +# +# I2O device support +# +# CONFIG_I2O is not set +# CONFIG_I2O_BLOCK is not set +# CONFIG_I2O_LAN is not set +# CONFIG_I2O_SCSI is not set +# CONFIG_I2O_PROC is not set + +# +# ISDN subsystem +# +# CONFIG_ISDN is not set + +# +# Input core support +# +# CONFIG_INPUT is not set +# CONFIG_INPUT_KEYBDEV is not set +# CONFIG_INPUT_MOUSEDEV is not set +# CONFIG_INPUT_JOYDEV is not set +# CONFIG_INPUT_EVDEV is not set + +# +# Character devices +# +CONFIG_VT=y +CONFIG_VT_CONSOLE=y +CONFIG_SERIAL=y +CONFIG_SERIAL_CONSOLE=y +# CONFIG_SERIAL_EXTENDED is not set +# CONFIG_SERIAL_NONSTANDARD is not set + +# +# Serial drivers +# +# CONFIG_SERIAL_ANAKIN is not set +# CONFIG_SERIAL_ANAKIN_CONSOLE is not set +# CONFIG_SERIAL_AMBA is not set +# CONFIG_SERIAL_AMBA_CONSOLE is not set +# CONFIG_SERIAL_CLPS711X is not set +# CONFIG_SERIAL_CLPS711X_CONSOLE is not set +# CONFIG_SERIAL_21285 is not set +# CONFIG_SERIAL_21285_OLD is not set +# CONFIG_SERIAL_21285_CONSOLE is not set +# CONFIG_SERIAL_UART00 is not set +# CONFIG_SERIAL_UART00_CONSOLE is not set +CONFIG_SERIAL_SA1100=y +CONFIG_SERIAL_SA1100_CONSOLE=y +CONFIG_SA1100_DEFAULT_BAUDRATE=9600 +# CONFIG_SERIAL_8250 is not set +# CONFIG_SERIAL_8250_CONSOLE is not set +# CONFIG_SERIAL_8250_EXTENDED is not set +# CONFIG_SERIAL_8250_MANY_PORTS is not set +# CONFIG_SERIAL_8250_SHARE_IRQ is not set +# CONFIG_SERIAL_8250_DETECT_IRQ is not set +# CONFIG_SERIAL_8250_MULTIPORT is not set +# CONFIG_SERIAL_8250_HUB6 is not set +CONFIG_SERIAL_CORE=y +CONFIG_SERIAL_CORE_CONSOLE=y +# CONFIG_UNIX98_PTYS is not set + +# +# I2C support +# +# CONFIG_I2C is not set + +# +# L3 serial bus support +# +# CONFIG_L3 is not set +# CONFIG_L3_ALGOBIT is not set +# CONFIG_L3_BIT_SA1100_GPIO is not set +# CONFIG_L3_SA1111 is not set +# CONFIG_BIT_SA1100_GPIO is not set + +# +# Mice +# +# CONFIG_BUSMOUSE is not set +# CONFIG_MOUSE is not set + +# +# Joysticks +# +# CONFIG_INPUT_GAMEPORT is not set +# CONFIG_QIC02_TAPE is not set + +# +# Watchdog Cards +# +# CONFIG_WATCHDOG is not set +# CONFIG_INTEL_RNG is not set +# CONFIG_NVRAM is not set +# CONFIG_RTC is not set +# CONFIG_SA1100_RTC is not set +# CONFIG_DTLK is not set +# CONFIG_R3964 is not set +# CONFIG_APPLICOM is not set + +# +# Ftape, the floppy tape device driver +# +# CONFIG_FTAPE is not set +# CONFIG_AGP is not set +# CONFIG_DRM is not set + +# +# Multimedia devices +# +# CONFIG_VIDEO_DEV is not set + +# +# File systems +# +# CONFIG_QUOTA is not set +# CONFIG_AUTOFS_FS is not set +# CONFIG_AUTOFS4_FS is not set +# CONFIG_REISERFS_FS is not set +# CONFIG_REISERFS_CHECK is not set +# CONFIG_REISERFS_PROC_INFO is not set +# CONFIG_ADFS_FS is not set +# CONFIG_ADFS_FS_RW is not set +# CONFIG_AFFS_FS is not set +# CONFIG_HFS_FS is not set +# CONFIG_BFS_FS is not set +# CONFIG_EXT3_FS is not set +# CONFIG_JBD is not set +# CONFIG_JBD_DEBUG is not set +# CONFIG_FAT_FS is not set +# CONFIG_MSDOS_FS is not set +# CONFIG_UMSDOS_FS is not set +# CONFIG_VFAT_FS is not set +# CONFIG_EFS_FS is not set +# CONFIG_JFFS_FS is not set +# CONFIG_JFFS2_FS is not set +# CONFIG_CRAMFS is not set +# CONFIG_TMPFS is not set +# CONFIG_RAMFS is not set +# CONFIG_ISO9660_FS is not set +# CONFIG_JOLIET is not set +# CONFIG_ZISOFS is not set +# CONFIG_MINIX_FS is not set +# CONFIG_VXFS_FS is not set +# CONFIG_NTFS_FS is not set +# CONFIG_NTFS_RW is not set +# CONFIG_HPFS_FS is not set +CONFIG_PROC_FS=y +# CONFIG_DEVFS_FS is not set +# CONFIG_DEVFS_MOUNT is not set +# CONFIG_DEVFS_DEBUG is not set +# CONFIG_DEVPTS_FS is not set +# CONFIG_QNX4FS_FS is not set +# CONFIG_QNX4FS_RW is not set +# CONFIG_ROMFS_FS is not set +CONFIG_EXT2_FS=y +# CONFIG_SYSV_FS is not set +# CONFIG_UDF_FS is not set +# CONFIG_UDF_RW is not set +# CONFIG_UFS_FS is not set +# CONFIG_UFS_FS_WRITE is not set + +# +# Network File Systems +# +# CONFIG_CODA_FS is not set +# CONFIG_INTERMEZZO_FS is not set +# CONFIG_NFS_FS is not set +# CONFIG_NFS_V3 is not set +# CONFIG_ROOT_NFS is not set +# CONFIG_NFSD is not set +# CONFIG_NFSD_V3 is not set +# CONFIG_SUNRPC is not set +# CONFIG_LOCKD is not set +# CONFIG_SMB_FS is not set +# CONFIG_NCP_FS is not set +# CONFIG_NCPFS_PACKET_SIGNING is not set +# CONFIG_NCPFS_IOCTL_LOCKING is not set +# CONFIG_NCPFS_STRONG is not set +# CONFIG_NCPFS_NFS_NS is not set +# CONFIG_NCPFS_OS2_NS is not set +# CONFIG_NCPFS_SMALLDOS is not set +# CONFIG_NCPFS_NLS is not set +# CONFIG_NCPFS_EXTRAS is not set +# CONFIG_ZISOFS_FS is not set +# CONFIG_ZLIB_FS_INFLATE is not set + +# +# Partition Types +# +# CONFIG_PARTITION_ADVANCED is not set +CONFIG_MSDOS_PARTITION=y +# CONFIG_SMB_NLS is not set +# CONFIG_NLS is not set + +# +# Console drivers +# +CONFIG_PC_KEYMAP=y +# CONFIG_VGA_CONSOLE is not set + +# +# Frame-buffer support +# +# CONFIG_FB is not set + +# +# Sound +# +# CONFIG_SOUND is not set + +# +# Multimedia Capabilities Port drivers +# +# CONFIG_MCP is not set +# CONFIG_MCP_SA1100 is not set +# CONFIG_MCP_UCB1200 is not set +# CONFIG_MCP_UCB1200_AUDIO is not set +# CONFIG_MCP_UCB1200_TS is not set + +# +# USB support +# +# CONFIG_USB is not set +# CONFIG_USB_UHCI is not set +# CONFIG_USB_UHCI_ALT is not set +# CONFIG_USB_OHCI is not set +# CONFIG_USB_OHCI_SA1111 is not set +# CONFIG_USB_AUDIO is not set +# CONFIG_USB_BLUETOOTH is not set +# CONFIG_USB_STORAGE is not set +# CONFIG_USB_STORAGE_DEBUG is not set +# CONFIG_USB_STORAGE_DATAFAB is not set +# CONFIG_USB_STORAGE_FREECOM is not set +# CONFIG_USB_STORAGE_ISD200 is not set +# CONFIG_USB_STORAGE_DPCM is not set +# CONFIG_USB_STORAGE_HP8200e is not set +# CONFIG_USB_STORAGE_SDDR09 is not set +# CONFIG_USB_STORAGE_JUMPSHOT is not set +# CONFIG_USB_ACM is not set +# CONFIG_USB_PRINTER is not set +# CONFIG_USB_DC2XX is not set +# CONFIG_USB_MDC800 is not set +# CONFIG_USB_SCANNER is not set +# CONFIG_USB_MICROTEK is not set +# CONFIG_USB_HPUSBSCSI is not set +# CONFIG_USB_PEGASUS is not set +# CONFIG_USB_KAWETH is not set +# CONFIG_USB_CATC is not set +# CONFIG_USB_CDCETHER is not set +# CONFIG_USB_USBNET is not set +# CONFIG_USB_USS720 is not set + +# +# USB Serial Converter support +# +# CONFIG_USB_SERIAL is not set +# CONFIG_USB_SERIAL_GENERIC is not set +# CONFIG_USB_SERIAL_BELKIN is not set +# CONFIG_USB_SERIAL_WHITEHEAT is not set +# CONFIG_USB_SERIAL_DIGI_ACCELEPORT is not set +# CONFIG_USB_SERIAL_EMPEG is not set +# CONFIG_USB_SERIAL_FTDI_SIO is not set +# CONFIG_USB_SERIAL_VISOR is not set +# CONFIG_USB_SERIAL_IPAQ is not set +# CONFIG_USB_SERIAL_IR is not set +# CONFIG_USB_SERIAL_EDGEPORT is not set +# CONFIG_USB_SERIAL_KEYSPAN_PDA is not set +# CONFIG_USB_SERIAL_KEYSPAN is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XA is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XB is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA18X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19W is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA49W is not set +# CONFIG_USB_SERIAL_MCT_U232 is not set +# CONFIG_USB_SERIAL_KLSI is not set +# CONFIG_USB_SERIAL_PL2303 is not set +# CONFIG_USB_SERIAL_CYBERJACK is not set +# CONFIG_USB_SERIAL_XIRCOM is not set +# CONFIG_USB_SERIAL_OMNINET is not set +# CONFIG_USB_RIO500 is not set + +# +# Kernel hacking +# +# CONFIG_NO_FRAME_POINTER is not set +CONFIG_DEBUG_USER=y +# CONFIG_DEBUG_INFO is not set +# CONFIG_NO_PGT_CACHE is not set +# CONFIG_DEBUG_KERNEL is not set +# CONFIG_DEBUG_SLAB is not set +# CONFIG_MAGIC_SYSRQ is not set +# CONFIG_DEBUG_SPINLOCK is not set +# CONFIG_DEBUG_WAITQ is not set +# CONFIG_DEBUG_BUGVERBOSE is not set +# CONFIG_DEBUG_ERRORS is not set +# CONFIG_DEBUG_LL is not set +# CONFIG_DEBUG_DC21285_PORT is not set +# CONFIG_DEBUG_CLPS711X_UART2 is not set diff -urN orig/arch/arm/def-configs/epxa10db linux/arch/arm/def-configs/epxa10db --- orig/arch/arm/def-configs/epxa10db Mon Aug 5 13:29:43 2002 +++ linux/arch/arm/def-configs/epxa10db Fri Feb 21 15:17:33 2003 @@ -8,6 +8,8 @@ CONFIG_UID16=y CONFIG_RWSEM_GENERIC_SPINLOCK=y # CONFIG_RWSEM_XCHGADD_ALGORITHM is not set +# CONFIG_GENERIC_BUST_SPINLOCK is not set +# CONFIG_GENERIC_ISA_DMA is not set # # Code maturity level options @@ -34,10 +36,14 @@ CONFIG_ARCH_CAMELOT=y # CONFIG_ARCH_FOOTBRIDGE is not set # CONFIG_ARCH_INTEGRATOR is not set +# CONFIG_ARCH_OMAHA is not set # CONFIG_ARCH_L7200 is not set +# CONFIG_ARCH_MX1ADS is not set # CONFIG_ARCH_RPC is not set +# CONFIG_ARCH_RISCSTATION is not set # CONFIG_ARCH_SA1100 is not set # CONFIG_ARCH_SHARK is not set +# CONFIG_ARCH_AT91RM9200DK is not set # # Archimedes/A5000 Implementations @@ -57,17 +63,25 @@ # # SA11x0 Implementations # +# CONFIG_SA1100_ACCELENT is not set # CONFIG_SA1100_ASSABET is not set # CONFIG_ASSABET_NEPONSET is not set # CONFIG_SA1100_ADSBITSY is not set # CONFIG_SA1100_BRUTUS is not set +# CONFIG_SA1100_CEP is not set # CONFIG_SA1100_CERF is not set +# CONFIG_SA1100_H3100 is not set # CONFIG_SA1100_H3600 is not set +# CONFIG_SA1100_H3800 is not set +# CONFIG_SA1100_H3XXX is not set # CONFIG_SA1100_EXTENEX1 is not set # CONFIG_SA1100_FLEXANET is not set # CONFIG_SA1100_FREEBIRD is not set +# CONFIG_SA1100_FRODO is not set # CONFIG_SA1100_GRAPHICSCLIENT is not set # CONFIG_SA1100_GRAPHICSMASTER is not set +# CONFIG_SA1100_HACKKIT is not set +# CONFIG_SA1100_BADGE4 is not set # CONFIG_SA1100_JORNADA720 is not set # CONFIG_SA1100_HUW_WEBPANEL is not set # CONFIG_SA1100_ITSY is not set @@ -76,42 +90,59 @@ # CONFIG_SA1100_OMNIMETER is not set # CONFIG_SA1100_PANGOLIN is not set # CONFIG_SA1100_PLEB is not set +# CONFIG_SA1100_PT_SYSTEM3 is not set +# CONFIG_SA1100_SHANNON is not set # CONFIG_SA1100_SHERMAN is not set # CONFIG_SA1100_SIMPAD is not set +# CONFIG_SA1100_SIMPUTER is not set # CONFIG_SA1100_PFS168 is not set # CONFIG_SA1100_VICTOR is not set # CONFIG_SA1100_XP860 is not set # CONFIG_SA1100_YOPY is not set +# CONFIG_SA1100_USB is not set +# CONFIG_SA1100_USB_NETLINK is not set +# CONFIG_SA1100_USB_CHAR is not set +# CONFIG_H3600_SLEEVE is not set # # CLPS711X/EP721X Implementations # +# CONFIG_ARCH_AUTCPU12 is not set # CONFIG_ARCH_CDB89712 is not set # CONFIG_ARCH_CLEP7312 is not set # CONFIG_ARCH_EDB7211 is not set # CONFIG_ARCH_P720T is not set +# CONFIG_ARCH_FORTUNET is not set # CONFIG_ARCH_EP7211 is not set # CONFIG_ARCH_EP7212 is not set # CONFIG_ARCH_ACORN is not set # CONFIG_FOOTBRIDGE is not set # CONFIG_FOOTBRIDGE_HOST is not set # CONFIG_FOOTBRIDGE_ADDIN is not set +CONFIG_EPXA10DB=y +# CONFIG_EPXA1DB is not set +# CONFIG_EPXA10DB_R2 is not set +CONFIG_EPXA10DB_R3=y +CONFIG_PLD=y +# CONFIG_PLD_HOTSWAP is not set CONFIG_CPU_32=y # CONFIG_CPU_26 is not set -# CONFIG_CPU_32v3 is not set -CONFIG_CPU_32v4=y # CONFIG_CPU_ARM610 is not set # CONFIG_CPU_ARM710 is not set # CONFIG_CPU_ARM720T is not set +# CONFIG_CPU_ARM920T is not set CONFIG_CPU_ARM922T=y -CONFIG_CPU_ARM92X_CPU_IDLE=y -CONFIG_CPU_ARM92X_I_CACHE_ON=y -CONFIG_CPU_ARM92X_D_CACHE_ON=y -# CONFIG_CPU_ARM92X_WRITETHROUGH is not set +CONFIG_PLD=y +# CONFIG_CPU_ARM926T is not set # CONFIG_CPU_ARM1020 is not set +# CONFIG_CPU_ARM1026 is not set # CONFIG_CPU_SA110 is not set # CONFIG_CPU_SA1100 is not set +# CONFIG_CPU_32v3 is not set +CONFIG_CPU_32v4=y # CONFIG_ARM_THUMB is not set +# CONFIG_CPU_ICACHE_DISABLE is not set +# CONFIG_CPU_DCACHE_DISABLE is not set # CONFIG_DISCONTIGMEM is not set # @@ -120,6 +151,9 @@ # CONFIG_PCI is not set # CONFIG_ISA is not set # CONFIG_ISA_DMA is not set +# CONFIG_ZBOOT_ROM is not set +CONFIG_ZBOOT_ROM_TEXT=0 +CONFIG_ZBOOT_ROM_BSS=0 # CONFIG_HOTPLUG is not set # CONFIG_PCMCIA is not set CONFIG_NET=y @@ -130,7 +164,7 @@ # CONFIG_FPE_FASTFPE is not set CONFIG_KCORE_ELF=y # CONFIG_KCORE_AOUT is not set -CONFIG_BINFMT_AOUT=y +# CONFIG_BINFMT_AOUT is not set CONFIG_BINFMT_ELF=y # CONFIG_BINFMT_MISC is not set # CONFIG_PM is not set @@ -150,8 +184,9 @@ CONFIG_MTD_DEBUG=y CONFIG_MTD_DEBUG_VERBOSE=0 CONFIG_MTD_PARTITIONS=y -# CONFIG_MTD_REDBOOT_PARTS is not set -CONFIG_MTD_BOOTLDR_PARTS=y +# CONFIG_MTD_CONCAT is not set +CONFIG_MTD_REDBOOT_PARTS=y +# CONFIG_MTD_CMDLINE_PARTS is not set # CONFIG_MTD_AFS_PARTS is not set CONFIG_MTD_CHAR=y CONFIG_MTD_BLOCK=y @@ -171,42 +206,32 @@ # CONFIG_MTD_ROM is not set # CONFIG_MTD_ABSENT is not set # CONFIG_MTD_OBSOLETE_CHIPS is not set +# CONFIG_MTD_AMDSTD is not set +# CONFIG_MTD_SHARP is not set +# CONFIG_MTD_JEDEC is not set # # Mapping drivers for chip access # # CONFIG_MTD_PHYSMAP is not set -# CONFIG_MTD_SUN_UFLASH is not set # CONFIG_MTD_NORA is not set -# CONFIG_MTD_PNC2000 is not set -# CONFIG_MTD_RPXLITE is not set -# CONFIG_MTD_TQM8XXL is not set -# CONFIG_MTD_SC520CDP is not set -# CONFIG_MTD_NETSC520 is not set -# CONFIG_MTD_SBC_GXX is not set -# CONFIG_MTD_ELAN_104NC is not set -# CONFIG_MTD_DBOX2 is not set -# CONFIG_MTD_CSTM_MIPS_IXX is not set -CONFIG_MTD_EPXA10DB=y -# CONFIG_MTD_CFI_FLAGADM is not set -# CONFIG_MTD_SOLUTIONENGINE is not set -# CONFIG_MTD_MIXMEM is not set -# CONFIG_MTD_OCTAGON is not set -# CONFIG_MTD_VMAX is not set -# CONFIG_MTD_OCELOT is not set -# CONFIG_MTD_L440GX is not set # CONFIG_MTD_ARM_INTEGRATOR is not set # CONFIG_MTD_CDB89712 is not set # CONFIG_MTD_SA1100 is not set # CONFIG_MTD_DC21285 is not set # CONFIG_MTD_IQ80310 is not set +# CONFIG_MTD_FORTUNET is not set +CONFIG_MTD_EPXA=y +# CONFIG_MTD_AUTCPU12 is not set +# CONFIG_MTD_EDB7312 is not set +# CONFIG_MTD_IMPA7 is not set +# CONFIG_MTD_PCI is not set # # Self-contained MTD device drivers # # CONFIG_MTD_PMC551 is not set # CONFIG_MTD_SLRAM is not set -# CONFIG_MTD_LART is not set # CONFIG_MTD_MTDRAM is not set # CONFIG_MTD_BLKMTD is not set # CONFIG_MTD_DOC1000 is not set @@ -224,7 +249,6 @@ # # CONFIG_PNP is not set # CONFIG_ISAPNP is not set -# CONFIG_PNPBIOS is not set # # Block devices @@ -234,7 +258,9 @@ # CONFIG_PARIDE is not set # CONFIG_BLK_CPQ_DA is not set # CONFIG_BLK_CPQ_CISS_DA is not set +# CONFIG_CISS_SCSI_TAPE is not set # CONFIG_BLK_DEV_DAC960 is not set +# CONFIG_BLK_DEV_UMEM is not set # CONFIG_BLK_DEV_LOOP is not set # CONFIG_BLK_DEV_NBD is not set CONFIG_BLK_DEV_RAM=y @@ -250,6 +276,7 @@ # CONFIG_MD_RAID0 is not set # CONFIG_MD_RAID1 is not set # CONFIG_MD_RAID5 is not set +# CONFIG_MD_MULTIPATH is not set # CONFIG_BLK_DEV_LVM is not set # @@ -257,7 +284,7 @@ # CONFIG_PACKET=y # CONFIG_PACKET_MMAP is not set -# CONFIG_NETLINK is not set +# CONFIG_NETLINK_DEV is not set # CONFIG_NETFILTER is not set # CONFIG_FILTER is not set CONFIG_UNIX=y @@ -270,13 +297,20 @@ # CONFIG_IP_PNP_RARP is not set # CONFIG_NET_IPIP is not set # CONFIG_NET_IPGRE is not set +# CONFIG_ARPD is not set # CONFIG_INET_ECN is not set # CONFIG_SYN_COOKIES is not set # CONFIG_IPV6 is not set # CONFIG_KHTTPD is not set # CONFIG_ATM is not set +# CONFIG_VLAN_8021Q is not set # CONFIG_IPX is not set # CONFIG_ATALK is not set + +# +# Appletalk devices +# +# CONFIG_DEV_APPLETALK is not set # CONFIG_DECNET is not set # CONFIG_BRIDGE is not set # CONFIG_X25 is not set @@ -294,6 +328,11 @@ # CONFIG_NET_SCHED is not set # +# Network testing +# +# CONFIG_NET_PKTGEN is not set + +# # Network device support # CONFIG_NETDEVICES=y @@ -306,16 +345,18 @@ # CONFIG_BONDING is not set # CONFIG_EQUALIZER is not set # CONFIG_TUN is not set +# CONFIG_ETHERTAP is not set # # Ethernet (10 or 100Mbit) # CONFIG_NET_ETHERNET=y +# CONFIG_ARM_AM79C961A is not set +# CONFIG_ARM_CIRRUS is not set CONFIG_ETHER00=y # CONFIG_SUNLANCE is not set # CONFIG_SUNBMAC is not set # CONFIG_SUNQE is not set -# CONFIG_SUNLANCE is not set # CONFIG_SUNGEM is not set # CONFIG_NET_VENDOR_3COM is not set # CONFIG_LANCE is not set @@ -335,6 +376,7 @@ # CONFIG_HAMACHI is not set # CONFIG_YELLOWFIN is not set # CONFIG_SK98LIN is not set +# CONFIG_TIGON3 is not set # CONFIG_FDDI is not set # CONFIG_HIPPI is not set # CONFIG_PLIP is not set @@ -343,8 +385,8 @@ # CONFIG_PPP_FILTER is not set CONFIG_PPP_ASYNC=y CONFIG_PPP_SYNC_TTY=y -CONFIG_PPP_DEFLATE=y -CONFIG_PPP_BSDCOMP=y +# CONFIG_PPP_DEFLATE is not set +# CONFIG_PPP_BSDCOMP is not set # CONFIG_PPPOE is not set # CONFIG_SLIP is not set @@ -377,7 +419,7 @@ # CONFIG_IRDA is not set # -# ATA/IDE/MFM/RLL support +# ATA/ATAPI/MFM/RLL support # # CONFIG_IDE is not set # CONFIG_BLK_DEV_IDE_MODES is not set @@ -435,6 +477,8 @@ CONFIG_SERIAL_UART00_CONSOLE=y # CONFIG_SERIAL_SA1100 is not set # CONFIG_SERIAL_SA1100_CONSOLE is not set +# CONFIG_SERIAL_OMAHA is not set +# CONFIG_SERIAL_OMAHA_CONSOLE is not set # CONFIG_SERIAL_8250 is not set # CONFIG_SERIAL_8250_CONSOLE is not set # CONFIG_SERIAL_8250_EXTENDED is not set @@ -460,7 +504,6 @@ # CONFIG_L3_ALGOBIT is not set # CONFIG_L3_BIT_SA1100_GPIO is not set # CONFIG_L3_SA1111 is not set -# CONFIG_L3_DRV_UDA1341 is not set # CONFIG_BIT_SA1100_GPIO is not set # @@ -479,13 +522,11 @@ # Watchdog Cards # # CONFIG_WATCHDOG is not set -# CONFIG_INTEL_RNG is not set # CONFIG_NVRAM is not set # CONFIG_RTC is not set # CONFIG_DTLK is not set # CONFIG_R3964 is not set # CONFIG_APPLICOM is not set -# CONFIG_SONYPI is not set # # Ftape, the floppy tape device driver @@ -513,7 +554,6 @@ # CONFIG_AFFS_FS is not set # CONFIG_HFS_FS is not set # CONFIG_BFS_FS is not set -# CONFIG_CMS_FS is not set # CONFIG_EXT3_FS is not set # CONFIG_JBD is not set # CONFIG_JBD_DEBUG is not set @@ -522,16 +562,17 @@ # CONFIG_UMSDOS_FS is not set # CONFIG_VFAT_FS is not set # CONFIG_EFS_FS is not set -CONFIG_JFFS_FS=y -CONFIG_JFFS_FS_VERBOSE=0 -# CONFIG_JFFS2_FS is not set +# CONFIG_JFFS_FS is not set +CONFIG_JFFS2_FS=y +CONFIG_JFFS2_FS_DEBUG=0 # CONFIG_CRAMFS is not set -# CONFIG_TMPFS is not set -# CONFIG_RAMFS is not set +CONFIG_TMPFS=y +CONFIG_RAMFS=y # CONFIG_ISO9660_FS is not set # CONFIG_JOLIET is not set +# CONFIG_ZISOFS is not set # CONFIG_MINIX_FS is not set -# CONFIG_FREEVXFS_FS is not set +# CONFIG_VXFS_FS is not set # CONFIG_NTFS_FS is not set # CONFIG_NTFS_RW is not set # CONFIG_HPFS_FS is not set @@ -557,7 +598,7 @@ # CONFIG_INTERMEZZO_FS is not set CONFIG_NFS_FS=y # CONFIG_NFS_V3 is not set -# CONFIG_ROOT_NFS is not set +CONFIG_ROOT_NFS=y # CONFIG_NFSD is not set # CONFIG_NFSD_V3 is not set CONFIG_SUNRPC=y @@ -572,6 +613,8 @@ # CONFIG_NCPFS_SMALLDOS is not set # CONFIG_NCPFS_NLS is not set # CONFIG_NCPFS_EXTRAS is not set +# CONFIG_ZISOFS_FS is not set +# CONFIG_ZLIB_FS_INFLATE is not set # # Partition Types @@ -582,66 +625,18 @@ # CONFIG_NLS is not set # +# Multimedia Capabilities Port drivers +# +# CONFIG_MCP is not set +# CONFIG_MCP_SA1100 is not set +# CONFIG_MCP_UCB1200 is not set +# CONFIG_MCP_UCB1200_AUDIO is not set +# CONFIG_MCP_UCB1200_TS is not set + +# # USB support # # CONFIG_USB is not set -# CONFIG_USB_UHCI is not set -# CONFIG_USB_UHCI_ALT is not set -# CONFIG_USB_OHCI is not set -# CONFIG_USB_OHCI_SA1111 is not set -# CONFIG_USB_AUDIO is not set -# CONFIG_USB_BLUETOOTH is not set -# CONFIG_USB_STORAGE is not set -# CONFIG_USB_STORAGE_DEBUG is not set -# CONFIG_USB_STORAGE_DATAFAB is not set -# CONFIG_USB_STORAGE_FREECOM is not set -# CONFIG_USB_STORAGE_ISD200 is not set -# CONFIG_USB_STORAGE_JUMPSHOT is not set -# CONFIG_USB_STORAGE_DPCM is not set -# CONFIG_USB_STORAGE_HP8200e is not set -# CONFIG_USB_STORAGE_SDDR09 is not set -# CONFIG_USB_ACM is not set -# CONFIG_USB_PRINTER is not set -# CONFIG_USB_DC2XX is not set -# CONFIG_USB_MDC800 is not set -# CONFIG_USB_SCANNER is not set -# CONFIG_USB_MICROTEK is not set -# CONFIG_USB_HPUSBSCSI is not set -# CONFIG_USB_DABUSB is not set -# CONFIG_USB_PLUSB is not set -# CONFIG_USB_PEGASUS is not set -# CONFIG_USB_KAWETH is not set -# CONFIG_USB_CATC is not set -# CONFIG_USB_CDCETHER is not set -# CONFIG_USB_USBNET is not set -# CONFIG_USB_USS720 is not set - -# -# USB Serial Converter support -# -# CONFIG_USB_SERIAL is not set -# CONFIG_USB_SERIAL_GENERIC is not set -# CONFIG_USB_SERIAL_BELKIN is not set -# CONFIG_USB_SERIAL_WHITEHEAT is not set -# CONFIG_USB_SERIAL_DIGI_ACCELEPORT is not set -# CONFIG_USB_SERIAL_EMPEG is not set -# CONFIG_USB_SERIAL_FTDI_SIO is not set -# CONFIG_USB_SERIAL_VISOR is not set -# CONFIG_USB_SERIAL_EDGEPORT is not set -# CONFIG_USB_SERIAL_KEYSPAN_PDA is not set -# CONFIG_USB_SERIAL_KEYSPAN is not set -# CONFIG_USB_SERIAL_KEYSPAN_USA28 is not set -# CONFIG_USB_SERIAL_KEYSPAN_USA28X is not set -# CONFIG_USB_SERIAL_KEYSPAN_USA19 is not set -# CONFIG_USB_SERIAL_KEYSPAN_USA18X is not set -# CONFIG_USB_SERIAL_KEYSPAN_USA19W is not set -# CONFIG_USB_SERIAL_KEYSPAN_USA49W is not set -# CONFIG_USB_SERIAL_MCT_U232 is not set -# CONFIG_USB_SERIAL_PL2303 is not set -# CONFIG_USB_SERIAL_CYBERJACK is not set -# CONFIG_USB_SERIAL_OMNINET is not set -# CONFIG_USB_RIO500 is not set -# CONFIG_USB_ID75 is not set # # Bluetooth support @@ -652,11 +647,16 @@ # Kernel hacking # CONFIG_FRAME_POINTER=y -CONFIG_DEBUG_ERRORS=y CONFIG_DEBUG_USER=y CONFIG_DEBUG_INFO=y -CONFIG_MAGIC_SYSRQ=y # CONFIG_NO_PGT_CACHE is not set +CONFIG_DEBUG_KERNEL=y +CONFIG_DEBUG_SLAB=y +CONFIG_MAGIC_SYSRQ=y +CONFIG_DEBUG_SPINLOCK=y +CONFIG_DEBUG_WAITQ=y +CONFIG_DEBUG_BUGVERBOSE=y +CONFIG_DEBUG_ERRORS=y CONFIG_DEBUG_LL=y # CONFIG_DEBUG_DC21285_PORT is not set # CONFIG_DEBUG_CLPS711X_UART2 is not set diff -urN orig/arch/arm/def-configs/epxa1db linux/arch/arm/def-configs/epxa1db --- orig/arch/arm/def-configs/epxa1db Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/def-configs/epxa1db Mon Aug 5 23:25:12 2002 @@ -0,0 +1,703 @@ +# +# Automatically generated by make menuconfig: don't edit +# +CONFIG_ARM=y +# CONFIG_EISA is not set +# CONFIG_SBUS is not set +# CONFIG_MCA is not set +CONFIG_UID16=y +CONFIG_RWSEM_GENERIC_SPINLOCK=y +# CONFIG_RWSEM_XCHGADD_ALGORITHM is not set +# CONFIG_GENERIC_BUST_SPINLOCK is not set +# CONFIG_GENERIC_ISA_DMA is not set + +# +# Code maturity level options +# +CONFIG_EXPERIMENTAL=y +# CONFIG_OBSOLETE is not set + +# +# Loadable module support +# +CONFIG_MODULES=y +# CONFIG_MODVERSIONS is not set +# CONFIG_KMOD is not set + +# +# System Type +# +# CONFIG_ARCH_ANAKIN is not set +# CONFIG_ARCH_ARCA5K is not set +# CONFIG_ARCH_CLPS7500 is not set +# CONFIG_ARCH_CLPS711X is not set +# CONFIG_ARCH_CO285 is not set +# CONFIG_ARCH_EBSA110 is not set +CONFIG_ARCH_CAMELOT=y +# CONFIG_ARCH_FOOTBRIDGE is not set +# CONFIG_ARCH_INTEGRATOR is not set +# CONFIG_ARCH_L7200 is not set +# CONFIG_ARCH_MX1ADS is not set +# CONFIG_ARCH_RPC is not set +# CONFIG_ARCH_SA1100 is not set +# CONFIG_ARCH_SHARK is not set + +# +# Archimedes/A5000 Implementations +# +# CONFIG_ARCH_ARC is not set +# CONFIG_ARCH_A5K is not set + +# +# Footbridge Implementations +# +# CONFIG_ARCH_CATS is not set +# CONFIG_ARCH_PERSONAL_SERVER is not set +# CONFIG_ARCH_EBSA285_ADDIN is not set +# CONFIG_ARCH_EBSA285_HOST is not set +# CONFIG_ARCH_NETWINDER is not set + +# +# SA11x0 Implementations +# +# CONFIG_SA1100_ASSABET is not set +# CONFIG_ASSABET_NEPONSET is not set +# CONFIG_SA1100_ADSBITSY is not set +# CONFIG_SA1100_BRUTUS is not set +# CONFIG_SA1100_CEP is not set +# CONFIG_SA1100_CERF is not set +# CONFIG_SA1100_H3100 is not set +# CONFIG_SA1100_H3600 is not set +# CONFIG_SA1100_H3800 is not set +# CONFIG_SA1100_H3XXX is not set +# CONFIG_SA1100_EXTENEX1 is not set +# CONFIG_SA1100_FLEXANET is not set +# CONFIG_SA1100_FREEBIRD is not set +# CONFIG_SA1100_FRODO is not set +# CONFIG_SA1100_GRAPHICSCLIENT is not set +# CONFIG_SA1100_GRAPHICSMASTER is not set +# CONFIG_SA1100_BADGE4 is not set +# CONFIG_SA1100_JORNADA720 is not set +# CONFIG_SA1100_HUW_WEBPANEL is not set +# CONFIG_SA1100_ITSY is not set +# CONFIG_SA1100_LART is not set +# CONFIG_SA1100_NANOENGINE is not set +# CONFIG_SA1100_OMNIMETER is not set +# CONFIG_SA1100_PANGOLIN is not set +# CONFIG_SA1100_PLEB is not set +# CONFIG_SA1100_PT_SYSTEM3 is not set +# CONFIG_SA1100_SHANNON is not set +# CONFIG_SA1100_SHERMAN is not set +# CONFIG_SA1100_SIMPAD is not set +# CONFIG_SA1100_PFS168 is not set +# CONFIG_SA1100_VICTOR is not set +# CONFIG_SA1100_XP860 is not set +# CONFIG_SA1100_YOPY is not set +# CONFIG_SA1100_USB is not set +# CONFIG_SA1100_USB_NETLINK is not set +# CONFIG_SA1100_USB_CHAR is not set +# CONFIG_H3600_SLEEVE is not set + +# +# CLPS711X/EP721X Implementations +# +# CONFIG_ARCH_AUTCPU12 is not set +# CONFIG_ARCH_CDB89712 is not set +# CONFIG_ARCH_CLEP7312 is not set +# CONFIG_ARCH_EDB7211 is not set +# CONFIG_ARCH_P720T is not set +# CONFIG_ARCH_FORTUNET is not set +# CONFIG_ARCH_EP7211 is not set +# CONFIG_ARCH_EP7212 is not set +# CONFIG_ARCH_ACORN is not set +# CONFIG_FOOTBRIDGE is not set +# CONFIG_FOOTBRIDGE_HOST is not set +# CONFIG_FOOTBRIDGE_ADDIN is not set +CONFIG_CPU_32=y +# CONFIG_CPU_26 is not set +# CONFIG_EPXA10DB is not set +CONFIG_EPXA1DB=y +# CONFIG_CPU_32v3 is not set +CONFIG_CPU_32v4=y +# CONFIG_CPU_ARM610 is not set +# CONFIG_CPU_ARM710 is not set +# CONFIG_CPU_ARM720T is not set +# CONFIG_CPU_ARM920T is not set +CONFIG_CPU_ARM922T=y +CONFIG_PLD=y +CONFIG_CPU_ARM922_CPU_IDLE=y +CONFIG_CPU_ARM922_I_CACHE_ON=y +CONFIG_CPU_ARM922_D_CACHE_ON=y +# CONFIG_CPU_ARM922_WRITETHROUGH is not set +# CONFIG_CPU_ARM926T is not set +# CONFIG_CPU_ARM1020 is not set +# CONFIG_CPU_SA110 is not set +# CONFIG_CPU_SA1100 is not set +# CONFIG_ARM_THUMB is not set +# CONFIG_DISCONTIGMEM is not set + +# +# General setup +# +# CONFIG_PCI is not set +# CONFIG_ISA is not set +# CONFIG_ISA_DMA is not set +# CONFIG_ZBOOT_ROM is not set +CONFIG_ZBOOT_ROM_TEXT=0 +CONFIG_ZBOOT_ROM_BSS=0 +# CONFIG_HOTPLUG is not set +# CONFIG_PCMCIA is not set +CONFIG_NET=y +CONFIG_SYSVIPC=y +# CONFIG_BSD_PROCESS_ACCT is not set +CONFIG_SYSCTL=y +CONFIG_FPE_NWFPE=y +# CONFIG_FPE_FASTFPE is not set +CONFIG_KCORE_ELF=y +# CONFIG_KCORE_AOUT is not set +# CONFIG_BINFMT_AOUT is not set +CONFIG_BINFMT_ELF=y +# CONFIG_BINFMT_MISC is not set +# CONFIG_PM is not set +# CONFIG_ARTHUR is not set +CONFIG_CMDLINE="console=ttyUA0,38400 root=/dev/mtdblock0 rw" +CONFIG_ALIGNMENT_TRAP=y + +# +# Parallel port support +# +# CONFIG_PARPORT is not set + +# +# Memory Technology Devices (MTD) +# +CONFIG_MTD=y +CONFIG_MTD_DEBUG=y +CONFIG_MTD_DEBUG_VERBOSE=0 +CONFIG_MTD_PARTITIONS=y +# CONFIG_MTD_CONCAT is not set +CONFIG_MTD_REDBOOT_PARTS=y +# CONFIG_MTD_CMDLINE_PARTS is not set +# CONFIG_MTD_AFS_PARTS is not set +CONFIG_MTD_CHAR=y +CONFIG_MTD_BLOCK=y +# CONFIG_FTL is not set +# CONFIG_NFTL is not set + +# +# RAM/ROM/Flash chip drivers +# +CONFIG_MTD_CFI=y +# CONFIG_MTD_JEDECPROBE is not set +CONFIG_MTD_GEN_PROBE=y +# CONFIG_MTD_CFI_ADV_OPTIONS is not set +CONFIG_MTD_CFI_INTELEXT=y +# CONFIG_MTD_CFI_AMDSTD is not set +# CONFIG_MTD_RAM is not set +# CONFIG_MTD_ROM is not set +# CONFIG_MTD_ABSENT is not set +# CONFIG_MTD_OBSOLETE_CHIPS is not set +# CONFIG_MTD_AMDSTD is not set +# CONFIG_MTD_SHARP is not set +# CONFIG_MTD_JEDEC is not set + +# +# Mapping drivers for chip access +# +# CONFIG_MTD_PHYSMAP is not set +# CONFIG_MTD_NORA is not set +# CONFIG_MTD_ARM_INTEGRATOR is not set +# CONFIG_MTD_CDB89712 is not set +# CONFIG_MTD_SA1100 is not set +# CONFIG_MTD_DC21285 is not set +# CONFIG_MTD_IQ80310 is not set +CONFIG_MTD_EPXA10DB=y +# CONFIG_MTD_AUTCPU12 is not set +# CONFIG_MTD_EDB7312 is not set +# CONFIG_MTD_IMPA7 is not set +# CONFIG_MTD_PCI is not set + +# +# Self-contained MTD device drivers +# +# CONFIG_MTD_PMC551 is not set +# CONFIG_MTD_SLRAM is not set +# CONFIG_MTD_MTDRAM is not set +# CONFIG_MTD_BLKMTD is not set +# CONFIG_MTD_DOC1000 is not set +# CONFIG_MTD_DOC2000 is not set +# CONFIG_MTD_DOC2001 is not set +# CONFIG_MTD_DOCPROBE is not set + +# +# NAND Flash Device Drivers +# +# CONFIG_MTD_NAND is not set + +# +# Plug and Play configuration +# +# CONFIG_PNP is not set +# CONFIG_ISAPNP is not set + +# +# Block devices +# +# CONFIG_BLK_DEV_FD is not set +# CONFIG_BLK_DEV_XD is not set +# CONFIG_PARIDE is not set +# CONFIG_BLK_CPQ_DA is not set +# CONFIG_BLK_CPQ_CISS_DA is not set +# CONFIG_BLK_DEV_DAC960 is not set +# CONFIG_BLK_DEV_LOOP is not set +# CONFIG_BLK_DEV_NBD is not set +CONFIG_BLK_DEV_RAM=y +CONFIG_BLK_DEV_RAM_SIZE=8192 +# CONFIG_BLK_DEV_INITRD is not set + +# +# Multi-device support (RAID and LVM) +# +# CONFIG_MD is not set +# CONFIG_BLK_DEV_MD is not set +# CONFIG_MD_LINEAR is not set +# CONFIG_MD_RAID0 is not set +# CONFIG_MD_RAID1 is not set +# CONFIG_MD_RAID5 is not set +# CONFIG_MD_MULTIPATH is not set +# CONFIG_BLK_DEV_LVM is not set + +# +# Networking options +# +CONFIG_PACKET=y +# CONFIG_PACKET_MMAP is not set +# CONFIG_NETLINK_DEV is not set +# CONFIG_NETFILTER is not set +# CONFIG_FILTER is not set +CONFIG_UNIX=y +CONFIG_INET=y +# CONFIG_IP_MULTICAST is not set +# CONFIG_IP_ADVANCED_ROUTER is not set +CONFIG_IP_PNP=y +CONFIG_IP_PNP_DHCP=y +CONFIG_IP_PNP_BOOTP=y +# CONFIG_IP_PNP_RARP is not set +# CONFIG_NET_IPIP is not set +# CONFIG_NET_IPGRE is not set +# CONFIG_ARPD is not set +# CONFIG_INET_ECN is not set +# CONFIG_SYN_COOKIES is not set +# CONFIG_IPV6 is not set +# CONFIG_KHTTPD is not set +# CONFIG_ATM is not set +# CONFIG_VLAN_8021Q is not set +# CONFIG_IPX is not set +# CONFIG_ATALK is not set +# CONFIG_DECNET is not set +# CONFIG_BRIDGE is not set +# CONFIG_X25 is not set +# CONFIG_LAPB is not set +# CONFIG_LLC is not set +# CONFIG_NET_DIVERT is not set +# CONFIG_ECONET is not set +# CONFIG_WAN_ROUTER is not set +# CONFIG_NET_FASTROUTE is not set +# CONFIG_NET_HW_FLOWCONTROL is not set + +# +# QoS and/or fair queueing +# +# CONFIG_NET_SCHED is not set + +# +# Network device support +# +CONFIG_NETDEVICES=y + +# +# ARCnet devices +# +# CONFIG_ARCNET is not set +# CONFIG_DUMMY is not set +# CONFIG_BONDING is not set +# CONFIG_EQUALIZER is not set +# CONFIG_TUN is not set +# CONFIG_ETHERTAP is not set + +# +# Ethernet (10 or 100Mbit) +# +CONFIG_NET_ETHERNET=y +# CONFIG_ARM_AM79C961A is not set +# CONFIG_ETHER00 is not set +# CONFIG_SUNLANCE is not set +# CONFIG_SUNBMAC is not set +# CONFIG_SUNQE is not set +# CONFIG_SUNGEM is not set +# CONFIG_NET_VENDOR_3COM is not set +# CONFIG_LANCE is not set +CONFIG_NET_VENDOR_SMC=y +# CONFIG_WD80x3 is not set +# CONFIG_ULTRAMCA is not set +# CONFIG_ULTRA is not set +# CONFIG_ULTRA32 is not set +# CONFIG_SMC9194 is not set +CONFIG_SMC91111=y +# CONFIG_NET_VENDOR_RACAL is not set +# CONFIG_NET_ISA is not set +# CONFIG_NET_PCI is not set +# CONFIG_NET_POCKET is not set + +# +# Ethernet (1000 Mbit) +# +# CONFIG_ACENIC is not set +# CONFIG_DL2K is not set +# CONFIG_MYRI_SBUS is not set +# CONFIG_NS83820 is not set +# CONFIG_HAMACHI is not set +# CONFIG_YELLOWFIN is not set +# CONFIG_SK98LIN is not set +# CONFIG_FDDI is not set +# CONFIG_HIPPI is not set +# CONFIG_PLIP is not set +CONFIG_PPP=y +CONFIG_PPP_MULTILINK=y +# CONFIG_PPP_FILTER is not set +CONFIG_PPP_ASYNC=y +CONFIG_PPP_SYNC_TTY=y +# CONFIG_PPP_DEFLATE is not set +# CONFIG_PPP_BSDCOMP is not set +# CONFIG_PPPOE is not set +# CONFIG_SLIP is not set + +# +# Wireless LAN (non-hamradio) +# +# CONFIG_NET_RADIO is not set + +# +# Token Ring devices +# +# CONFIG_TR is not set +# CONFIG_NET_FC is not set +# CONFIG_RCPCI is not set +# CONFIG_SHAPER is not set + +# +# Wan interfaces +# +# CONFIG_WAN is not set + +# +# Amateur Radio support +# +# CONFIG_HAMRADIO is not set + +# +# IrDA (infrared) support +# +# CONFIG_IRDA is not set + +# +# ATA/IDE/MFM/RLL support +# +# CONFIG_IDE is not set +# CONFIG_BLK_DEV_IDE_MODES is not set +# CONFIG_BLK_DEV_HD is not set + +# +# SCSI support +# +# CONFIG_SCSI is not set + +# +# I2O device support +# +# CONFIG_I2O is not set +# CONFIG_I2O_BLOCK is not set +# CONFIG_I2O_LAN is not set +# CONFIG_I2O_SCSI is not set +# CONFIG_I2O_PROC is not set + +# +# ISDN subsystem +# +# CONFIG_ISDN is not set + +# +# Input core support +# +# CONFIG_INPUT is not set +# CONFIG_INPUT_KEYBDEV is not set +# CONFIG_INPUT_MOUSEDEV is not set +# CONFIG_INPUT_JOYDEV is not set +# CONFIG_INPUT_EVDEV is not set + +# +# Character devices +# +# CONFIG_VT is not set +# CONFIG_SERIAL is not set +# CONFIG_SERIAL_EXTENDED is not set +# CONFIG_SERIAL_NONSTANDARD is not set + +# +# Serial drivers +# +# CONFIG_SERIAL_ANAKIN is not set +# CONFIG_SERIAL_ANAKIN_CONSOLE is not set +# CONFIG_SERIAL_AMBA is not set +# CONFIG_SERIAL_AMBA_CONSOLE is not set +# CONFIG_SERIAL_CLPS711X is not set +# CONFIG_SERIAL_CLPS711X_CONSOLE is not set +# CONFIG_SERIAL_21285 is not set +# CONFIG_SERIAL_21285_OLD is not set +# CONFIG_SERIAL_21285_CONSOLE is not set +CONFIG_SERIAL_UART00=y +CONFIG_SERIAL_UART00_CONSOLE=y +# CONFIG_SERIAL_SA1100 is not set +# CONFIG_SERIAL_SA1100_CONSOLE is not set +# CONFIG_SERIAL_8250 is not set +# CONFIG_SERIAL_8250_CONSOLE is not set +# CONFIG_SERIAL_8250_EXTENDED is not set +# CONFIG_SERIAL_8250_MANY_PORTS is not set +# CONFIG_SERIAL_8250_SHARE_IRQ is not set +# CONFIG_SERIAL_8250_DETECT_IRQ is not set +# CONFIG_SERIAL_8250_MULTIPORT is not set +# CONFIG_SERIAL_8250_HUB6 is not set +CONFIG_SERIAL_CORE=y +CONFIG_SERIAL_CORE_CONSOLE=y +CONFIG_UNIX98_PTYS=y +CONFIG_UNIX98_PTY_COUNT=256 + +# +# I2C support +# +# CONFIG_I2C is not set + +# +# L3 serial bus support +# +# CONFIG_L3 is not set +# CONFIG_L3_ALGOBIT is not set +# CONFIG_L3_BIT_SA1100_GPIO is not set +# CONFIG_L3_SA1111 is not set +# CONFIG_BIT_SA1100_GPIO is not set + +# +# Mice +# +# CONFIG_BUSMOUSE is not set +# CONFIG_MOUSE is not set + +# +# Joysticks +# +# CONFIG_INPUT_GAMEPORT is not set +# CONFIG_QIC02_TAPE is not set + +# +# Watchdog Cards +# +# CONFIG_WATCHDOG is not set +# CONFIG_INTEL_RNG is not set +# CONFIG_NVRAM is not set +# CONFIG_RTC is not set +# CONFIG_DTLK is not set +# CONFIG_R3964 is not set +# CONFIG_APPLICOM is not set + +# +# Ftape, the floppy tape device driver +# +# CONFIG_FTAPE is not set +# CONFIG_AGP is not set +# CONFIG_DRM is not set + +# +# Multimedia devices +# +# CONFIG_VIDEO_DEV is not set + +# +# File systems +# +# CONFIG_QUOTA is not set +# CONFIG_AUTOFS_FS is not set +# CONFIG_AUTOFS4_FS is not set +# CONFIG_REISERFS_FS is not set +# CONFIG_REISERFS_CHECK is not set +# CONFIG_REISERFS_PROC_INFO is not set +# CONFIG_ADFS_FS is not set +# CONFIG_ADFS_FS_RW is not set +# CONFIG_AFFS_FS is not set +# CONFIG_HFS_FS is not set +# CONFIG_BFS_FS is not set +# CONFIG_EXT3_FS is not set +# CONFIG_JBD is not set +# CONFIG_JBD_DEBUG is not set +# CONFIG_FAT_FS is not set +# CONFIG_MSDOS_FS is not set +# CONFIG_UMSDOS_FS is not set +# CONFIG_VFAT_FS is not set +# CONFIG_EFS_FS is not set +# CONFIG_JFFS_FS is not set +CONFIG_JFFS2_FS=y +CONFIG_JFFS2_FS_DEBUG=0 +# CONFIG_CRAMFS is not set +# CONFIG_TMPFS is not set +# CONFIG_RAMFS is not set +# CONFIG_ISO9660_FS is not set +# CONFIG_JOLIET is not set +# CONFIG_ZISOFS is not set +# CONFIG_MINIX_FS is not set +# CONFIG_VXFS_FS is not set +# CONFIG_NTFS_FS is not set +# CONFIG_NTFS_RW is not set +# CONFIG_HPFS_FS is not set +CONFIG_PROC_FS=y +# CONFIG_DEVFS_FS is not set +# CONFIG_DEVFS_MOUNT is not set +# CONFIG_DEVFS_DEBUG is not set +CONFIG_DEVPTS_FS=y +# CONFIG_QNX4FS_FS is not set +# CONFIG_QNX4FS_RW is not set +CONFIG_ROMFS_FS=y +CONFIG_EXT2_FS=y +# CONFIG_SYSV_FS is not set +# CONFIG_UDF_FS is not set +# CONFIG_UDF_RW is not set +# CONFIG_UFS_FS is not set +# CONFIG_UFS_FS_WRITE is not set + +# +# Network File Systems +# +# CONFIG_CODA_FS is not set +# CONFIG_INTERMEZZO_FS is not set +CONFIG_NFS_FS=y +# CONFIG_NFS_V3 is not set +# CONFIG_ROOT_NFS is not set +# CONFIG_NFSD is not set +# CONFIG_NFSD_V3 is not set +CONFIG_SUNRPC=y +CONFIG_LOCKD=y +# CONFIG_SMB_FS is not set +# CONFIG_NCP_FS is not set +# CONFIG_NCPFS_PACKET_SIGNING is not set +# CONFIG_NCPFS_IOCTL_LOCKING is not set +# CONFIG_NCPFS_STRONG is not set +# CONFIG_NCPFS_NFS_NS is not set +# CONFIG_NCPFS_OS2_NS is not set +# CONFIG_NCPFS_SMALLDOS is not set +# CONFIG_NCPFS_NLS is not set +# CONFIG_NCPFS_EXTRAS is not set +# CONFIG_ZISOFS_FS is not set +# CONFIG_ZLIB_FS_INFLATE is not set + +# +# Partition Types +# +# CONFIG_PARTITION_ADVANCED is not set +CONFIG_MSDOS_PARTITION=y +# CONFIG_SMB_NLS is not set +# CONFIG_NLS is not set + +# +# Multimedia Capabilities Port drivers +# +# CONFIG_MCP is not set +# CONFIG_MCP_SA1100 is not set +# CONFIG_MCP_UCB1200 is not set +# CONFIG_MCP_UCB1200_AUDIO is not set +# CONFIG_MCP_UCB1200_TS is not set + +# +# USB support +# +# CONFIG_USB is not set +# CONFIG_USB_UHCI is not set +# CONFIG_USB_UHCI_ALT is not set +# CONFIG_USB_OHCI is not set +# CONFIG_USB_OHCI_SA1111 is not set +# CONFIG_USB_AUDIO is not set +# CONFIG_USB_BLUETOOTH is not set +# CONFIG_USB_STORAGE is not set +# CONFIG_USB_STORAGE_DEBUG is not set +# CONFIG_USB_STORAGE_DATAFAB is not set +# CONFIG_USB_STORAGE_FREECOM is not set +# CONFIG_USB_STORAGE_ISD200 is not set +# CONFIG_USB_STORAGE_DPCM is not set +# CONFIG_USB_STORAGE_HP8200e is not set +# CONFIG_USB_STORAGE_SDDR09 is not set +# CONFIG_USB_STORAGE_JUMPSHOT is not set +# CONFIG_USB_ACM is not set +# CONFIG_USB_PRINTER is not set +# CONFIG_USB_DC2XX is not set +# CONFIG_USB_MDC800 is not set +# CONFIG_USB_SCANNER is not set +# CONFIG_USB_MICROTEK is not set +# CONFIG_USB_HPUSBSCSI is not set +# CONFIG_USB_PEGASUS is not set +# CONFIG_USB_KAWETH is not set +# CONFIG_USB_CATC is not set +# CONFIG_USB_CDCETHER is not set +# CONFIG_USB_USBNET is not set +# CONFIG_USB_USS720 is not set + +# +# USB Serial Converter support +# +# CONFIG_USB_SERIAL is not set +# CONFIG_USB_SERIAL_GENERIC is not set +# CONFIG_USB_SERIAL_BELKIN is not set +# CONFIG_USB_SERIAL_WHITEHEAT is not set +# CONFIG_USB_SERIAL_DIGI_ACCELEPORT is not set +# CONFIG_USB_SERIAL_EMPEG is not set +# CONFIG_USB_SERIAL_FTDI_SIO is not set +# CONFIG_USB_SERIAL_VISOR is not set +# CONFIG_USB_SERIAL_IPAQ is not set +# CONFIG_USB_SERIAL_IR is not set +# CONFIG_USB_SERIAL_EDGEPORT is not set +# CONFIG_USB_SERIAL_KEYSPAN_PDA is not set +# CONFIG_USB_SERIAL_KEYSPAN is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XA is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XB is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA18X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19W is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA49W is not set +# CONFIG_USB_SERIAL_MCT_U232 is not set +# CONFIG_USB_SERIAL_KLSI is not set +# CONFIG_USB_SERIAL_PL2303 is not set +# CONFIG_USB_SERIAL_CYBERJACK is not set +# CONFIG_USB_SERIAL_XIRCOM is not set +# CONFIG_USB_SERIAL_OMNINET is not set +# CONFIG_USB_RIO500 is not set + +# +# Bluetooth support +# +# CONFIG_BLUEZ is not set + +# +# Kernel hacking +# +CONFIG_FRAME_POINTER=y +CONFIG_DEBUG_USER=y +CONFIG_DEBUG_INFO=y +# CONFIG_NO_PGT_CACHE is not set +CONFIG_DEBUG_KERNEL=y +CONFIG_DEBUG_SLAB=y +CONFIG_MAGIC_SYSRQ=y +CONFIG_DEBUG_SPINLOCK=y +CONFIG_DEBUG_WAITQ=y +CONFIG_DEBUG_BUGVERBOSE=y +CONFIG_DEBUG_ERRORS=y +CONFIG_DEBUG_LL=y +# CONFIG_DEBUG_DC21285_PORT is not set +# CONFIG_DEBUG_CLPS711X_UART2 is not set diff -urN orig/arch/arm/def-configs/frodo linux/arch/arm/def-configs/frodo --- orig/arch/arm/def-configs/frodo Mon Aug 5 13:29:43 2002 +++ linux/arch/arm/def-configs/frodo Thu Oct 24 13:01:38 2002 @@ -37,6 +37,7 @@ # CONFIG_ARCH_FOOTBRIDGE is not set # CONFIG_ARCH_INTEGRATOR is not set # CONFIG_ARCH_L7200 is not set +# CONFIG_ARCH_MX1ADS is not set # CONFIG_ARCH_RPC is not set CONFIG_ARCH_SA1100=y # CONFIG_ARCH_SHARK is not set @@ -59,10 +60,12 @@ # # SA11x0 Implementations # +# CONFIG_SA1100_ACCELENT is not set # CONFIG_SA1100_ASSABET is not set # CONFIG_ASSABET_NEPONSET is not set # CONFIG_SA1100_ADSBITSY is not set # CONFIG_SA1100_BRUTUS is not set +# CONFIG_SA1100_CEP is not set # CONFIG_SA1100_CERF is not set # CONFIG_SA1100_H3100 is not set # CONFIG_SA1100_H3600 is not set @@ -71,6 +74,7 @@ # CONFIG_SA1100_EXTENEX1 is not set # CONFIG_SA1100_FLEXANET is not set # CONFIG_SA1100_FREEBIRD is not set +CONFIG_SA1100_FRODO=y # CONFIG_SA1100_GRAPHICSCLIENT is not set # CONFIG_SA1100_GRAPHICSMASTER is not set # CONFIG_SA1100_BADGE4 is not set @@ -86,11 +90,11 @@ # CONFIG_SA1100_SHANNON is not set # CONFIG_SA1100_SHERMAN is not set # CONFIG_SA1100_SIMPAD is not set +# CONFIG_SA1100_SIMPUTER is not set # CONFIG_SA1100_PFS168 is not set # CONFIG_SA1100_VICTOR is not set # CONFIG_SA1100_XP860 is not set # CONFIG_SA1100_YOPY is not set -CONFIG_SA1100_FRODO=y # CONFIG_SA1100_USB is not set # CONFIG_SA1100_USB_NETLINK is not set # CONFIG_SA1100_USB_CHAR is not set @@ -113,18 +117,18 @@ # CONFIG_FOOTBRIDGE_ADDIN is not set CONFIG_CPU_32=y # CONFIG_CPU_26 is not set -# CONFIG_CPU_32v3 is not set -CONFIG_CPU_32v4=y # CONFIG_CPU_ARM610 is not set # CONFIG_CPU_ARM710 is not set # CONFIG_CPU_ARM720T is not set # CONFIG_CPU_ARM920T is not set # CONFIG_CPU_ARM922T is not set +# CONFIG_PLD is not set # CONFIG_CPU_ARM926T is not set # CONFIG_CPU_ARM1020 is not set # CONFIG_CPU_SA110 is not set CONFIG_CPU_SA1100=y -# CONFIG_ARM_THUMB is not set +# CONFIG_CPU_32v3 is not set +CONFIG_CPU_32v4=y CONFIG_DISCONTIGMEM=y # @@ -133,15 +137,28 @@ # CONFIG_PCI is not set CONFIG_ISA=y # CONFIG_ISA_DMA is not set +# CONFIG_ZBOOT_ROM is not set +CONFIG_ZBOOT_ROM_TEXT=0 +CONFIG_ZBOOT_ROM_BSS=0 # CONFIG_CPU_FREQ is not set -# CONFIG_HOTPLUG is not set -# CONFIG_PCMCIA is not set +CONFIG_HOTPLUG=y + +# +# PCMCIA/CardBus support +# +CONFIG_PCMCIA=y +CONFIG_PCMCIA_PROBE=y +# CONFIG_I82092 is not set +# CONFIG_I82365 is not set +# CONFIG_TCIC is not set +# CONFIG_PCMCIA_CLPS6700 is not set +CONFIG_PCMCIA_SA1100=y CONFIG_NET=y CONFIG_SYSVIPC=y # CONFIG_BSD_PROCESS_ACCT is not set CONFIG_SYSCTL=y -# CONFIG_FPE_NWFPE is not set -CONFIG_FPE_FASTFPE=y +CONFIG_FPE_NWFPE=y +# CONFIG_FPE_FASTFPE is not set CONFIG_KCORE_ELF=y # CONFIG_KCORE_AOUT is not set # CONFIG_BINFMT_AOUT is not set @@ -149,7 +166,7 @@ # CONFIG_BINFMT_MISC is not set # CONFIG_PM is not set # CONFIG_ARTHUR is not set -CONFIG_CMDLINE="console=ttySA0,9600 ramdisk_size=4096 root=/dev/ram idebus=33" +CONFIG_CMDLINE="console=ttySA0,115200 ramdisk_size=12288 root=/dev/ram idebus=33 mem=63M" CONFIG_LEDS=y CONFIG_LEDS_TIMER=y CONFIG_LEDS_CPU=y @@ -163,7 +180,68 @@ # # Memory Technology Devices (MTD) # -# CONFIG_MTD is not set +CONFIG_MTD=y +# CONFIG_MTD_DEBUG is not set +CONFIG_MTD_PARTITIONS=y +# CONFIG_MTD_CONCAT is not set +# CONFIG_MTD_REDBOOT_PARTS is not set +# CONFIG_MTD_CMDLINE_PARTS is not set +# CONFIG_MTD_AFS_PARTS is not set +CONFIG_MTD_CHAR=y +CONFIG_MTD_BLOCK=y +# CONFIG_FTL is not set +# CONFIG_NFTL is not set + +# +# RAM/ROM/Flash chip drivers +# +CONFIG_MTD_CFI=y +# CONFIG_MTD_JEDECPROBE is not set +CONFIG_MTD_GEN_PROBE=y +# CONFIG_MTD_CFI_ADV_OPTIONS is not set +CONFIG_MTD_CFI_INTELEXT=y +# CONFIG_MTD_CFI_AMDSTD is not set +# CONFIG_MTD_RAM is not set +# CONFIG_MTD_ROM is not set +# CONFIG_MTD_ABSENT is not set +# CONFIG_MTD_OBSOLETE_CHIPS is not set +# CONFIG_MTD_AMDSTD is not set +# CONFIG_MTD_SHARP is not set +# CONFIG_MTD_JEDEC is not set + +# +# Mapping drivers for chip access +# +# CONFIG_MTD_PHYSMAP is not set +# CONFIG_MTD_NORA is not set +# CONFIG_MTD_ARM_INTEGRATOR is not set +# CONFIG_MTD_CDB89712 is not set +CONFIG_MTD_SA1100=y +# CONFIG_MTD_DC21285 is not set +# CONFIG_MTD_IQ80310 is not set +# CONFIG_MTD_FORTUNET is not set +# CONFIG_MTD_EPXA10DB is not set +# CONFIG_MTD_AUTCPU12 is not set +# CONFIG_MTD_EDB7312 is not set +# CONFIG_MTD_IMPA7 is not set +# CONFIG_MTD_PCI is not set + +# +# Self-contained MTD device drivers +# +# CONFIG_MTD_PMC551 is not set +# CONFIG_MTD_SLRAM is not set +# CONFIG_MTD_MTDRAM is not set +# CONFIG_MTD_BLKMTD is not set +# CONFIG_MTD_DOC1000 is not set +# CONFIG_MTD_DOC2000 is not set +# CONFIG_MTD_DOC2001 is not set +# CONFIG_MTD_DOCPROBE is not set + +# +# NAND Flash Device Drivers +# +# CONFIG_MTD_NAND is not set # # Plug and Play configuration @@ -179,11 +257,13 @@ # CONFIG_PARIDE is not set # CONFIG_BLK_CPQ_DA is not set # CONFIG_BLK_CPQ_CISS_DA is not set +# CONFIG_CISS_SCSI_TAPE is not set # CONFIG_BLK_DEV_DAC960 is not set +# CONFIG_BLK_DEV_UMEM is not set # CONFIG_BLK_DEV_LOOP is not set # CONFIG_BLK_DEV_NBD is not set CONFIG_BLK_DEV_RAM=y -CONFIG_BLK_DEV_RAM_SIZE=4096 +CONFIG_BLK_DEV_RAM_SIZE=12288 CONFIG_BLK_DEV_INITRD=y # @@ -201,7 +281,8 @@ # # Networking options # -# CONFIG_PACKET is not set +CONFIG_PACKET=y +CONFIG_PACKET_MMAP=y # CONFIG_NETLINK_DEV is not set # CONFIG_NETFILTER is not set # CONFIG_FILTER is not set @@ -221,6 +302,11 @@ # CONFIG_VLAN_8021Q is not set # CONFIG_IPX is not set # CONFIG_ATALK is not set + +# +# Appletalk devices +# +# CONFIG_DEV_APPLETALK is not set # CONFIG_DECNET is not set # CONFIG_BRIDGE is not set # CONFIG_X25 is not set @@ -238,9 +324,98 @@ # CONFIG_NET_SCHED is not set # +# Network testing +# +# CONFIG_NET_PKTGEN is not set + +# # Network device support # -# CONFIG_NETDEVICES is not set +CONFIG_NETDEVICES=y + +# +# ARCnet devices +# +# CONFIG_ARCNET is not set +# CONFIG_DUMMY is not set +# CONFIG_BONDING is not set +# CONFIG_EQUALIZER is not set +# CONFIG_TUN is not set +# CONFIG_ETHERTAP is not set + +# +# Ethernet (10 or 100Mbit) +# +CONFIG_NET_ETHERNET=y +CONFIG_CIRRUS=y +# CONFIG_CIRRUS_DEBUG is not set +# CONFIG_CIRRUS_DUPLEX is not set +# CONFIG_ARM_AM79C961A is not set +# CONFIG_SUNLANCE is not set +# CONFIG_SUNBMAC is not set +# CONFIG_SUNQE is not set +# CONFIG_SUNGEM is not set +# CONFIG_NET_VENDOR_3COM is not set +# CONFIG_LANCE is not set +# CONFIG_NET_VENDOR_SMC is not set +# CONFIG_NET_VENDOR_RACAL is not set +# CONFIG_AT1700 is not set +# CONFIG_DEPCA is not set +# CONFIG_HP100 is not set +# CONFIG_NET_ISA is not set +# CONFIG_NET_PCI is not set +# CONFIG_NET_POCKET is not set + +# +# Ethernet (1000 Mbit) +# +# CONFIG_ACENIC is not set +# CONFIG_DL2K is not set +# CONFIG_MYRI_SBUS is not set +# CONFIG_NS83820 is not set +# CONFIG_HAMACHI is not set +# CONFIG_YELLOWFIN is not set +# CONFIG_SK98LIN is not set +# CONFIG_TIGON3 is not set +# CONFIG_FDDI is not set +# CONFIG_HIPPI is not set +# CONFIG_PLIP is not set +# CONFIG_PPP is not set +# CONFIG_SLIP is not set + +# +# Wireless LAN (non-hamradio) +# +# CONFIG_NET_RADIO is not set + +# +# Token Ring devices +# +# CONFIG_TR is not set +# CONFIG_NET_FC is not set +# CONFIG_RCPCI is not set +# CONFIG_SHAPER is not set + +# +# Wan interfaces +# +# CONFIG_WAN is not set + +# +# PCMCIA network device support +# +CONFIG_NET_PCMCIA=y +CONFIG_PCMCIA_3C589=y +# CONFIG_PCMCIA_3C574 is not set +# CONFIG_PCMCIA_FMVJ18X is not set +# CONFIG_PCMCIA_PCNET is not set +# CONFIG_PCMCIA_AXNET is not set +# CONFIG_PCMCIA_NMCLAN is not set +# CONFIG_PCMCIA_SMC91C92 is not set +# CONFIG_PCMCIA_XIRC2PS is not set +# CONFIG_ARCNET_COM20020_CS is not set +# CONFIG_PCMCIA_IBMTR is not set +# CONFIG_NET_PCMCIA_RADIO is not set # # Amateur Radio support @@ -253,7 +428,7 @@ # CONFIG_IRDA is not set # -# ATA/IDE/MFM/RLL support +# ATA/ATAPI/MFM/RLL support # CONFIG_IDE=y @@ -265,6 +440,7 @@ # CONFIG_BLK_DEV_HD is not set CONFIG_BLK_DEV_IDEDISK=y # CONFIG_IDEDISK_MULTI_MODE is not set +# CONFIG_IDEDISK_STROKE is not set # CONFIG_BLK_DEV_IDEDISK_VENDOR is not set # CONFIG_BLK_DEV_IDEDISK_FUJITSU is not set # CONFIG_BLK_DEV_IDEDISK_IBM is not set @@ -274,11 +450,12 @@ # CONFIG_BLK_DEV_IDEDISK_WD is not set # CONFIG_BLK_DEV_COMMERIAL is not set # CONFIG_BLK_DEV_TIVO is not set -# CONFIG_BLK_DEV_IDECS is not set +CONFIG_BLK_DEV_IDECS=y CONFIG_BLK_DEV_IDECD=y # CONFIG_BLK_DEV_IDETAPE is not set # CONFIG_BLK_DEV_IDEFLOPPY is not set # CONFIG_BLK_DEV_IDESCSI is not set +# CONFIG_IDE_TASK_IOCTL is not set # CONFIG_BLK_DEV_CMD640 is not set # CONFIG_BLK_DEV_CMD640_ENHANCED is not set # CONFIG_BLK_DEV_ISAPNP is not set @@ -312,11 +489,13 @@ # # Input core support # -# CONFIG_INPUT is not set -# CONFIG_INPUT_KEYBDEV is not set -# CONFIG_INPUT_MOUSEDEV is not set +CONFIG_INPUT=y +CONFIG_INPUT_KEYBDEV=y +CONFIG_INPUT_MOUSEDEV=y +CONFIG_INPUT_MOUSEDEV_SCREEN_X=640 +CONFIG_INPUT_MOUSEDEV_SCREEN_Y=480 # CONFIG_INPUT_JOYDEV is not set -# CONFIG_INPUT_EVDEV is not set +CONFIG_INPUT_EVDEV=y # # Character devices @@ -343,7 +522,7 @@ # CONFIG_SERIAL_UART00_CONSOLE is not set CONFIG_SERIAL_SA1100=y CONFIG_SERIAL_SA1100_CONSOLE=y -CONFIG_SA1100_DEFAULT_BAUDRATE=9600 +CONFIG_SA1100_DEFAULT_BAUDRATE=115200 # CONFIG_SERIAL_8250 is not set # CONFIG_SERIAL_8250_CONSOLE is not set # CONFIG_SERIAL_8250_EXTENDED is not set @@ -355,12 +534,22 @@ CONFIG_SERIAL_CORE=y CONFIG_SERIAL_CORE_CONSOLE=y CONFIG_UNIX98_PTYS=y -CONFIG_UNIX98_PTY_COUNT=16 +CONFIG_UNIX98_PTY_COUNT=64 # # I2C support # -# CONFIG_I2C is not set +CONFIG_I2C=y +CONFIG_I2C_ALGOBIT=y +# CONFIG_I2C_PHILIPSPAR is not set +# CONFIG_I2C_ELV is not set +# CONFIG_I2C_VELLEMAN is not set +CONFIG_I2C_FRODO=y +# CONFIG_I2C_BIT_SA1100_GPIO is not set +# CONFIG_I2C_ALGOPCF is not set +CONFIG_I2C_CHARDEV=y +CONFIG_I2C_PROC=y +CONFIG_I2C_DS1307=y # # L3 serial bus support @@ -381,16 +570,61 @@ # Joysticks # # CONFIG_INPUT_GAMEPORT is not set +# CONFIG_INPUT_NS558 is not set +# CONFIG_INPUT_LIGHTNING is not set +# CONFIG_INPUT_PCIGAME is not set +# CONFIG_INPUT_CS461X is not set +# CONFIG_INPUT_EMU10K1 is not set +# CONFIG_INPUT_SERIO is not set +# CONFIG_INPUT_SERPORT is not set +# CONFIG_INPUT_ANALOG is not set +# CONFIG_INPUT_A3D is not set +# CONFIG_INPUT_ADI is not set +# CONFIG_INPUT_COBRA is not set +# CONFIG_INPUT_GF2K is not set +# CONFIG_INPUT_GRIP is not set +# CONFIG_INPUT_INTERACT is not set +# CONFIG_INPUT_TMDC is not set +# CONFIG_INPUT_SIDEWINDER is not set +# CONFIG_INPUT_IFORCE_USB is not set +# CONFIG_INPUT_IFORCE_232 is not set +# CONFIG_INPUT_WARRIOR is not set +# CONFIG_INPUT_MAGELLAN is not set +# CONFIG_INPUT_SPACEORB is not set +# CONFIG_INPUT_SPACEBALL is not set +# CONFIG_INPUT_STINGER is not set +# CONFIG_INPUT_DB9 is not set +# CONFIG_INPUT_GAMECON is not set +# CONFIG_INPUT_TURBOGRAFX is not set # CONFIG_QIC02_TAPE is not set # # Watchdog Cards # -# CONFIG_WATCHDOG is not set -# CONFIG_INTEL_RNG is not set -# CONFIG_NVRAM is not set -# CONFIG_RTC is not set -CONFIG_SA1100_RTC=y +CONFIG_WATCHDOG=y +# CONFIG_WATCHDOG_NOWAYOUT is not set +# CONFIG_ACQUIRE_WDT is not set +# CONFIG_ADVANTECH_WDT is not set +# CONFIG_ALIM7101_WDT is not set +# CONFIG_SC520_WDT is not set +# CONFIG_PCWATCHDOG is not set +# CONFIG_21285_WATCHDOG is not set +# CONFIG_977_WATCHDOG is not set +CONFIG_SA1100_WATCHDOG=y +# CONFIG_EUROTECH_WDT is not set +# CONFIG_IB700_WDT is not set +# CONFIG_WAFER_WDT is not set +# CONFIG_I810_TCO is not set +# CONFIG_MIXCOMWD is not set +# CONFIG_60XX_WDT is not set +# CONFIG_SC1200_WDT is not set +# CONFIG_SOFT_WATCHDOG is not set +# CONFIG_W83877F_WDT is not set +# CONFIG_WDT is not set +# CONFIG_WDTPCI is not set +# CONFIG_MACHZ_WDT is not set +CONFIG_DS1307_RTC=y +CONFIG_DS1307_NVRAM=y # CONFIG_DTLK is not set # CONFIG_R3964 is not set # CONFIG_APPLICOM is not set @@ -403,9 +637,50 @@ # CONFIG_DRM is not set # +# PCMCIA character devices +# + +# # Multimedia devices # -# CONFIG_VIDEO_DEV is not set +CONFIG_VIDEO_DEV=y + +# +# Video For Linux +# +# CONFIG_I2C_PARPORT is not set +# CONFIG_VIDEO_BT848 is not set +# CONFIG_VIDEO_PMS is not set +# CONFIG_VIDEO_CPIA is not set +# CONFIG_VIDEO_SAA5249 is not set +# CONFIG_TUNER_3036 is not set +# CONFIG_VIDEO_STRADIS is not set +# CONFIG_VIDEO_ZORAN is not set +# CONFIG_VIDEO_ZORAN_BUZ is not set +# CONFIG_VIDEO_ZORAN_DC10 is not set +# CONFIG_VIDEO_ZORAN_LML33 is not set +# CONFIG_VIDEO_ZR36120 is not set +# CONFIG_VIDEO_MEYE is not set +# CONFIG_VIDEO_CYBERPRO is not set + +# +# Radio Adapters +# +# CONFIG_RADIO_CADET is not set +# CONFIG_RADIO_RTRACK is not set +# CONFIG_RADIO_RTRACK2 is not set +# CONFIG_RADIO_AZTECH is not set +# CONFIG_RADIO_GEMTEK is not set +# CONFIG_RADIO_GEMTEK_PCI is not set +# CONFIG_RADIO_MAXIRADIO is not set +# CONFIG_RADIO_MAESTRO is not set +# CONFIG_RADIO_MIROPCM20 is not set +# CONFIG_RADIO_MIROPCM20_RDS is not set +# CONFIG_RADIO_SF16FMI is not set +# CONFIG_RADIO_TERRATEC is not set +# CONFIG_RADIO_TRUST is not set +# CONFIG_RADIO_TYPHOON is not set +# CONFIG_RADIO_ZOLTRIX is not set # # File systems @@ -424,32 +699,33 @@ # CONFIG_EXT3_FS is not set # CONFIG_JBD is not set # CONFIG_JBD_DEBUG is not set -# CONFIG_FAT_FS is not set -# CONFIG_MSDOS_FS is not set +CONFIG_FAT_FS=y +CONFIG_MSDOS_FS=y # CONFIG_UMSDOS_FS is not set -# CONFIG_VFAT_FS is not set +CONFIG_VFAT_FS=y # CONFIG_EFS_FS is not set # CONFIG_JFFS_FS is not set -# CONFIG_JFFS2_FS is not set +CONFIG_JFFS2_FS=y +CONFIG_JFFS2_FS_DEBUG=0 # CONFIG_CRAMFS is not set # CONFIG_TMPFS is not set -# CONFIG_RAMFS is not set -# CONFIG_ISO9660_FS is not set -# CONFIG_JOLIET is not set -# CONFIG_ZISOFS is not set +CONFIG_RAMFS=y +CONFIG_ISO9660_FS=y +CONFIG_JOLIET=y +CONFIG_ZISOFS=y # CONFIG_MINIX_FS is not set # CONFIG_VXFS_FS is not set # CONFIG_NTFS_FS is not set # CONFIG_NTFS_RW is not set # CONFIG_HPFS_FS is not set CONFIG_PROC_FS=y -# CONFIG_DEVFS_FS is not set -# CONFIG_DEVFS_MOUNT is not set +CONFIG_DEVFS_FS=y +CONFIG_DEVFS_MOUNT=y # CONFIG_DEVFS_DEBUG is not set CONFIG_DEVPTS_FS=y # CONFIG_QNX4FS_FS is not set # CONFIG_QNX4FS_RW is not set -CONFIG_ROMFS_FS=y +# CONFIG_ROMFS_FS is not set CONFIG_EXT2_FS=y # CONFIG_SYSV_FS is not set # CONFIG_UDF_FS is not set @@ -479,8 +755,8 @@ # CONFIG_NCPFS_SMALLDOS is not set # CONFIG_NCPFS_NLS is not set # CONFIG_NCPFS_EXTRAS is not set -# CONFIG_ZISOFS_FS is not set -# CONFIG_ZLIB_FS_INFLATE is not set +CONFIG_ZISOFS_FS=y +CONFIG_ZLIB_FS_INFLATE=y # # Partition Types @@ -488,11 +764,54 @@ # CONFIG_PARTITION_ADVANCED is not set CONFIG_MSDOS_PARTITION=y # CONFIG_SMB_NLS is not set -# CONFIG_NLS is not set +CONFIG_NLS=y + +# +# Native Language Support +# +CONFIG_NLS_DEFAULT="iso8859-1" +# CONFIG_NLS_CODEPAGE_437 is not set +# CONFIG_NLS_CODEPAGE_737 is not set +# CONFIG_NLS_CODEPAGE_775 is not set +# CONFIG_NLS_CODEPAGE_850 is not set +# CONFIG_NLS_CODEPAGE_852 is not set +# CONFIG_NLS_CODEPAGE_855 is not set +# CONFIG_NLS_CODEPAGE_857 is not set +# CONFIG_NLS_CODEPAGE_860 is not set +# CONFIG_NLS_CODEPAGE_861 is not set +# CONFIG_NLS_CODEPAGE_862 is not set +# CONFIG_NLS_CODEPAGE_863 is not set +# CONFIG_NLS_CODEPAGE_864 is not set +# CONFIG_NLS_CODEPAGE_865 is not set +# CONFIG_NLS_CODEPAGE_866 is not set +# CONFIG_NLS_CODEPAGE_869 is not set +# CONFIG_NLS_CODEPAGE_936 is not set +# CONFIG_NLS_CODEPAGE_950 is not set +# CONFIG_NLS_CODEPAGE_932 is not set +# CONFIG_NLS_CODEPAGE_949 is not set +# CONFIG_NLS_CODEPAGE_874 is not set +# CONFIG_NLS_ISO8859_8 is not set +# CONFIG_NLS_CODEPAGE_1250 is not set +# CONFIG_NLS_CODEPAGE_1251 is not set +# CONFIG_NLS_ISO8859_1 is not set +# CONFIG_NLS_ISO8859_2 is not set +# CONFIG_NLS_ISO8859_3 is not set +# CONFIG_NLS_ISO8859_4 is not set +# CONFIG_NLS_ISO8859_5 is not set +# CONFIG_NLS_ISO8859_6 is not set +# CONFIG_NLS_ISO8859_7 is not set +# CONFIG_NLS_ISO8859_9 is not set +# CONFIG_NLS_ISO8859_13 is not set +# CONFIG_NLS_ISO8859_14 is not set +# CONFIG_NLS_ISO8859_15 is not set +# CONFIG_NLS_KOI8_R is not set +# CONFIG_NLS_KOI8_U is not set +# CONFIG_NLS_UTF8 is not set # # Console drivers # +CONFIG_PC_KEYB=y CONFIG_PC_KEYMAP=y # CONFIG_VGA_CONSOLE is not set @@ -507,15 +826,32 @@ CONFIG_FB_SA1100=y # CONFIG_FB_CYBER2000 is not set # CONFIG_FB_VIRTUAL is not set -# CONFIG_FBCON_ADVANCED is not set -CONFIG_FBCON_CFB2=y -CONFIG_FBCON_CFB4=y -CONFIG_FBCON_CFB8=y +CONFIG_FBCON_ADVANCED=y +# CONFIG_FBCON_MFB is not set +# CONFIG_FBCON_CFB2 is not set +# CONFIG_FBCON_CFB4 is not set +# CONFIG_FBCON_CFB8 is not set CONFIG_FBCON_CFB16=y +# CONFIG_FBCON_CFB24 is not set +# CONFIG_FBCON_CFB32 is not set +# CONFIG_FBCON_AFB is not set +# CONFIG_FBCON_ILBM is not set +# CONFIG_FBCON_IPLAN2P2 is not set +# CONFIG_FBCON_IPLAN2P4 is not set +# CONFIG_FBCON_IPLAN2P8 is not set +# CONFIG_FBCON_MAC is not set +# CONFIG_FBCON_VGA_PLANES is not set +# CONFIG_FBCON_VGA is not set +# CONFIG_FBCON_HGA is not set # CONFIG_FBCON_FONTWIDTH8_ONLY is not set -# CONFIG_FBCON_FONTS is not set -CONFIG_FONT_8x8=y +CONFIG_FBCON_FONTS=y +# CONFIG_FONT_8x8 is not set CONFIG_FONT_8x16=y +# CONFIG_FONT_SUN8x16 is not set +# CONFIG_FONT_SUN12x22 is not set +# CONFIG_FONT_6x11 is not set +# CONFIG_FONT_PEARL_8x8 is not set +# CONFIG_FONT_ACORN_8x8 is not set # # Sound @@ -534,12 +870,19 @@ # # USB support # -# CONFIG_USB is not set +CONFIG_USB=y +# CONFIG_USB_DEBUG is not set +CONFIG_USB_DEVICEFS=y +# CONFIG_USB_BANDWIDTH is not set +# CONFIG_USB_LONG_TIMEOUT is not set +# CONFIG_USB_EHCI_HCD is not set # CONFIG_USB_UHCI is not set # CONFIG_USB_UHCI_ALT is not set # CONFIG_USB_OHCI is not set # CONFIG_USB_OHCI_SA1111 is not set +CONFIG_USB_1161=y # CONFIG_USB_AUDIO is not set +# CONFIG_USB_EMI26 is not set # CONFIG_USB_BLUETOOTH is not set # CONFIG_USB_STORAGE is not set # CONFIG_USB_STORAGE_DEBUG is not set @@ -552,12 +895,25 @@ # CONFIG_USB_STORAGE_JUMPSHOT is not set # CONFIG_USB_ACM is not set # CONFIG_USB_PRINTER is not set +CONFIG_USB_HID=y +CONFIG_USB_HIDINPUT=y +CONFIG_USB_HIDDEV=y +# CONFIG_USB_WACOM is not set # CONFIG_USB_DC2XX is not set # CONFIG_USB_MDC800 is not set # CONFIG_USB_SCANNER is not set # CONFIG_USB_MICROTEK is not set # CONFIG_USB_HPUSBSCSI is not set +# CONFIG_USB_IBMCAM is not set +# CONFIG_USB_OV511 is not set +# CONFIG_USB_PWC is not set +# CONFIG_USB_SE401 is not set +# CONFIG_USB_STV680 is not set +# CONFIG_USB_VICAM is not set +# CONFIG_USB_DSBR is not set +# CONFIG_USB_DABUSB is not set # CONFIG_USB_PEGASUS is not set +# CONFIG_USB_RTL8150 is not set # CONFIG_USB_KAWETH is not set # CONFIG_USB_CATC is not set # CONFIG_USB_CDCETHER is not set @@ -575,6 +931,7 @@ # CONFIG_USB_SERIAL_EMPEG is not set # CONFIG_USB_SERIAL_FTDI_SIO is not set # CONFIG_USB_SERIAL_VISOR is not set +# CONFIG_USB_SERIAL_IPAQ is not set # CONFIG_USB_SERIAL_IR is not set # CONFIG_USB_SERIAL_EDGEPORT is not set # CONFIG_USB_SERIAL_KEYSPAN_PDA is not set @@ -588,11 +945,14 @@ # CONFIG_USB_SERIAL_KEYSPAN_USA19W is not set # CONFIG_USB_SERIAL_KEYSPAN_USA49W is not set # CONFIG_USB_SERIAL_MCT_U232 is not set +# CONFIG_USB_SERIAL_KLSI is not set # CONFIG_USB_SERIAL_PL2303 is not set # CONFIG_USB_SERIAL_CYBERJACK is not set # CONFIG_USB_SERIAL_XIRCOM is not set # CONFIG_USB_SERIAL_OMNINET is not set # CONFIG_USB_RIO500 is not set +# CONFIG_USB_AUERSWALD is not set +# CONFIG_USB_BRLVGER is not set # # Bluetooth support diff -urN orig/arch/arm/def-configs/graphicsclient linux/arch/arm/def-configs/graphicsclient --- orig/arch/arm/def-configs/graphicsclient Mon Aug 5 13:29:43 2002 +++ linux/arch/arm/def-configs/graphicsclient Fri Feb 21 15:17:34 2003 @@ -8,6 +8,8 @@ CONFIG_UID16=y CONFIG_RWSEM_GENERIC_SPINLOCK=y # CONFIG_RWSEM_XCHGADD_ALGORITHM is not set +# CONFIG_GENERIC_BUST_SPINLOCK is not set +# CONFIG_GENERIC_ISA_DMA is not set # # Code maturity level options @@ -31,12 +33,17 @@ # CONFIG_ARCH_CLPS711X is not set # CONFIG_ARCH_CO285 is not set # CONFIG_ARCH_EBSA110 is not set +# CONFIG_ARCH_CAMELOT is not set # CONFIG_ARCH_FOOTBRIDGE is not set # CONFIG_ARCH_INTEGRATOR is not set +# CONFIG_ARCH_OMAHA is not set # CONFIG_ARCH_L7200 is not set +# CONFIG_ARCH_MX1ADS is not set # CONFIG_ARCH_RPC is not set +# CONFIG_ARCH_RISCSTATION is not set CONFIG_ARCH_SA1100=y # CONFIG_ARCH_SHARK is not set +# CONFIG_ARCH_AT91RM9200DK is not set # # Archimedes/A5000 Implementations @@ -60,15 +67,27 @@ # # SA11x0 Implementations # +# CONFIG_SA1100_ACCELENT is not set # CONFIG_SA1100_ASSABET is not set # CONFIG_ASSABET_NEPONSET is not set +# CONFIG_SA1100_ADSAGC is not set +# CONFIG_SA1100_ADSBITSY is not set +# CONFIG_SA1100_ADSBITSYPLUS is not set # CONFIG_SA1100_BRUTUS is not set +# CONFIG_SA1100_CEP is not set # CONFIG_SA1100_CERF is not set -# CONFIG_SA1100_BITSY is not set +# CONFIG_SA1100_H3100 is not set +# CONFIG_SA1100_H3600 is not set +# CONFIG_SA1100_H3800 is not set +# CONFIG_SA1100_H3XXX is not set # CONFIG_SA1100_EXTENEX1 is not set # CONFIG_SA1100_FLEXANET is not set # CONFIG_SA1100_FREEBIRD is not set +# CONFIG_SA1100_FRODO is not set CONFIG_SA1100_GRAPHICSCLIENT=y +# CONFIG_SA1100_GRAPHICSMASTER is not set +# CONFIG_SA1100_HACKKIT is not set +# CONFIG_SA1100_BADGE4 is not set # CONFIG_SA1100_JORNADA720 is not set # CONFIG_SA1100_HUW_WEBPANEL is not set # CONFIG_SA1100_ITSY is not set @@ -77,43 +96,59 @@ # CONFIG_SA1100_OMNIMETER is not set # CONFIG_SA1100_PANGOLIN is not set # CONFIG_SA1100_PLEB is not set +# CONFIG_SA1100_PT_SYSTEM3 is not set +# CONFIG_SA1100_SHANNON is not set # CONFIG_SA1100_SHERMAN is not set # CONFIG_SA1100_SIMPAD is not set +# CONFIG_SA1100_SIMPUTER is not set # CONFIG_SA1100_PFS168 is not set # CONFIG_SA1100_VICTOR is not set # CONFIG_SA1100_XP860 is not set # CONFIG_SA1100_YOPY is not set -# CONFIG_SA1100_GRAPHICSMASTER is not set -# CONFIG_SA1100_ADSBITSY is not set # CONFIG_SA1100_USB is not set # CONFIG_SA1100_USB_NETLINK is not set # CONFIG_SA1100_USB_CHAR is not set +# CONFIG_H3600_SLEEVE is not set # # CLPS711X/EP721X Implementations # +# CONFIG_ARCH_AUTCPU12 is not set +# CONFIG_ARCH_CDB89712 is not set +# CONFIG_ARCH_CLEP7312 is not set +# CONFIG_ARCH_EDB7211 is not set # CONFIG_ARCH_P720T is not set +# CONFIG_ARCH_FORTUNET is not set +# CONFIG_ARCH_EP7211 is not set +# CONFIG_ARCH_EP7212 is not set # CONFIG_ARCH_ACORN is not set # CONFIG_FOOTBRIDGE is not set # CONFIG_FOOTBRIDGE_HOST is not set # CONFIG_FOOTBRIDGE_ADDIN is not set -CONFIG_CPU_32=y -# CONFIG_CPU_26 is not set # # Processor Type # -# CONFIG_CPU_32v3 is not set -CONFIG_CPU_32v4=y +CONFIG_CPU_32=y +# CONFIG_CPU_26 is not set # CONFIG_CPU_ARM610 is not set # CONFIG_CPU_ARM710 is not set # CONFIG_CPU_ARM720T is not set # CONFIG_CPU_ARM920T is not set +# CONFIG_CPU_ARM922T is not set +# CONFIG_PLD is not set +# CONFIG_CPU_ARM926T is not set # CONFIG_CPU_ARM1020 is not set +# CONFIG_CPU_ARM1026 is not set # CONFIG_CPU_SA110 is not set CONFIG_CPU_SA1100=y +# CONFIG_CPU_32v3 is not set +CONFIG_CPU_32v4=y + +# +# Processor Features +# CONFIG_DISCONTIGMEM=y -# CONFIG_CPU_BIG_ENDIAN is not set # # General setup @@ -121,6 +156,9 @@ # CONFIG_PCI is not set # CONFIG_ISA is not set # CONFIG_ISA_DMA is not set +# CONFIG_ZBOOT_ROM is not set +CONFIG_ZBOOT_ROM_TEXT=0 +CONFIG_ZBOOT_ROM_BSS=0 # CONFIG_CPU_FREQ is not set CONFIG_HOTPLUG=y @@ -128,6 +166,8 @@ # PCMCIA/CardBus support # CONFIG_PCMCIA=y +CONFIG_PCMCIA_PROBE=y +# CONFIG_I82092 is not set # CONFIG_I82365 is not set # CONFIG_TCIC is not set # CONFIG_PCMCIA_CLPS6700 is not set @@ -136,6 +176,10 @@ CONFIG_SYSVIPC=y # CONFIG_BSD_PROCESS_ACCT is not set CONFIG_SYSCTL=y + +# +# At least one math emulation must be selected +# CONFIG_FPE_NWFPE=y # CONFIG_FPE_FASTFPE is not set CONFIG_KCORE_ELF=y @@ -143,11 +187,9 @@ # CONFIG_BINFMT_AOUT is not set CONFIG_BINFMT_ELF=y # CONFIG_BINFMT_MISC is not set -# CONFIG_PM is not set -# CONFIG_APM is not set +CONFIG_PM=y # CONFIG_ARTHUR is not set -CONFIG_CMDLINE="ip=off" -# CONFIG_PFS168_CMDLINE is not set +CONFIG_CMDLINE="console=ttySA0 mem=16m@0xC0000000 mem=16m@0xC8000000 rw ramdisk_size=8192 initrd=0xC0800000,3m root=/dev/ram" CONFIG_LEDS=y CONFIG_LEDS_TIMER=y CONFIG_LEDS_CPU=y @@ -164,8 +206,9 @@ CONFIG_MTD=y # CONFIG_MTD_DEBUG is not set CONFIG_MTD_PARTITIONS=y +# CONFIG_MTD_CONCAT is not set # CONFIG_MTD_REDBOOT_PARTS is not set -# CONFIG_MTD_BOOTLDR_PARTS is not set +CONFIG_MTD_CMDLINE_PARTS=y # CONFIG_MTD_AFS_PARTS is not set # @@ -180,53 +223,44 @@ # RAM/ROM/Flash chip drivers # CONFIG_MTD_CFI=y +# CONFIG_MTD_JEDECPROBE is not set +CONFIG_MTD_GEN_PROBE=y # CONFIG_MTD_CFI_ADV_OPTIONS is not set CONFIG_MTD_CFI_INTELEXT=y # CONFIG_MTD_CFI_AMDSTD is not set -# CONFIG_MTD_AMDSTD is not set -# CONFIG_MTD_SHARP is not set # CONFIG_MTD_RAM is not set # CONFIG_MTD_ROM is not set +# CONFIG_MTD_ABSENT is not set +# CONFIG_MTD_OBSOLETE_CHIPS is not set +# CONFIG_MTD_AMDSTD is not set +# CONFIG_MTD_SHARP is not set # CONFIG_MTD_JEDEC is not set # # Mapping drivers for chip access # CONFIG_MTD_PHYSMAP=y -CONFIG_MTD_PHYSMAP_START=800000 -CONFIG_MTD_PHYSMAP_LEN=1000000 +CONFIG_MTD_PHYSMAP_START=8000000 +CONFIG_MTD_PHYSMAP_LEN=2000000 CONFIG_MTD_PHYSMAP_BUSWIDTH=4 -# CONFIG_MTD_SUN_UFLASH is not set # CONFIG_MTD_NORA is not set -# CONFIG_MTD_PNC2000 is not set -# CONFIG_MTD_RPXLITE is not set -# CONFIG_MTD_TQM8XXL is not set -# CONFIG_MTD_SC520CDP is not set -# CONFIG_MTD_NETSC520 is not set -# CONFIG_MTD_SBC_GXX is not set -# CONFIG_MTD_ELAN_104NC is not set +# CONFIG_MTD_ARM_INTEGRATOR is not set +# CONFIG_MTD_CDB89712 is not set CONFIG_MTD_SA1100=y -# CONFIG_MTD_SA1100_REDBOOT_PARTITIONS is not set -# CONFIG_MTD_SA1100_BOOTLDR_PARTITIONS is not set # CONFIG_MTD_DC21285 is not set # CONFIG_MTD_IQ80310 is not set -# CONFIG_MTD_DBOX2 is not set -# CONFIG_MTD_CSTM_MIPS_IXX is not set -# CONFIG_MTD_CFI_FLAGADM is not set -# CONFIG_MTD_ARM_INTEGRATOR is not set -# CONFIG_MTD_SOLUTIONENGINE is not set -# CONFIG_MTD_MIXMEM is not set -# CONFIG_MTD_OCTAGON is not set -# CONFIG_MTD_VMAX is not set -# CONFIG_MTD_OCELOT is not set -# CONFIG_MTD_L440GX is not set +# CONFIG_MTD_FORTUNET is not set +# CONFIG_MTD_EPXA10DB is not set +# CONFIG_MTD_AUTCPU12 is not set +# CONFIG_MTD_EDB7312 is not set +# CONFIG_MTD_IMPA7 is not set +# CONFIG_MTD_PCI is not set # # Self-contained MTD device drivers # # CONFIG_MTD_PMC551 is not set # CONFIG_MTD_SLRAM is not set -# CONFIG_MTD_LART is not set # CONFIG_MTD_MTDRAM is not set # CONFIG_MTD_BLKMTD is not set @@ -257,8 +291,10 @@ # CONFIG_PARIDE is not set # CONFIG_BLK_CPQ_DA is not set # CONFIG_BLK_CPQ_CISS_DA is not set +# CONFIG_CISS_SCSI_TAPE is not set # CONFIG_BLK_DEV_DAC960 is not set -CONFIG_BLK_DEV_LOOP=m +# CONFIG_BLK_DEV_UMEM is not set +CONFIG_BLK_DEV_LOOP=y # CONFIG_BLK_DEV_NBD is not set CONFIG_BLK_DEV_RAM=y CONFIG_BLK_DEV_RAM_SIZE=8192 @@ -273,6 +309,7 @@ # CONFIG_MD_RAID0 is not set # CONFIG_MD_RAID1 is not set # CONFIG_MD_RAID5 is not set +# CONFIG_MD_MULTIPATH is not set # CONFIG_BLK_DEV_LVM is not set # @@ -280,7 +317,7 @@ # CONFIG_PACKET=y # CONFIG_PACKET_MMAP is not set -# CONFIG_NETLINK is not set +# CONFIG_NETLINK_DEV is not set # CONFIG_NETFILTER is not set # CONFIG_FILTER is not set CONFIG_UNIX=y @@ -293,17 +330,24 @@ # CONFIG_IP_PNP_RARP is not set # CONFIG_NET_IPIP is not set # CONFIG_NET_IPGRE is not set +# CONFIG_ARPD is not set # CONFIG_INET_ECN is not set # CONFIG_SYN_COOKIES is not set # CONFIG_IPV6 is not set # CONFIG_KHTTPD is not set # CONFIG_ATM is not set +# CONFIG_VLAN_8021Q is not set # -# +# # # CONFIG_IPX is not set # CONFIG_ATALK is not set + +# +# Appletalk devices +# +# CONFIG_DEV_APPLETALK is not set # CONFIG_DECNET is not set # CONFIG_BRIDGE is not set # CONFIG_X25 is not set @@ -321,6 +365,11 @@ # CONFIG_NET_SCHED is not set # +# Network testing +# +# CONFIG_NET_PKTGEN is not set + +# # Network device support # CONFIG_NETDEVICES=y @@ -333,15 +382,17 @@ # CONFIG_BONDING is not set # CONFIG_EQUALIZER is not set # CONFIG_TUN is not set +# CONFIG_ETHERTAP is not set # # Ethernet (10 or 100Mbit) # CONFIG_NET_ETHERNET=y +# CONFIG_ARM_AM79C961A is not set +# CONFIG_ARM_CIRRUS is not set # CONFIG_SUNLANCE is not set # CONFIG_SUNBMAC is not set # CONFIG_SUNQE is not set -# CONFIG_SUNLANCE is not set # CONFIG_SUNGEM is not set # CONFIG_NET_VENDOR_3COM is not set # CONFIG_LANCE is not set @@ -351,6 +402,7 @@ # CONFIG_ULTRA is not set # CONFIG_ULTRA32 is not set CONFIG_SMC9194=y +# CONFIG_SMC91111 is not set # CONFIG_NET_VENDOR_RACAL is not set # CONFIG_NET_ISA is not set # CONFIG_NET_PCI is not set @@ -360,21 +412,44 @@ # Ethernet (1000 Mbit) # # CONFIG_ACENIC is not set -# CONFIG_ACENIC_OMIT_TIGON_I is not set +# CONFIG_DL2K is not set # CONFIG_MYRI_SBUS is not set +# CONFIG_NS83820 is not set # CONFIG_HAMACHI is not set # CONFIG_YELLOWFIN is not set # CONFIG_SK98LIN is not set +# CONFIG_TIGON3 is not set # CONFIG_FDDI is not set # CONFIG_HIPPI is not set # CONFIG_PLIP is not set -# CONFIG_PPP is not set +CONFIG_PPP=y +# CONFIG_PPP_MULTILINK is not set +# CONFIG_PPP_FILTER is not set +CONFIG_PPP_ASYNC=y +# CONFIG_PPP_SYNC_TTY is not set +# CONFIG_PPP_DEFLATE is not set +# CONFIG_PPP_BSDCOMP is not set +# CONFIG_PPPOE is not set # CONFIG_SLIP is not set # # Wireless LAN (non-hamradio) # -# CONFIG_NET_RADIO is not set +CONFIG_NET_RADIO=y +# CONFIG_STRIP is not set +# CONFIG_WAVELAN is not set +# CONFIG_ARLAN is not set +# CONFIG_AIRONET4500 is not set +# CONFIG_AIRONET4500_NONCS is not set +# CONFIG_AIRONET4500_PROC is not set +# CONFIG_HERMES is not set + +# +# Wireless Pcmcia cards support +# +# CONFIG_PCMCIA_HERMES is not set +# CONFIG_AIRO_CS is not set +CONFIG_NET_WIRELESS=y # # Token Ring devices @@ -397,6 +472,7 @@ # CONFIG_PCMCIA_3C574 is not set # CONFIG_PCMCIA_FMVJ18X is not set CONFIG_PCMCIA_PCNET=y +# CONFIG_PCMCIA_AXNET is not set # CONFIG_PCMCIA_NMCLAN is not set # CONFIG_PCMCIA_SMC91C92 is not set # CONFIG_PCMCIA_XIRC2PS is not set @@ -415,7 +491,7 @@ # CONFIG_IRDA is not set # -# ATA/IDE/MFM/RLL support +# ATA/ATAPI/MFM/RLL support # CONFIG_IDE=y @@ -431,6 +507,7 @@ # CONFIG_BLK_DEV_HD is not set CONFIG_BLK_DEV_IDEDISK=y # CONFIG_IDEDISK_MULTI_MODE is not set +# CONFIG_IDEDISK_STROKE is not set # CONFIG_BLK_DEV_IDEDISK_VENDOR is not set # CONFIG_BLK_DEV_IDEDISK_FUJITSU is not set # CONFIG_BLK_DEV_IDEDISK_IBM is not set @@ -445,6 +522,7 @@ # CONFIG_BLK_DEV_IDETAPE is not set # CONFIG_BLK_DEV_IDEFLOPPY is not set # CONFIG_BLK_DEV_IDESCSI is not set +# CONFIG_IDE_TASK_IOCTL is not set # # IDE chipset support/bugfixes @@ -456,6 +534,9 @@ # CONFIG_IDEDMA_AUTO is not set # CONFIG_DMA_NONPCI is not set # CONFIG_BLK_DEV_IDE_MODES is not set +# CONFIG_BLK_DEV_ATARAID is not set +# CONFIG_BLK_DEV_ATARAID_PDC is not set +# CONFIG_BLK_DEV_ATARAID_HPT is not set # # SCSI support @@ -479,46 +560,84 @@ # # Input core support # -# CONFIG_INPUT is not set +CONFIG_INPUT=y +# CONFIG_INPUT_KEYBDEV is not set +CONFIG_INPUT_MOUSEDEV=y +CONFIG_INPUT_MOUSEDEV_SCREEN_X=640 +CONFIG_INPUT_MOUSEDEV_SCREEN_Y=480 +# CONFIG_INPUT_JOYDEV is not set +# CONFIG_INPUT_EVDEV is not set # # Character devices # CONFIG_VT=y -# CONFIG_VT_CONSOLE is not set +CONFIG_VT_CONSOLE=y # CONFIG_SERIAL is not set # CONFIG_SERIAL_EXTENDED is not set # CONFIG_SERIAL_NONSTANDARD is not set -# CONFIG_SERIAL_21285 is not set -# CONFIG_SERIAL_21285_OLD is not set -# CONFIG_SERIAL_21285_CONSOLE is not set + +# +# Serial drivers +# +# CONFIG_SERIAL_ANAKIN is not set +# CONFIG_SERIAL_ANAKIN_CONSOLE is not set # CONFIG_SERIAL_AMBA is not set # CONFIG_SERIAL_AMBA_CONSOLE is not set # CONFIG_SERIAL_CLPS711X is not set # CONFIG_SERIAL_CLPS711X_CONSOLE is not set +# CONFIG_SERIAL_21285 is not set +# CONFIG_SERIAL_21285_OLD is not set +# CONFIG_SERIAL_21285_CONSOLE is not set +# CONFIG_SERIAL_UART00 is not set +# CONFIG_SERIAL_UART00_CONSOLE is not set CONFIG_SERIAL_SA1100=y CONFIG_SERIAL_SA1100_CONSOLE=y CONFIG_SA1100_DEFAULT_BAUDRATE=38400 +# CONFIG_SERIAL_OMAHA is not set +# CONFIG_SERIAL_OMAHA_CONSOLE is not set # CONFIG_SERIAL_8250 is not set # CONFIG_SERIAL_8250_CONSOLE is not set +# CONFIG_SERIAL_8250_EXTENDED is not set +# CONFIG_SERIAL_8250_MANY_PORTS is not set +# CONFIG_SERIAL_8250_SHARE_IRQ is not set +# CONFIG_SERIAL_8250_DETECT_IRQ is not set +# CONFIG_SERIAL_8250_MULTIPORT is not set +# CONFIG_SERIAL_8250_HUB6 is not set CONFIG_SERIAL_CORE=y CONFIG_SERIAL_CORE_CONSOLE=y -CONFIG_UNIX98_PTYS=y -CONFIG_UNIX98_PTY_COUNT=32 +# CONFIG_TOUCHSCREEN_ADS784X is not set CONFIG_UCB1200=y CONFIG_TOUCHSCREEN_UCB1200=y CONFIG_AUDIO_UCB1200=y CONFIG_ADC_UCB1200=y -# CONFIG_TOUCHSCREEN_BITSY is not set -# CONFIG_PROFILER is not set -# CONFIG_PFS168_SPI is not set -# CONFIG_PFS168_DTMF is not set -# CONFIG_PFS168_MISC is not set +CONFIG_UNIX98_PTYS=y +CONFIG_UNIX98_PTY_COUNT=32 # # I2C support # -# CONFIG_I2C is not set +CONFIG_I2C=y +# CONFIG_I2C_ALGOBIT is not set +# CONFIG_I2C_ALGOPCF is not set +CONFIG_ADSSIO_I2C_ALGO=y +# CONFIG_I2C_CHARDEV is not set +# CONFIG_I2C_PROC is not set +CONFIG_I2C_DS1307=y +CONFIG_I2C_DS1307_ADS=y + +# +# L3 serial bus support +# +# CONFIG_L3 is not set +# CONFIG_L3_ALGOBIT is not set +# CONFIG_L3_BIT_SA1100_GPIO is not set + +# +# Other L3 adapters +# +# CONFIG_L3_SA1111 is not set +# CONFIG_BIT_SA1100_GPIO is not set # # Mice @@ -529,18 +648,43 @@ # # Joysticks # -# CONFIG_JOYSTICK is not set +# CONFIG_INPUT_GAMEPORT is not set +# CONFIG_INPUT_NS558 is not set +# CONFIG_INPUT_LIGHTNING is not set +# CONFIG_INPUT_PCIGAME is not set +# CONFIG_INPUT_CS461X is not set +# CONFIG_INPUT_EMU10K1 is not set +# CONFIG_INPUT_SERIO is not set +# CONFIG_INPUT_SERPORT is not set # -# Input core support is needed for joysticks +# Joysticks # +# CONFIG_INPUT_ANALOG is not set +# CONFIG_INPUT_A3D is not set +# CONFIG_INPUT_ADI is not set +# CONFIG_INPUT_COBRA is not set +# CONFIG_INPUT_GF2K is not set +# CONFIG_INPUT_GRIP is not set +# CONFIG_INPUT_INTERACT is not set +# CONFIG_INPUT_TMDC is not set +# CONFIG_INPUT_SIDEWINDER is not set +# CONFIG_INPUT_IFORCE_USB is not set +# CONFIG_INPUT_IFORCE_232 is not set +# CONFIG_INPUT_WARRIOR is not set +# CONFIG_INPUT_MAGELLAN is not set +# CONFIG_INPUT_SPACEORB is not set +# CONFIG_INPUT_SPACEBALL is not set +# CONFIG_INPUT_STINGER is not set +# CONFIG_INPUT_DB9 is not set +# CONFIG_INPUT_GAMECON is not set +# CONFIG_INPUT_TURBOGRAFX is not set # CONFIG_QIC02_TAPE is not set # # Watchdog Cards # # CONFIG_WATCHDOG is not set -# CONFIG_INTEL_RNG is not set # CONFIG_NVRAM is not set # CONFIG_RTC is not set # CONFIG_SA1100_RTC is not set @@ -572,18 +716,21 @@ # CONFIG_AUTOFS4_FS is not set # CONFIG_REISERFS_FS is not set # CONFIG_REISERFS_CHECK is not set +# CONFIG_REISERFS_PROC_INFO is not set # CONFIG_ADFS_FS is not set # CONFIG_ADFS_FS_RW is not set # CONFIG_AFFS_FS is not set # CONFIG_HFS_FS is not set # CONFIG_BFS_FS is not set +# CONFIG_EXT3_FS is not set +# CONFIG_JBD is not set +# CONFIG_JBD_DEBUG is not set CONFIG_FAT_FS=y # CONFIG_MSDOS_FS is not set # CONFIG_UMSDOS_FS is not set CONFIG_VFAT_FS=y # CONFIG_EFS_FS is not set -CONFIG_JFFS_FS=y -CONFIG_JFFS_FS_VERBOSE=0 +# CONFIG_JFFS_FS is not set CONFIG_JFFS2_FS=y CONFIG_JFFS2_FS_DEBUG=0 CONFIG_CRAMFS=y @@ -591,6 +738,7 @@ CONFIG_RAMFS=y # CONFIG_ISO9660_FS is not set # CONFIG_JOLIET is not set +# CONFIG_ZISOFS is not set # CONFIG_MINIX_FS is not set # CONFIG_VXFS_FS is not set # CONFIG_NTFS_FS is not set @@ -615,6 +763,7 @@ # Network File Systems # # CONFIG_CODA_FS is not set +# CONFIG_INTERMEZZO_FS is not set CONFIG_NFS_FS=y # CONFIG_NFS_V3 is not set CONFIG_ROOT_NFS=y @@ -632,6 +781,8 @@ # CONFIG_NCPFS_SMALLDOS is not set # CONFIG_NCPFS_NLS is not set # CONFIG_NCPFS_EXTRAS is not set +# CONFIG_ZISOFS_FS is not set +CONFIG_ZLIB_FS_INFLATE=y # # Partition Types @@ -666,6 +817,7 @@ # CONFIG_NLS_CODEPAGE_949 is not set # CONFIG_NLS_CODEPAGE_874 is not set # CONFIG_NLS_ISO8859_8 is not set +# CONFIG_NLS_CODEPAGE_1250 is not set # CONFIG_NLS_CODEPAGE_1251 is not set # CONFIG_NLS_ISO8859_1 is not set # CONFIG_NLS_ISO8859_2 is not set @@ -694,11 +846,11 @@ CONFIG_FB=y CONFIG_DUMMY_CONSOLE=y # CONFIG_FB_ACORN is not set +# CONFIG_FB_ANAKIN is not set # CONFIG_FB_CLPS711X is not set -# CONFIG_FB_CYBER2000 is not set CONFIG_FB_SA1100=y -# CONFIG_FB_ANAKIN is not set -# CONFIG_FB_E1355 is not set +# CONFIG_FB_CYBER2000 is not set +# CONFIG_FB_E13806 is not set # CONFIG_FB_VIRTUAL is not set # CONFIG_FBCON_ADVANCED is not set CONFIG_FBCON_CFB2=y @@ -719,6 +871,15 @@ # CONFIG_SOUND is not set # +# Multimedia Capabilities Port drivers +# +# CONFIG_MCP is not set +# CONFIG_MCP_SA1100 is not set +# CONFIG_MCP_UCB1200 is not set +# CONFIG_MCP_UCB1200_AUDIO is not set +# CONFIG_MCP_UCB1200_TS is not set + +# # USB support # # CONFIG_USB is not set @@ -732,11 +893,16 @@ # Kernel hacking # CONFIG_FRAME_POINTER=y -CONFIG_DEBUG_ERRORS=y -CONFIG_DEBUG_USER=y +# CONFIG_DEBUG_USER is not set # CONFIG_DEBUG_INFO is not set -# CONFIG_MAGIC_SYSRQ is not set # CONFIG_NO_PGT_CACHE is not set +# CONFIG_DEBUG_KERNEL is not set +# CONFIG_DEBUG_SLAB is not set +# CONFIG_MAGIC_SYSRQ is not set +# CONFIG_DEBUG_SPINLOCK is not set +# CONFIG_DEBUG_WAITQ is not set +# CONFIG_DEBUG_BUGVERBOSE is not set +# CONFIG_DEBUG_ERRORS is not set # CONFIG_DEBUG_LL is not set # CONFIG_DEBUG_DC21285_PORT is not set # CONFIG_DEBUG_CLPS711X_UART2 is not set diff -urN orig/arch/arm/def-configs/graphicsmaster linux/arch/arm/def-configs/graphicsmaster --- orig/arch/arm/def-configs/graphicsmaster Mon Aug 5 13:29:43 2002 +++ linux/arch/arm/def-configs/graphicsmaster Thu Feb 27 23:20:10 2003 @@ -1,5 +1,5 @@ # -# Automatically generated by make menuconfig: don't edit +# Automatically generated make config: don't edit # CONFIG_ARM=y # CONFIG_EISA is not set @@ -8,6 +8,8 @@ CONFIG_UID16=y CONFIG_RWSEM_GENERIC_SPINLOCK=y # CONFIG_RWSEM_XCHGADD_ALGORITHM is not set +# CONFIG_GENERIC_BUST_SPINLOCK is not set +# CONFIG_GENERIC_ISA_DMA is not set # # Code maturity level options @@ -31,16 +33,25 @@ # CONFIG_ARCH_CLPS711X is not set # CONFIG_ARCH_CO285 is not set # CONFIG_ARCH_EBSA110 is not set +# CONFIG_ARCH_CAMELOT is not set # CONFIG_ARCH_FOOTBRIDGE is not set # CONFIG_ARCH_INTEGRATOR is not set +# CONFIG_ARCH_OMAHA is not set # CONFIG_ARCH_L7200 is not set +# CONFIG_ARCH_MX1ADS is not set # CONFIG_ARCH_RPC is not set +# CONFIG_ARCH_RISCSTATION is not set CONFIG_ARCH_SA1100=y # CONFIG_ARCH_SHARK is not set +# CONFIG_ARCH_AT91RM9200DK is not set # # Archimedes/A5000 Implementations # + +# +# Archimedes/A5000 Implementations (select only ONE) +# # CONFIG_ARCH_ARC is not set # CONFIG_ARCH_A5K is not set @@ -56,15 +67,27 @@ # # SA11x0 Implementations # +# CONFIG_SA1100_ACCELENT is not set # CONFIG_SA1100_ASSABET is not set # CONFIG_ASSABET_NEPONSET is not set +# CONFIG_SA1100_ADSAGC is not set +# CONFIG_SA1100_ADSBITSY is not set +# CONFIG_SA1100_ADSBITSYPLUS is not set # CONFIG_SA1100_BRUTUS is not set +# CONFIG_SA1100_CEP is not set # CONFIG_SA1100_CERF is not set -# CONFIG_SA1100_BITSY is not set +# CONFIG_SA1100_H3100 is not set +# CONFIG_SA1100_H3600 is not set +# CONFIG_SA1100_H3800 is not set +# CONFIG_SA1100_H3XXX is not set # CONFIG_SA1100_EXTENEX1 is not set # CONFIG_SA1100_FLEXANET is not set # CONFIG_SA1100_FREEBIRD is not set +# CONFIG_SA1100_FRODO is not set # CONFIG_SA1100_GRAPHICSCLIENT is not set +CONFIG_SA1100_GRAPHICSMASTER=y +# CONFIG_SA1100_HACKKIT is not set +# CONFIG_SA1100_BADGE4 is not set # CONFIG_SA1100_JORNADA720 is not set # CONFIG_SA1100_HUW_WEBPANEL is not set # CONFIG_SA1100_ITSY is not set @@ -73,40 +96,61 @@ # CONFIG_SA1100_OMNIMETER is not set # CONFIG_SA1100_PANGOLIN is not set # CONFIG_SA1100_PLEB is not set +# CONFIG_SA1100_PT_SYSTEM3 is not set +# CONFIG_SA1100_SHANNON is not set # CONFIG_SA1100_SHERMAN is not set # CONFIG_SA1100_SIMPAD is not set +# CONFIG_SA1100_SIMPUTER is not set # CONFIG_SA1100_PFS168 is not set # CONFIG_SA1100_VICTOR is not set # CONFIG_SA1100_XP860 is not set # CONFIG_SA1100_YOPY is not set -CONFIG_SA1100_GRAPHICSMASTER=y -# CONFIG_SA1100_ADSBITSY is not set -CONFIG_SA1111=y # CONFIG_SA1100_USB is not set # CONFIG_SA1100_USB_NETLINK is not set # CONFIG_SA1100_USB_CHAR is not set +# CONFIG_H3600_SLEEVE is not set # # CLPS711X/EP721X Implementations # +# CONFIG_ARCH_AUTCPU12 is not set +# CONFIG_ARCH_CDB89712 is not set +# CONFIG_ARCH_CLEP7312 is not set +# CONFIG_ARCH_EDB7211 is not set # CONFIG_ARCH_P720T is not set +# CONFIG_ARCH_FORTUNET is not set +# CONFIG_ARCH_EP7211 is not set +# CONFIG_ARCH_EP7212 is not set # CONFIG_ARCH_ACORN is not set # CONFIG_FOOTBRIDGE is not set # CONFIG_FOOTBRIDGE_HOST is not set # CONFIG_FOOTBRIDGE_ADDIN is not set +CONFIG_SA1111=y +CONFIG_FORCE_MAX_ZONEORDER=9 + +# +# Processor Type +# CONFIG_CPU_32=y # CONFIG_CPU_26 is not set -# CONFIG_CPU_32v3 is not set -CONFIG_CPU_32v4=y # CONFIG_CPU_ARM610 is not set # CONFIG_CPU_ARM710 is not set # CONFIG_CPU_ARM720T is not set # CONFIG_CPU_ARM920T is not set +# CONFIG_CPU_ARM922T is not set +# CONFIG_PLD is not set +# CONFIG_CPU_ARM926T is not set # CONFIG_CPU_ARM1020 is not set +# CONFIG_CPU_ARM1026 is not set # CONFIG_CPU_SA110 is not set CONFIG_CPU_SA1100=y +# CONFIG_CPU_32v3 is not set +CONFIG_CPU_32v4=y + +# +# Processor Features +# CONFIG_DISCONTIGMEM=y -# CONFIG_CPU_BIG_ENDIAN is not set # # General setup @@ -114,6 +158,9 @@ # CONFIG_PCI is not set # CONFIG_ISA is not set # CONFIG_ISA_DMA is not set +# CONFIG_ZBOOT_ROM is not set +CONFIG_ZBOOT_ROM_TEXT=0 +CONFIG_ZBOOT_ROM_BSS=0 # CONFIG_CPU_FREQ is not set CONFIG_HOTPLUG=y @@ -121,6 +168,8 @@ # PCMCIA/CardBus support # CONFIG_PCMCIA=y +CONFIG_PCMCIA_PROBE=y +# CONFIG_I82092 is not set # CONFIG_I82365 is not set # CONFIG_TCIC is not set # CONFIG_PCMCIA_CLPS6700 is not set @@ -129,6 +178,10 @@ CONFIG_SYSVIPC=y # CONFIG_BSD_PROCESS_ACCT is not set CONFIG_SYSCTL=y + +# +# At least one math emulation must be selected +# CONFIG_FPE_NWFPE=y # CONFIG_FPE_FASTFPE is not set CONFIG_KCORE_ELF=y @@ -136,11 +189,9 @@ # CONFIG_BINFMT_AOUT is not set CONFIG_BINFMT_ELF=y # CONFIG_BINFMT_MISC is not set -# CONFIG_PM is not set -# CONFIG_APM is not set +CONFIG_PM=y # CONFIG_ARTHUR is not set -CONFIG_CMDLINE="ip=off" -# CONFIG_PFS168_CMDLINE is not set +CONFIG_CMDLINE="console=ttySA0 mem=16m@0xC0000000 mem=16m@0xC8000000 rw ramdisk_size=8192 initrd=0xC0800000,3m root=/dev/ram" CONFIG_LEDS=y CONFIG_LEDS_TIMER=y CONFIG_LEDS_CPU=y @@ -157,9 +208,14 @@ CONFIG_MTD=y # CONFIG_MTD_DEBUG is not set CONFIG_MTD_PARTITIONS=y +# CONFIG_MTD_CONCAT is not set # CONFIG_MTD_REDBOOT_PARTS is not set -# CONFIG_MTD_BOOTLDR_PARTS is not set +CONFIG_MTD_CMDLINE_PARTS=y # CONFIG_MTD_AFS_PARTS is not set + +# +# User Modules And Translation Layers +# CONFIG_MTD_CHAR=y CONFIG_MTD_BLOCK=y # CONFIG_FTL is not set @@ -169,55 +225,50 @@ # RAM/ROM/Flash chip drivers # CONFIG_MTD_CFI=y +# CONFIG_MTD_JEDECPROBE is not set +CONFIG_MTD_GEN_PROBE=y # CONFIG_MTD_CFI_ADV_OPTIONS is not set CONFIG_MTD_CFI_INTELEXT=y # CONFIG_MTD_CFI_AMDSTD is not set -# CONFIG_MTD_AMDSTD is not set -# CONFIG_MTD_SHARP is not set # CONFIG_MTD_RAM is not set # CONFIG_MTD_ROM is not set +# CONFIG_MTD_ABSENT is not set +# CONFIG_MTD_OBSOLETE_CHIPS is not set +# CONFIG_MTD_AMDSTD is not set +# CONFIG_MTD_SHARP is not set # CONFIG_MTD_JEDEC is not set # # Mapping drivers for chip access # CONFIG_MTD_PHYSMAP=y -CONFIG_MTD_PHYSMAP_START=800000 -CONFIG_MTD_PHYSMAP_LEN=1000000 +CONFIG_MTD_PHYSMAP_START=8000000 +CONFIG_MTD_PHYSMAP_LEN=4000000 CONFIG_MTD_PHYSMAP_BUSWIDTH=4 -# CONFIG_MTD_SUN_UFLASH is not set # CONFIG_MTD_NORA is not set -# CONFIG_MTD_PNC2000 is not set -# CONFIG_MTD_RPXLITE is not set -# CONFIG_MTD_TQM8XXL is not set -# CONFIG_MTD_SC520CDP is not set -# CONFIG_MTD_NETSC520 is not set -# CONFIG_MTD_SBC_GXX is not set -# CONFIG_MTD_ELAN_104NC is not set +# CONFIG_MTD_ARM_INTEGRATOR is not set +# CONFIG_MTD_CDB89712 is not set CONFIG_MTD_SA1100=y -# CONFIG_MTD_SA1100_REDBOOT_PARTITIONS is not set -# CONFIG_MTD_SA1100_BOOTLDR_PARTITIONS is not set # CONFIG_MTD_DC21285 is not set # CONFIG_MTD_IQ80310 is not set -# CONFIG_MTD_DBOX2 is not set -# CONFIG_MTD_CSTM_MIPS_IXX is not set -# CONFIG_MTD_CFI_FLAGADM is not set -# CONFIG_MTD_ARM_INTEGRATOR is not set -# CONFIG_MTD_SOLUTIONENGINE is not set -# CONFIG_MTD_MIXMEM is not set -# CONFIG_MTD_OCTAGON is not set -# CONFIG_MTD_VMAX is not set -# CONFIG_MTD_OCELOT is not set -# CONFIG_MTD_L440GX is not set +# CONFIG_MTD_FORTUNET is not set +# CONFIG_MTD_EPXA10DB is not set +# CONFIG_MTD_AUTCPU12 is not set +# CONFIG_MTD_EDB7312 is not set +# CONFIG_MTD_IMPA7 is not set +# CONFIG_MTD_PCI is not set # # Self-contained MTD device drivers # # CONFIG_MTD_PMC551 is not set # CONFIG_MTD_SLRAM is not set -# CONFIG_MTD_LART is not set # CONFIG_MTD_MTDRAM is not set # CONFIG_MTD_BLKMTD is not set + +# +# Disk-On-Chip Device Drivers +# # CONFIG_MTD_DOC1000 is not set # CONFIG_MTD_DOC2000 is not set # CONFIG_MTD_DOC2001 is not set @@ -242,8 +293,10 @@ # CONFIG_PARIDE is not set # CONFIG_BLK_CPQ_DA is not set # CONFIG_BLK_CPQ_CISS_DA is not set +# CONFIG_CISS_SCSI_TAPE is not set # CONFIG_BLK_DEV_DAC960 is not set -CONFIG_BLK_DEV_LOOP=m +# CONFIG_BLK_DEV_UMEM is not set +CONFIG_BLK_DEV_LOOP=y # CONFIG_BLK_DEV_NBD is not set CONFIG_BLK_DEV_RAM=y CONFIG_BLK_DEV_RAM_SIZE=8192 @@ -258,6 +311,7 @@ # CONFIG_MD_RAID0 is not set # CONFIG_MD_RAID1 is not set # CONFIG_MD_RAID5 is not set +# CONFIG_MD_MULTIPATH is not set # CONFIG_BLK_DEV_LVM is not set # @@ -265,7 +319,7 @@ # CONFIG_PACKET=y # CONFIG_PACKET_MMAP is not set -# CONFIG_NETLINK is not set +# CONFIG_NETLINK_DEV is not set # CONFIG_NETFILTER is not set # CONFIG_FILTER is not set CONFIG_UNIX=y @@ -278,13 +332,24 @@ # CONFIG_IP_PNP_RARP is not set # CONFIG_NET_IPIP is not set # CONFIG_NET_IPGRE is not set +# CONFIG_ARPD is not set # CONFIG_INET_ECN is not set # CONFIG_SYN_COOKIES is not set # CONFIG_IPV6 is not set # CONFIG_KHTTPD is not set # CONFIG_ATM is not set +# CONFIG_VLAN_8021Q is not set + +# +# +# # CONFIG_IPX is not set # CONFIG_ATALK is not set + +# +# Appletalk devices +# +# CONFIG_DEV_APPLETALK is not set # CONFIG_DECNET is not set # CONFIG_BRIDGE is not set # CONFIG_X25 is not set @@ -302,6 +367,11 @@ # CONFIG_NET_SCHED is not set # +# Network testing +# +# CONFIG_NET_PKTGEN is not set + +# # Network device support # CONFIG_NETDEVICES=y @@ -314,15 +384,17 @@ # CONFIG_BONDING is not set # CONFIG_EQUALIZER is not set # CONFIG_TUN is not set +# CONFIG_ETHERTAP is not set # # Ethernet (10 or 100Mbit) # CONFIG_NET_ETHERNET=y +# CONFIG_ARM_AM79C961A is not set +# CONFIG_ARM_CIRRUS is not set # CONFIG_SUNLANCE is not set # CONFIG_SUNBMAC is not set # CONFIG_SUNQE is not set -# CONFIG_SUNLANCE is not set # CONFIG_SUNGEM is not set # CONFIG_NET_VENDOR_3COM is not set # CONFIG_LANCE is not set @@ -332,6 +404,7 @@ # CONFIG_ULTRA is not set # CONFIG_ULTRA32 is not set CONFIG_SMC9194=y +# CONFIG_SMC91111 is not set # CONFIG_NET_VENDOR_RACAL is not set # CONFIG_NET_ISA is not set # CONFIG_NET_PCI is not set @@ -341,21 +414,44 @@ # Ethernet (1000 Mbit) # # CONFIG_ACENIC is not set -# CONFIG_ACENIC_OMIT_TIGON_I is not set +# CONFIG_DL2K is not set # CONFIG_MYRI_SBUS is not set +# CONFIG_NS83820 is not set # CONFIG_HAMACHI is not set # CONFIG_YELLOWFIN is not set # CONFIG_SK98LIN is not set +# CONFIG_TIGON3 is not set # CONFIG_FDDI is not set # CONFIG_HIPPI is not set # CONFIG_PLIP is not set -# CONFIG_PPP is not set +CONFIG_PPP=y +# CONFIG_PPP_MULTILINK is not set +# CONFIG_PPP_FILTER is not set +CONFIG_PPP_ASYNC=y +# CONFIG_PPP_SYNC_TTY is not set +# CONFIG_PPP_DEFLATE is not set +# CONFIG_PPP_BSDCOMP is not set +# CONFIG_PPPOE is not set # CONFIG_SLIP is not set # # Wireless LAN (non-hamradio) # -# CONFIG_NET_RADIO is not set +CONFIG_NET_RADIO=y +# CONFIG_STRIP is not set +# CONFIG_WAVELAN is not set +# CONFIG_ARLAN is not set +# CONFIG_AIRONET4500 is not set +# CONFIG_AIRONET4500_NONCS is not set +# CONFIG_AIRONET4500_PROC is not set +# CONFIG_HERMES is not set + +# +# Wireless Pcmcia cards support +# +# CONFIG_PCMCIA_HERMES is not set +# CONFIG_AIRO_CS is not set +CONFIG_NET_WIRELESS=y # # Token Ring devices @@ -378,6 +474,7 @@ # CONFIG_PCMCIA_3C574 is not set # CONFIG_PCMCIA_FMVJ18X is not set CONFIG_PCMCIA_PCNET=y +# CONFIG_PCMCIA_AXNET is not set # CONFIG_PCMCIA_NMCLAN is not set # CONFIG_PCMCIA_SMC91C92 is not set # CONFIG_PCMCIA_XIRC2PS is not set @@ -396,7 +493,7 @@ # CONFIG_IRDA is not set # -# ATA/IDE/MFM/RLL support +# ATA/ATAPI/MFM/RLL support # CONFIG_IDE=y @@ -404,10 +501,15 @@ # IDE, ATA and ATAPI Block devices # CONFIG_BLK_DEV_IDE=y + +# +# Please see Documentation/ide.txt for help/info on IDE drives +# # CONFIG_BLK_DEV_HD_IDE is not set # CONFIG_BLK_DEV_HD is not set CONFIG_BLK_DEV_IDEDISK=y # CONFIG_IDEDISK_MULTI_MODE is not set +# CONFIG_IDEDISK_STROKE is not set # CONFIG_BLK_DEV_IDEDISK_VENDOR is not set # CONFIG_BLK_DEV_IDEDISK_FUJITSU is not set # CONFIG_BLK_DEV_IDEDISK_IBM is not set @@ -422,6 +524,11 @@ # CONFIG_BLK_DEV_IDETAPE is not set # CONFIG_BLK_DEV_IDEFLOPPY is not set # CONFIG_BLK_DEV_IDESCSI is not set +# CONFIG_IDE_TASK_IOCTL is not set + +# +# IDE chipset support/bugfixes +# # CONFIG_BLK_DEV_CMD640 is not set # CONFIG_BLK_DEV_CMD640_ENHANCED is not set # CONFIG_BLK_DEV_ISAPNP is not set @@ -429,6 +536,9 @@ # CONFIG_IDEDMA_AUTO is not set # CONFIG_DMA_NONPCI is not set # CONFIG_BLK_DEV_IDE_MODES is not set +# CONFIG_BLK_DEV_ATARAID is not set +# CONFIG_BLK_DEV_ATARAID_PDC is not set +# CONFIG_BLK_DEV_ATARAID_HPT is not set # # SCSI support @@ -464,35 +574,47 @@ # Character devices # CONFIG_VT=y -# CONFIG_VT_CONSOLE is not set +CONFIG_VT_CONSOLE=y # CONFIG_SERIAL is not set # CONFIG_SERIAL_EXTENDED is not set # CONFIG_SERIAL_NONSTANDARD is not set -# CONFIG_SERIAL_21285 is not set -# CONFIG_SERIAL_21285_OLD is not set -# CONFIG_SERIAL_21285_CONSOLE is not set + +# +# Serial drivers +# +# CONFIG_SERIAL_ANAKIN is not set +# CONFIG_SERIAL_ANAKIN_CONSOLE is not set # CONFIG_SERIAL_AMBA is not set # CONFIG_SERIAL_AMBA_CONSOLE is not set # CONFIG_SERIAL_CLPS711X is not set # CONFIG_SERIAL_CLPS711X_CONSOLE is not set +# CONFIG_SERIAL_21285 is not set +# CONFIG_SERIAL_21285_OLD is not set +# CONFIG_SERIAL_21285_CONSOLE is not set +# CONFIG_SERIAL_UART00 is not set +# CONFIG_SERIAL_UART00_CONSOLE is not set CONFIG_SERIAL_SA1100=y CONFIG_SERIAL_SA1100_CONSOLE=y CONFIG_SA1100_DEFAULT_BAUDRATE=38400 -# CONFIG_SERIAL_8250 is not set -# CONFIG_SERIAL_8250_CONSOLE is not set +# CONFIG_SERIAL_OMAHA is not set +# CONFIG_SERIAL_OMAHA_CONSOLE is not set +CONFIG_SERIAL_8250=y +CONFIG_SERIAL_8250_CONSOLE=y +# CONFIG_SERIAL_8250_EXTENDED is not set +# CONFIG_SERIAL_8250_MANY_PORTS is not set +# CONFIG_SERIAL_8250_SHARE_IRQ is not set +# CONFIG_SERIAL_8250_DETECT_IRQ is not set +# CONFIG_SERIAL_8250_MULTIPORT is not set +# CONFIG_SERIAL_8250_HUB6 is not set CONFIG_SERIAL_CORE=y CONFIG_SERIAL_CORE_CONSOLE=y -CONFIG_UNIX98_PTYS=y -CONFIG_UNIX98_PTY_COUNT=32 +# CONFIG_TOUCHSCREEN_ADS784X is not set CONFIG_UCB1200=y CONFIG_TOUCHSCREEN_UCB1200=y CONFIG_AUDIO_UCB1200=y CONFIG_ADC_UCB1200=y -# CONFIG_TOUCHSCREEN_BITSY is not set -# CONFIG_PROFILER is not set -# CONFIG_PFS168_SPI is not set -# CONFIG_PFS168_DTMF is not set -# CONFIG_PFS168_MISC is not set +CONFIG_UNIX98_PTYS=y +CONFIG_UNIX98_PTY_COUNT=32 # # I2C support @@ -500,6 +622,19 @@ # CONFIG_I2C is not set # +# L3 serial bus support +# +# CONFIG_L3 is not set +# CONFIG_L3_ALGOBIT is not set +# CONFIG_L3_BIT_SA1100_GPIO is not set + +# +# Other L3 adapters +# +# CONFIG_L3_SA1111 is not set +# CONFIG_BIT_SA1100_GPIO is not set + +# # Mice # # CONFIG_BUSMOUSE is not set @@ -508,14 +643,43 @@ # # Joysticks # -# CONFIG_JOYSTICK is not set +# CONFIG_INPUT_GAMEPORT is not set +# CONFIG_INPUT_NS558 is not set +# CONFIG_INPUT_LIGHTNING is not set +# CONFIG_INPUT_PCIGAME is not set +# CONFIG_INPUT_CS461X is not set +# CONFIG_INPUT_EMU10K1 is not set +# CONFIG_INPUT_SERIO is not set +# CONFIG_INPUT_SERPORT is not set + +# +# Joysticks +# +# CONFIG_INPUT_ANALOG is not set +# CONFIG_INPUT_A3D is not set +# CONFIG_INPUT_ADI is not set +# CONFIG_INPUT_COBRA is not set +# CONFIG_INPUT_GF2K is not set +# CONFIG_INPUT_GRIP is not set +# CONFIG_INPUT_INTERACT is not set +# CONFIG_INPUT_TMDC is not set +# CONFIG_INPUT_SIDEWINDER is not set +# CONFIG_INPUT_IFORCE_USB is not set +# CONFIG_INPUT_IFORCE_232 is not set +# CONFIG_INPUT_WARRIOR is not set +# CONFIG_INPUT_MAGELLAN is not set +# CONFIG_INPUT_SPACEORB is not set +# CONFIG_INPUT_SPACEBALL is not set +# CONFIG_INPUT_STINGER is not set +# CONFIG_INPUT_DB9 is not set +# CONFIG_INPUT_GAMECON is not set +# CONFIG_INPUT_TURBOGRAFX is not set # CONFIG_QIC02_TAPE is not set # # Watchdog Cards # # CONFIG_WATCHDOG is not set -# CONFIG_INTEL_RNG is not set # CONFIG_NVRAM is not set # CONFIG_RTC is not set # CONFIG_SA1100_RTC is not set @@ -533,6 +697,7 @@ # # PCMCIA character devices # +# CONFIG_PCMCIA_SERIAL_CS is not set # # Multimedia devices @@ -547,18 +712,21 @@ # CONFIG_AUTOFS4_FS is not set # CONFIG_REISERFS_FS is not set # CONFIG_REISERFS_CHECK is not set +# CONFIG_REISERFS_PROC_INFO is not set # CONFIG_ADFS_FS is not set # CONFIG_ADFS_FS_RW is not set # CONFIG_AFFS_FS is not set # CONFIG_HFS_FS is not set # CONFIG_BFS_FS is not set +# CONFIG_EXT3_FS is not set +# CONFIG_JBD is not set +# CONFIG_JBD_DEBUG is not set CONFIG_FAT_FS=y # CONFIG_MSDOS_FS is not set # CONFIG_UMSDOS_FS is not set CONFIG_VFAT_FS=y # CONFIG_EFS_FS is not set -CONFIG_JFFS_FS=y -CONFIG_JFFS_FS_VERBOSE=0 +# CONFIG_JFFS_FS is not set CONFIG_JFFS2_FS=y CONFIG_JFFS2_FS_DEBUG=0 CONFIG_CRAMFS=y @@ -566,6 +734,7 @@ CONFIG_RAMFS=y # CONFIG_ISO9660_FS is not set # CONFIG_JOLIET is not set +# CONFIG_ZISOFS is not set # CONFIG_MINIX_FS is not set # CONFIG_VXFS_FS is not set # CONFIG_NTFS_FS is not set @@ -590,6 +759,7 @@ # Network File Systems # # CONFIG_CODA_FS is not set +# CONFIG_INTERMEZZO_FS is not set CONFIG_NFS_FS=y # CONFIG_NFS_V3 is not set CONFIG_ROOT_NFS=y @@ -607,6 +777,8 @@ # CONFIG_NCPFS_SMALLDOS is not set # CONFIG_NCPFS_NLS is not set # CONFIG_NCPFS_EXTRAS is not set +# CONFIG_ZISOFS_FS is not set +CONFIG_ZLIB_FS_INFLATE=y # # Partition Types @@ -641,6 +813,7 @@ # CONFIG_NLS_CODEPAGE_949 is not set # CONFIG_NLS_CODEPAGE_874 is not set # CONFIG_NLS_ISO8859_8 is not set +# CONFIG_NLS_CODEPAGE_1250 is not set # CONFIG_NLS_CODEPAGE_1251 is not set # CONFIG_NLS_ISO8859_1 is not set # CONFIG_NLS_ISO8859_2 is not set @@ -669,11 +842,11 @@ CONFIG_FB=y CONFIG_DUMMY_CONSOLE=y # CONFIG_FB_ACORN is not set +# CONFIG_FB_ANAKIN is not set # CONFIG_FB_CLPS711X is not set -# CONFIG_FB_CYBER2000 is not set CONFIG_FB_SA1100=y -# CONFIG_FB_ANAKIN is not set -# CONFIG_FB_E1355 is not set +# CONFIG_FB_CYBER2000 is not set +# CONFIG_FB_E13806 is not set # CONFIG_FB_VIRTUAL is not set # CONFIG_FBCON_ADVANCED is not set CONFIG_FBCON_CFB2=y @@ -691,50 +864,169 @@ # # Sound # -# CONFIG_SOUND is not set +CONFIG_SOUND=y +# CONFIG_SOUND_BT878 is not set +# CONFIG_SOUND_CMPCI is not set +# CONFIG_SOUND_EMU10K1 is not set +# CONFIG_MIDI_EMU10K1 is not set +# CONFIG_SOUND_FUSION is not set +# CONFIG_SOUND_CS4281 is not set +# CONFIG_SOUND_ES1370 is not set +# CONFIG_SOUND_ES1371 is not set +# CONFIG_SOUND_ESSSOLO1 is not set +# CONFIG_SOUND_MAESTRO is not set +# CONFIG_SOUND_MAESTRO3 is not set +# CONFIG_SOUND_ICH is not set +# CONFIG_SOUND_RME96XX is not set +# CONFIG_SOUND_SONICVIBES is not set +# CONFIG_SOUND_TRIDENT is not set +# CONFIG_SOUND_MSNDCLAS is not set +# CONFIG_SOUND_MSNDPIN is not set +# CONFIG_SOUND_VIA82CXXX is not set +# CONFIG_MIDI_VIA82CXXX is not set +CONFIG_SOUND_SA1100=y +# CONFIG_SOUND_UDA1341 is not set +# CONFIG_SOUND_ASSABET_UDA1341 is not set +# CONFIG_SOUND_H3600_UDA1341 is not set +# CONFIG_SOUND_PANGOLIN_UDA1341 is not set +# CONFIG_SOUND_SA1111_UDA1341 is not set +# CONFIG_SOUND_SA1111_AC97 is not set +# CONFIG_SOUND_SA1100SSP is not set +# CONFIG_SOUND_OSS is not set +# CONFIG_SOUND_VIDC is not set +# CONFIG_SOUND_WAVEARTIST is not set +# CONFIG_SOUND_TVMIXER is not set + +# +# Multimedia Capabilities Port drivers +# +# CONFIG_MCP is not set +# CONFIG_MCP_SA1100 is not set +# CONFIG_MCP_UCB1200 is not set +# CONFIG_MCP_UCB1200_AUDIO is not set +# CONFIG_MCP_UCB1200_TS is not set # # USB support # CONFIG_USB=y # CONFIG_USB_DEBUG is not set + +# +# Miscellaneous USB options +# CONFIG_USB_DEVICEFS=y # CONFIG_USB_BANDWIDTH is not set +# CONFIG_USB_LONG_TIMEOUT is not set + +# +# USB Host Controller Drivers +# +# CONFIG_USB_EHCI_HCD is not set # CONFIG_USB_UHCI is not set # CONFIG_USB_UHCI_ALT is not set -CONFIG_USB_OHCI=y -CONFIG_USB_OHCI_NOPCI=y +# CONFIG_USB_OHCI is not set +CONFIG_USB_OHCI_SA1111=y + +# +# USB Device Class drivers +# # CONFIG_USB_AUDIO is not set +# CONFIG_USB_EMI26 is not set # CONFIG_USB_BLUETOOTH is not set + +# +# SCSI support is needed for USB Storage +# # CONFIG_USB_STORAGE is not set +# CONFIG_USB_STORAGE_DEBUG is not set +# CONFIG_USB_STORAGE_DATAFAB is not set +# CONFIG_USB_STORAGE_FREECOM is not set +# CONFIG_USB_STORAGE_ISD200 is not set +# CONFIG_USB_STORAGE_DPCM is not set +# CONFIG_USB_STORAGE_HP8200e is not set +# CONFIG_USB_STORAGE_SDDR09 is not set +# CONFIG_USB_STORAGE_JUMPSHOT is not set # CONFIG_USB_ACM is not set # CONFIG_USB_PRINTER is not set -# CONFIG_USB_HID is not set -# CONFIG_USB_KBD is not set -CONFIG_USB_MOUSE=y + +# +# USB Human Interface Devices (HID) +# +CONFIG_USB_HID=y +CONFIG_USB_HIDINPUT=y +# CONFIG_USB_HIDDEV is not set # CONFIG_USB_WACOM is not set + +# +# USB Imaging devices +# # CONFIG_USB_DC2XX is not set # CONFIG_USB_MDC800 is not set # CONFIG_USB_SCANNER is not set # CONFIG_USB_MICROTEK is not set -# CONFIG_USB_IBMCAM is not set -# CONFIG_USB_OV511 is not set -# CONFIG_USB_PWC is not set -# CONFIG_USB_SE401 is not set -# CONFIG_USB_DSBR is not set -# CONFIG_USB_DABUSB is not set -# CONFIG_USB_PLUSB is not set +# CONFIG_USB_HPUSBSCSI is not set + +# +# USB Multimedia devices +# + +# +# Video4Linux support is needed for USB Multimedia device support +# + +# +# USB Network adaptors +# # CONFIG_USB_PEGASUS is not set +# CONFIG_USB_RTL8150 is not set +# CONFIG_USB_KAWETH is not set # CONFIG_USB_CATC is not set -# CONFIG_USB_NET1080 is not set +# CONFIG_USB_CDCETHER is not set # CONFIG_USB_USBNET is not set + +# +# USB port drivers +# # CONFIG_USB_USS720 is not set # # USB Serial Converter support # # CONFIG_USB_SERIAL is not set +# CONFIG_USB_SERIAL_GENERIC is not set +# CONFIG_USB_SERIAL_BELKIN is not set +# CONFIG_USB_SERIAL_WHITEHEAT is not set +# CONFIG_USB_SERIAL_DIGI_ACCELEPORT is not set +# CONFIG_USB_SERIAL_EMPEG is not set +# CONFIG_USB_SERIAL_FTDI_SIO is not set +# CONFIG_USB_SERIAL_VISOR is not set +# CONFIG_USB_SERIAL_IPAQ is not set +# CONFIG_USB_SERIAL_IR is not set +# CONFIG_USB_SERIAL_EDGEPORT is not set +# CONFIG_USB_SERIAL_KEYSPAN_PDA is not set +# CONFIG_USB_SERIAL_KEYSPAN is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XA is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XB is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA18X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19W is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA49W is not set +# CONFIG_USB_SERIAL_MCT_U232 is not set +# CONFIG_USB_SERIAL_KLSI is not set +# CONFIG_USB_SERIAL_PL2303 is not set +# CONFIG_USB_SERIAL_CYBERJACK is not set +# CONFIG_USB_SERIAL_XIRCOM is not set +# CONFIG_USB_SERIAL_OMNINET is not set + +# +# USB Miscellaneous drivers +# # CONFIG_USB_RIO500 is not set +# CONFIG_USB_AUERSWALD is not set +# CONFIG_USB_BRLVGER is not set # # Bluetooth support @@ -745,11 +1037,16 @@ # Kernel hacking # CONFIG_FRAME_POINTER=y -CONFIG_DEBUG_ERRORS=y -CONFIG_DEBUG_USER=y +# CONFIG_DEBUG_USER is not set # CONFIG_DEBUG_INFO is not set -# CONFIG_MAGIC_SYSRQ is not set # CONFIG_NO_PGT_CACHE is not set +# CONFIG_DEBUG_KERNEL is not set +# CONFIG_DEBUG_SLAB is not set +# CONFIG_MAGIC_SYSRQ is not set +# CONFIG_DEBUG_SPINLOCK is not set +# CONFIG_DEBUG_WAITQ is not set +# CONFIG_DEBUG_BUGVERBOSE is not set +# CONFIG_DEBUG_ERRORS is not set # CONFIG_DEBUG_LL is not set # CONFIG_DEBUG_DC21285_PORT is not set # CONFIG_DEBUG_CLPS711X_UART2 is not set diff -urN orig/arch/arm/def-configs/lusl7200 linux/arch/arm/def-configs/lusl7200 --- orig/arch/arm/def-configs/lusl7200 Wed May 31 13:12:12 2000 +++ linux/arch/arm/def-configs/lusl7200 Sat May 18 14:48:58 2002 @@ -23,6 +23,9 @@ # CONFIG_CPU_26 is not set CONFIG_CPU_32v4=y CONFIG_CPU_ARM720=y +CONFIG_ZBOOT_ROM=y +CONFIG_ZBOOT_ROM_TEXT=0x00010000 +CONFIG_ZBOOT_ROM_BSS=0xf03e0000 # CONFIG_PCI is not set # CONFIG_ISA is not set # CONFIG_ISA_DMA is not set diff -urN orig/arch/arm/def-configs/nanoengine linux/arch/arm/def-configs/nanoengine --- orig/arch/arm/def-configs/nanoengine Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/def-configs/nanoengine Fri Jun 21 14:13:59 2002 @@ -0,0 +1,775 @@ +# +# Automatically generated make config: don't edit +# +CONFIG_ARM=y +# CONFIG_EISA is not set +# CONFIG_SBUS is not set +# CONFIG_MCA is not set +CONFIG_UID16=y +CONFIG_RWSEM_GENERIC_SPINLOCK=y +# CONFIG_RWSEM_XCHGADD_ALGORITHM is not set +# CONFIG_GENERIC_BUST_SPINLOCK is not set +# CONFIG_GENERIC_ISA_DMA is not set + +# +# Code maturity level options +# +CONFIG_EXPERIMENTAL=y +# CONFIG_OBSOLETE is not set + +# +# Loadable module support +# +CONFIG_MODULES=y +# CONFIG_MODVERSIONS is not set +# CONFIG_KMOD is not set + +# +# System Type +# +# CONFIG_ARCH_ANAKIN is not set +# CONFIG_ARCH_ARCA5K is not set +# CONFIG_ARCH_CLPS7500 is not set +# CONFIG_ARCH_CLPS711X is not set +# CONFIG_ARCH_CO285 is not set +# CONFIG_ARCH_EBSA110 is not set +# CONFIG_ARCH_CAMELOT is not set +# CONFIG_ARCH_FOOTBRIDGE is not set +# CONFIG_ARCH_INTEGRATOR is not set +# CONFIG_ARCH_L7200 is not set +# CONFIG_ARCH_MX1ADS is not set +# CONFIG_ARCH_RPC is not set +CONFIG_ARCH_SA1100=y +# CONFIG_ARCH_SHARK is not set + +# +# Archimedes/A5000 Implementations +# + +# +# Archimedes/A5000 Implementations (select only ONE) +# +# CONFIG_ARCH_ARC is not set +# CONFIG_ARCH_A5K is not set + +# +# Footbridge Implementations +# +# CONFIG_ARCH_CATS is not set +# CONFIG_ARCH_PERSONAL_SERVER is not set +# CONFIG_ARCH_EBSA285_ADDIN is not set +# CONFIG_ARCH_EBSA285_HOST is not set +# CONFIG_ARCH_NETWINDER is not set + +# +# SA11x0 Implementations +# +# CONFIG_SA1100_ASSABET is not set +# CONFIG_ASSABET_NEPONSET is not set +# CONFIG_SA1100_ADSBITSY is not set +# CONFIG_SA1100_BRUTUS is not set +# CONFIG_SA1100_CERF is not set +# CONFIG_SA1100_H3100 is not set +# CONFIG_SA1100_H3600 is not set +# CONFIG_SA1100_H3800 is not set +# CONFIG_SA1100_H3XXX is not set +# CONFIG_SA1100_EXTENEX1 is not set +# CONFIG_SA1100_FLEXANET is not set +# CONFIG_SA1100_FREEBIRD is not set +# CONFIG_SA1100_FRODO is not set +# CONFIG_SA1100_GRAPHICSCLIENT is not set +# CONFIG_SA1100_GRAPHICSMASTER is not set +# CONFIG_SA1100_BADGE4 is not set +# CONFIG_SA1100_JORNADA720 is not set +# CONFIG_SA1100_HUW_WEBPANEL is not set +# CONFIG_SA1100_ITSY is not set +# CONFIG_SA1100_LART is not set +CONFIG_SA1100_NANOENGINE=y +# CONFIG_SA1100_OMNIMETER is not set +# CONFIG_SA1100_PANGOLIN is not set +# CONFIG_SA1100_PLEB is not set +# CONFIG_SA1100_PT_SYSTEM3 is not set +# CONFIG_SA1100_SHANNON is not set +# CONFIG_SA1100_SHERMAN is not set +# CONFIG_SA1100_SIMPAD is not set +# CONFIG_SA1100_PFS168 is not set +# CONFIG_SA1100_VICTOR is not set +# CONFIG_SA1100_XP860 is not set +# CONFIG_SA1100_YOPY is not set +# CONFIG_SA1100_USB is not set +# CONFIG_SA1100_USB_NETLINK is not set +# CONFIG_SA1100_USB_CHAR is not set +# CONFIG_H3600_SLEEVE is not set + +# +# CLPS711X/EP721X Implementations +# +# CONFIG_ARCH_AUTCPU12 is not set +# CONFIG_ARCH_CDB89712 is not set +# CONFIG_ARCH_CLEP7312 is not set +# CONFIG_ARCH_EDB7211 is not set +# CONFIG_ARCH_P720T is not set +# CONFIG_ARCH_FORTUNET is not set +# CONFIG_ARCH_EP7211 is not set +# CONFIG_ARCH_EP7212 is not set +# CONFIG_ARCH_ACORN is not set +# CONFIG_FOOTBRIDGE is not set +# CONFIG_FOOTBRIDGE_HOST is not set +# CONFIG_FOOTBRIDGE_ADDIN is not set +CONFIG_CPU_32=y +# CONFIG_CPU_26 is not set + +# +# Processor Type +# +# CONFIG_CPU_32v3 is not set +CONFIG_CPU_32v4=y +# CONFIG_CPU_ARM610 is not set +# CONFIG_CPU_ARM710 is not set +# CONFIG_CPU_ARM720T is not set +# CONFIG_CPU_ARM920T is not set +# CONFIG_CPU_ARM922T is not set +# CONFIG_CPU_ARM926T is not set +# CONFIG_CPU_ARM1020 is not set +# CONFIG_CPU_SA110 is not set +CONFIG_CPU_SA1100=y +# CONFIG_ARM_THUMB is not set +CONFIG_DISCONTIGMEM=y + +# +# General setup +# +# CONFIG_PCI is not set +CONFIG_ISA=y +# CONFIG_ISA_DMA is not set +CONFIG_ZBOOT_ROM=y +CONFIG_ZBOOT_ROM_TEXT=11000 +CONFIG_ZBOOT_ROM_BSS=c1fc0000 +# CONFIG_CPU_FREQ is not set +# CONFIG_HOTPLUG is not set +# CONFIG_PCMCIA is not set +CONFIG_NET=y +CONFIG_SYSVIPC=y +# CONFIG_BSD_PROCESS_ACCT is not set +CONFIG_SYSCTL=y + +# +# At least one math emulation must be selected +# +CONFIG_FPE_NWFPE=y +# CONFIG_FPE_FASTFPE is not set +CONFIG_KCORE_ELF=y +# CONFIG_KCORE_AOUT is not set +# CONFIG_BINFMT_AOUT is not set +CONFIG_BINFMT_ELF=y +# CONFIG_BINFMT_MISC is not set +# CONFIG_PM is not set +# CONFIG_ARTHUR is not set +CONFIG_CMDLINE="root=/dev/ram" +# CONFIG_LEDS is not set +CONFIG_ALIGNMENT_TRAP=y + +# +# Parallel port support +# +# CONFIG_PARPORT is not set + +# +# Memory Technology Devices (MTD) +# +CONFIG_MTD=y +CONFIG_MTD_DEBUG=y +CONFIG_MTD_DEBUG_VERBOSE=1 +CONFIG_MTD_PARTITIONS=y +# CONFIG_MTD_CONCAT is not set +# CONFIG_MTD_REDBOOT_PARTS is not set +# CONFIG_MTD_BOOTLDR_PARTS is not set +# CONFIG_MTD_AFS_PARTS is not set + +# +# User Modules And Translation Layers +# +CONFIG_MTD_CHAR=y +CONFIG_MTD_BLOCK=y +# CONFIG_FTL is not set +# CONFIG_NFTL is not set + +# +# RAM/ROM/Flash chip drivers +# +CONFIG_MTD_CFI=y +CONFIG_MTD_JEDECPROBE=y +CONFIG_MTD_GEN_PROBE=y +# CONFIG_MTD_CFI_ADV_OPTIONS is not set +CONFIG_MTD_CFI_INTELEXT=y +CONFIG_MTD_CFI_AMDSTD=y +# CONFIG_MTD_RAM is not set +# CONFIG_MTD_ROM is not set +# CONFIG_MTD_ABSENT is not set +# CONFIG_MTD_OBSOLETE_CHIPS is not set +# CONFIG_MTD_AMDSTD is not set +# CONFIG_MTD_SHARP is not set +# CONFIG_MTD_JEDEC is not set + +# +# Mapping drivers for chip access +# +# CONFIG_MTD_PHYSMAP is not set +# CONFIG_MTD_NORA is not set +# CONFIG_MTD_ARM_INTEGRATOR is not set +# CONFIG_MTD_CDB89712 is not set +CONFIG_MTD_SA1100=y +# CONFIG_MTD_DC21285 is not set +# CONFIG_MTD_IQ80310 is not set +# CONFIG_MTD_FORTUNET is not set +# CONFIG_MTD_EPXA10DB is not set +# CONFIG_MTD_AUTCPU12 is not set +# CONFIG_MTD_PCI is not set + +# +# Self-contained MTD device drivers +# +# CONFIG_MTD_PMC551 is not set +# CONFIG_MTD_SLRAM is not set +CONFIG_MTD_MTDRAM=y +CONFIG_MTDRAM_TOTAL_SIZE=2048 +CONFIG_MTDRAM_ERASE_SIZE=128 +CONFIG_MTDRAM_ABS_POS=0 +# CONFIG_MTD_BLKMTD is not set + +# +# Disk-On-Chip Device Drivers +# +# CONFIG_MTD_DOC1000 is not set +# CONFIG_MTD_DOC2000 is not set +# CONFIG_MTD_DOC2001 is not set +# CONFIG_MTD_DOCPROBE is not set + +# +# NAND Flash Device Drivers +# +# CONFIG_MTD_NAND is not set + +# +# Plug and Play configuration +# +# CONFIG_PNP is not set +# CONFIG_ISAPNP is not set + +# +# Block devices +# +# CONFIG_BLK_DEV_FD is not set +# CONFIG_BLK_DEV_XD is not set +# CONFIG_PARIDE is not set +# CONFIG_BLK_CPQ_DA is not set +# CONFIG_BLK_CPQ_CISS_DA is not set +# CONFIG_BLK_DEV_DAC960 is not set +CONFIG_BLK_DEV_LOOP=y +CONFIG_BLK_DEV_NBD=y +CONFIG_BLK_DEV_RAM=y +CONFIG_BLK_DEV_RAM_SIZE=8192 +CONFIG_BLK_DEV_INITRD=y + +# +# Multi-device support (RAID and LVM) +# +# CONFIG_MD is not set +# CONFIG_BLK_DEV_MD is not set +# CONFIG_MD_LINEAR is not set +# CONFIG_MD_RAID0 is not set +# CONFIG_MD_RAID1 is not set +# CONFIG_MD_RAID5 is not set +# CONFIG_MD_MULTIPATH is not set +# CONFIG_BLK_DEV_LVM is not set + +# +# Networking options +# +CONFIG_PACKET=y +# CONFIG_PACKET_MMAP is not set +# CONFIG_NETLINK_DEV is not set +# CONFIG_NETFILTER is not set +# CONFIG_FILTER is not set +CONFIG_UNIX=y +CONFIG_INET=y +# CONFIG_IP_MULTICAST is not set +# CONFIG_IP_ADVANCED_ROUTER is not set +CONFIG_IP_PNP=y +CONFIG_IP_PNP_DHCP=y +CONFIG_IP_PNP_BOOTP=y +# CONFIG_IP_PNP_RARP is not set +# CONFIG_NET_IPIP is not set +# CONFIG_NET_IPGRE is not set +# CONFIG_ARPD is not set +# CONFIG_INET_ECN is not set +# CONFIG_SYN_COOKIES is not set +# CONFIG_IPV6 is not set +# CONFIG_KHTTPD is not set +# CONFIG_ATM is not set +# CONFIG_VLAN_8021Q is not set + +# +# +# +# CONFIG_IPX is not set +# CONFIG_ATALK is not set +# CONFIG_DECNET is not set +# CONFIG_BRIDGE is not set +# CONFIG_X25 is not set +# CONFIG_LAPB is not set +# CONFIG_LLC is not set +# CONFIG_NET_DIVERT is not set +# CONFIG_ECONET is not set +# CONFIG_WAN_ROUTER is not set +# CONFIG_NET_FASTROUTE is not set +# CONFIG_NET_HW_FLOWCONTROL is not set + +# +# QoS and/or fair queueing +# +# CONFIG_NET_SCHED is not set + +# +# Network device support +# +CONFIG_NETDEVICES=y + +# +# ARCnet devices +# +# CONFIG_ARCNET is not set +# CONFIG_DUMMY is not set +# CONFIG_BONDING is not set +# CONFIG_EQUALIZER is not set +# CONFIG_TUN is not set +# CONFIG_ETHERTAP is not set + +# +# Ethernet (10 or 100Mbit) +# +CONFIG_NET_ETHERNET=y +# CONFIG_ARM_AM79C961A is not set +# CONFIG_SUNLANCE is not set +# CONFIG_SUNBMAC is not set +# CONFIG_SUNQE is not set +# CONFIG_SUNGEM is not set +# CONFIG_NET_VENDOR_3COM is not set +# CONFIG_LANCE is not set +# CONFIG_NET_VENDOR_SMC is not set +# CONFIG_NET_VENDOR_RACAL is not set +# CONFIG_AT1700 is not set +# CONFIG_DEPCA is not set +# CONFIG_HP100 is not set +# CONFIG_NET_ISA is not set +# CONFIG_NET_PCI is not set +# CONFIG_NET_POCKET is not set + +# +# Ethernet (1000 Mbit) +# +# CONFIG_ACENIC is not set +# CONFIG_DL2K is not set +# CONFIG_MYRI_SBUS is not set +# CONFIG_NS83820 is not set +# CONFIG_HAMACHI is not set +# CONFIG_YELLOWFIN is not set +# CONFIG_SK98LIN is not set +# CONFIG_FDDI is not set +# CONFIG_HIPPI is not set +# CONFIG_PLIP is not set +# CONFIG_PPP is not set +# CONFIG_SLIP is not set + +# +# Wireless LAN (non-hamradio) +# +# CONFIG_NET_RADIO is not set + +# +# Token Ring devices +# +# CONFIG_TR is not set +# CONFIG_NET_FC is not set +# CONFIG_RCPCI is not set +# CONFIG_SHAPER is not set + +# +# Wan interfaces +# +# CONFIG_WAN is not set + +# +# Amateur Radio support +# +# CONFIG_HAMRADIO is not set + +# +# IrDA (infrared) support +# +# CONFIG_IRDA is not set + +# +# ATA/IDE/MFM/RLL support +# +# CONFIG_IDE is not set +# CONFIG_BLK_DEV_IDE_MODES is not set +# CONFIG_BLK_DEV_HD is not set + +# +# SCSI support +# +# CONFIG_SCSI is not set + +# +# I2O device support +# +# CONFIG_I2O is not set +# CONFIG_I2O_BLOCK is not set +# CONFIG_I2O_LAN is not set +# CONFIG_I2O_SCSI is not set +# CONFIG_I2O_PROC is not set + +# +# ISDN subsystem +# +# CONFIG_ISDN is not set + +# +# Input core support +# +# CONFIG_INPUT is not set +# CONFIG_INPUT_KEYBDEV is not set +# CONFIG_INPUT_MOUSEDEV is not set +# CONFIG_INPUT_JOYDEV is not set +# CONFIG_INPUT_EVDEV is not set + +# +# Character devices +# +# CONFIG_VT is not set +# CONFIG_SERIAL is not set +# CONFIG_SERIAL_EXTENDED is not set +# CONFIG_SERIAL_NONSTANDARD is not set + +# +# Serial drivers +# +# CONFIG_SERIAL_ANAKIN is not set +# CONFIG_SERIAL_ANAKIN_CONSOLE is not set +# CONFIG_SERIAL_AMBA is not set +# CONFIG_SERIAL_AMBA_CONSOLE is not set +# CONFIG_SERIAL_CLPS711X is not set +# CONFIG_SERIAL_CLPS711X_CONSOLE is not set +# CONFIG_SERIAL_21285 is not set +# CONFIG_SERIAL_21285_OLD is not set +# CONFIG_SERIAL_21285_CONSOLE is not set +# CONFIG_SERIAL_UART00 is not set +# CONFIG_SERIAL_UART00_CONSOLE is not set +CONFIG_SERIAL_SA1100=y +CONFIG_SERIAL_SA1100_CONSOLE=y +CONFIG_SA1100_DEFAULT_BAUDRATE=9600 +# CONFIG_SERIAL_8250 is not set +# CONFIG_SERIAL_8250_CONSOLE is not set +# CONFIG_SERIAL_8250_EXTENDED is not set +# CONFIG_SERIAL_8250_MANY_PORTS is not set +# CONFIG_SERIAL_8250_SHARE_IRQ is not set +# CONFIG_SERIAL_8250_DETECT_IRQ is not set +# CONFIG_SERIAL_8250_MULTIPORT is not set +# CONFIG_SERIAL_8250_HUB6 is not set +CONFIG_SERIAL_CORE=y +CONFIG_SERIAL_CORE_CONSOLE=y +CONFIG_UNIX98_PTYS=y +CONFIG_UNIX98_PTY_COUNT=64 + +# +# I2C support +# +# CONFIG_I2C is not set + +# +# L3 serial bus support +# +# CONFIG_L3 is not set +# CONFIG_L3_ALGOBIT is not set +# CONFIG_L3_BIT_SA1100_GPIO is not set + +# +# Other L3 adapters +# +# CONFIG_L3_SA1111 is not set +# CONFIG_BIT_SA1100_GPIO is not set + +# +# Mice +# +# CONFIG_BUSMOUSE is not set +# CONFIG_MOUSE is not set + +# +# Joysticks +# +# CONFIG_INPUT_GAMEPORT is not set + +# +# Input core support is needed for gameports +# + +# +# Input core support is needed for joysticks +# +# CONFIG_QIC02_TAPE is not set + +# +# Watchdog Cards +# +# CONFIG_WATCHDOG is not set +# CONFIG_INTEL_RNG is not set +# CONFIG_NVRAM is not set +# CONFIG_RTC is not set +CONFIG_SA1100_RTC=y +# CONFIG_DTLK is not set +# CONFIG_R3964 is not set +# CONFIG_APPLICOM is not set + +# +# Ftape, the floppy tape device driver +# +# CONFIG_FTAPE is not set +# CONFIG_AGP is not set +# CONFIG_DRM is not set + +# +# Multimedia devices +# +# CONFIG_VIDEO_DEV is not set + +# +# File systems +# +# CONFIG_QUOTA is not set +# CONFIG_AUTOFS_FS is not set +# CONFIG_AUTOFS4_FS is not set +# CONFIG_REISERFS_FS is not set +# CONFIG_REISERFS_CHECK is not set +# CONFIG_REISERFS_PROC_INFO is not set +# CONFIG_ADFS_FS is not set +# CONFIG_ADFS_FS_RW is not set +# CONFIG_AFFS_FS is not set +# CONFIG_HFS_FS is not set +# CONFIG_BFS_FS is not set +# CONFIG_EXT3_FS is not set +# CONFIG_JBD is not set +# CONFIG_JBD_DEBUG is not set +# CONFIG_FAT_FS is not set +# CONFIG_MSDOS_FS is not set +# CONFIG_UMSDOS_FS is not set +# CONFIG_VFAT_FS is not set +# CONFIG_EFS_FS is not set +# CONFIG_JFFS_FS is not set +CONFIG_JFFS2_FS=y +CONFIG_JFFS2_FS_DEBUG=1 +# CONFIG_CRAMFS is not set +# CONFIG_TMPFS is not set +# CONFIG_RAMFS is not set +# CONFIG_ISO9660_FS is not set +# CONFIG_JOLIET is not set +# CONFIG_ZISOFS is not set +# CONFIG_MINIX_FS is not set +# CONFIG_VXFS_FS is not set +# CONFIG_NTFS_FS is not set +# CONFIG_NTFS_RW is not set +# CONFIG_HPFS_FS is not set +CONFIG_PROC_FS=y +# CONFIG_DEVFS_FS is not set +# CONFIG_DEVFS_MOUNT is not set +# CONFIG_DEVFS_DEBUG is not set +CONFIG_DEVPTS_FS=y +# CONFIG_QNX4FS_FS is not set +# CONFIG_QNX4FS_RW is not set +# CONFIG_ROMFS_FS is not set +CONFIG_EXT2_FS=y +# CONFIG_SYSV_FS is not set +# CONFIG_UDF_FS is not set +# CONFIG_UDF_RW is not set +# CONFIG_UFS_FS is not set +# CONFIG_UFS_FS_WRITE is not set + +# +# Network File Systems +# +# CONFIG_CODA_FS is not set +# CONFIG_INTERMEZZO_FS is not set +CONFIG_NFS_FS=y +# CONFIG_NFS_V3 is not set +CONFIG_ROOT_NFS=y +# CONFIG_NFSD is not set +# CONFIG_NFSD_V3 is not set +CONFIG_SUNRPC=y +CONFIG_LOCKD=y +# CONFIG_SMB_FS is not set +# CONFIG_NCP_FS is not set +# CONFIG_NCPFS_PACKET_SIGNING is not set +# CONFIG_NCPFS_IOCTL_LOCKING is not set +# CONFIG_NCPFS_STRONG is not set +# CONFIG_NCPFS_NFS_NS is not set +# CONFIG_NCPFS_OS2_NS is not set +# CONFIG_NCPFS_SMALLDOS is not set +# CONFIG_NCPFS_NLS is not set +# CONFIG_NCPFS_EXTRAS is not set +# CONFIG_ZISOFS_FS is not set +# CONFIG_ZLIB_FS_INFLATE is not set + +# +# Partition Types +# +# CONFIG_PARTITION_ADVANCED is not set +CONFIG_MSDOS_PARTITION=y +# CONFIG_SMB_NLS is not set +# CONFIG_NLS is not set + +# +# Sound +# +# CONFIG_SOUND is not set + +# +# Multimedia Capabilities Port drivers +# +# CONFIG_MCP is not set +# CONFIG_MCP_SA1100 is not set +# CONFIG_MCP_UCB1200 is not set +# CONFIG_MCP_UCB1200_AUDIO is not set +# CONFIG_MCP_UCB1200_TS is not set + +# +# USB support +# +# CONFIG_USB is not set + +# +# USB Controllers +# +# CONFIG_USB_UHCI is not set +# CONFIG_USB_UHCI_ALT is not set +# CONFIG_USB_OHCI is not set +# CONFIG_USB_OHCI_SA1111 is not set + +# +# USB Device Class drivers +# +# CONFIG_USB_AUDIO is not set +# CONFIG_USB_BLUETOOTH is not set + +# +# SCSI support is needed for USB Storage +# +# CONFIG_USB_STORAGE is not set +# CONFIG_USB_STORAGE_DEBUG is not set +# CONFIG_USB_STORAGE_DATAFAB is not set +# CONFIG_USB_STORAGE_FREECOM is not set +# CONFIG_USB_STORAGE_ISD200 is not set +# CONFIG_USB_STORAGE_DPCM is not set +# CONFIG_USB_STORAGE_HP8200e is not set +# CONFIG_USB_STORAGE_SDDR09 is not set +# CONFIG_USB_STORAGE_JUMPSHOT is not set +# CONFIG_USB_ACM is not set +# CONFIG_USB_PRINTER is not set + +# +# USB Human Interface Devices (HID) +# + +# +# Input core support is needed for USB HID +# + +# +# USB Imaging devices +# +# CONFIG_USB_DC2XX is not set +# CONFIG_USB_MDC800 is not set +# CONFIG_USB_SCANNER is not set +# CONFIG_USB_MICROTEK is not set +# CONFIG_USB_HPUSBSCSI is not set + +# +# USB Multimedia devices +# + +# +# Video4Linux support is needed for USB Multimedia device support +# + +# +# USB Network adaptors +# +# CONFIG_USB_PEGASUS is not set +# CONFIG_USB_KAWETH is not set +# CONFIG_USB_CATC is not set +# CONFIG_USB_CDCETHER is not set +# CONFIG_USB_USBNET is not set + +# +# USB port drivers +# +# CONFIG_USB_USS720 is not set + +# +# USB Serial Converter support +# +# CONFIG_USB_SERIAL is not set +# CONFIG_USB_SERIAL_GENERIC is not set +# CONFIG_USB_SERIAL_BELKIN is not set +# CONFIG_USB_SERIAL_WHITEHEAT is not set +# CONFIG_USB_SERIAL_DIGI_ACCELEPORT is not set +# CONFIG_USB_SERIAL_EMPEG is not set +# CONFIG_USB_SERIAL_FTDI_SIO is not set +# CONFIG_USB_SERIAL_VISOR is not set +# CONFIG_USB_SERIAL_IPAQ is not set +# CONFIG_USB_SERIAL_IR is not set +# CONFIG_USB_SERIAL_EDGEPORT is not set +# CONFIG_USB_SERIAL_KEYSPAN_PDA is not set +# CONFIG_USB_SERIAL_KEYSPAN is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XA is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XB is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA18X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19W is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA49W is not set +# CONFIG_USB_SERIAL_MCT_U232 is not set +# CONFIG_USB_SERIAL_KLSI is not set +# CONFIG_USB_SERIAL_PL2303 is not set +# CONFIG_USB_SERIAL_CYBERJACK is not set +# CONFIG_USB_SERIAL_XIRCOM is not set +# CONFIG_USB_SERIAL_OMNINET is not set + +# +# USB Miscellaneous drivers +# +# CONFIG_USB_RIO500 is not set + +# +# Bluetooth support +# +# CONFIG_BLUEZ is not set + +# +# Kernel hacking +# +CONFIG_FRAME_POINTER=y +CONFIG_DEBUG_USER=y +# CONFIG_DEBUG_INFO is not set +# CONFIG_NO_PGT_CACHE is not set +# CONFIG_DEBUG_KERNEL is not set +# CONFIG_DEBUG_SLAB is not set +# CONFIG_MAGIC_SYSRQ is not set +# CONFIG_DEBUG_SPINLOCK is not set +# CONFIG_DEBUG_WAITQ is not set +# CONFIG_DEBUG_BUGVERBOSE is not set +# CONFIG_DEBUG_ERRORS is not set +# CONFIG_DEBUG_LL is not set +# CONFIG_DEBUG_DC21285_PORT is not set +# CONFIG_DEBUG_CLPS711X_UART2 is not set diff -urN orig/arch/arm/def-configs/omaha linux/arch/arm/def-configs/omaha --- orig/arch/arm/def-configs/omaha Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/def-configs/omaha Thu Oct 24 14:58:03 2002 @@ -0,0 +1,798 @@ +# +# Automatically generated make config: don't edit +# +CONFIG_ARM=y +# CONFIG_EISA is not set +# CONFIG_SBUS is not set +# CONFIG_MCA is not set +CONFIG_UID16=y +CONFIG_RWSEM_GENERIC_SPINLOCK=y +# CONFIG_RWSEM_XCHGADD_ALGORITHM is not set +# CONFIG_GENERIC_BUST_SPINLOCK is not set +# CONFIG_GENERIC_ISA_DMA is not set + +# +# Code maturity level options +# +# CONFIG_EXPERIMENTAL is not set +# CONFIG_OBSOLETE is not set + +# +# Loadable module support +# +# CONFIG_MODULES is not set +# CONFIG_MODVERSIONS is not set +# CONFIG_KMOD is not set + +# +# System Type +# +# CONFIG_ARCH_ANAKIN is not set +# CONFIG_ARCH_ARCA5K is not set +# CONFIG_ARCH_CLPS7500 is not set +# CONFIG_ARCH_CLPS711X is not set +# CONFIG_ARCH_CO285 is not set +# CONFIG_ARCH_EBSA110 is not set +# CONFIG_ARCH_CAMELOT is not set +# CONFIG_ARCH_FOOTBRIDGE is not set +# CONFIG_ARCH_INTEGRATOR is not set +CONFIG_ARCH_OMAHA=y +# CONFIG_ARCH_L7200 is not set +# CONFIG_ARCH_MX1ADS is not set +# CONFIG_ARCH_RPC is not set +# CONFIG_ARCH_SA1100 is not set +# CONFIG_ARCH_SHARK is not set + +# +# Archimedes/A5000 Implementations +# + +# +# Archimedes/A5000 Implementations (select only ONE) +# +# CONFIG_ARCH_ARC is not set +# CONFIG_ARCH_A5K is not set + +# +# Footbridge Implementations +# +# CONFIG_ARCH_CATS is not set +# CONFIG_ARCH_PERSONAL_SERVER is not set +# CONFIG_ARCH_EBSA285_ADDIN is not set +# CONFIG_ARCH_EBSA285_HOST is not set +# CONFIG_ARCH_NETWINDER is not set + +# +# SA11x0 Implementations +# +# CONFIG_SA1100_ASSABET is not set +# CONFIG_ASSABET_NEPONSET is not set +# CONFIG_SA1100_ADSBITSY is not set +# CONFIG_SA1100_BRUTUS is not set +# CONFIG_SA1100_CERF is not set +# CONFIG_SA1100_H3100 is not set +# CONFIG_SA1100_H3600 is not set +# CONFIG_SA1100_H3800 is not set +# CONFIG_SA1100_H3XXX is not set +# CONFIG_SA1100_EXTENEX1 is not set +# CONFIG_SA1100_FLEXANET is not set +# CONFIG_SA1100_FREEBIRD is not set +# CONFIG_SA1100_FRODO is not set +# CONFIG_SA1100_GRAPHICSCLIENT is not set +# CONFIG_SA1100_GRAPHICSMASTER is not set +# CONFIG_SA1100_BADGE4 is not set +# CONFIG_SA1100_JORNADA720 is not set +# CONFIG_SA1100_HUW_WEBPANEL is not set +# CONFIG_SA1100_ITSY is not set +# CONFIG_SA1100_LART is not set +# CONFIG_SA1100_NANOENGINE is not set +# CONFIG_SA1100_OMNIMETER is not set +# CONFIG_SA1100_PANGOLIN is not set +# CONFIG_SA1100_PLEB is not set +# CONFIG_SA1100_PT_SYSTEM3 is not set +# CONFIG_SA1100_SHANNON is not set +# CONFIG_SA1100_SHERMAN is not set +# CONFIG_SA1100_SIMPAD is not set +# CONFIG_SA1100_PFS168 is not set +# CONFIG_SA1100_VICTOR is not set +# CONFIG_SA1100_XP860 is not set +# CONFIG_SA1100_YOPY is not set +# CONFIG_SA1100_USB is not set +# CONFIG_SA1100_USB_NETLINK is not set +# CONFIG_SA1100_USB_CHAR is not set +# CONFIG_H3600_SLEEVE is not set + +# +# CLPS711X/EP721X Implementations +# +# CONFIG_ARCH_AUTCPU12 is not set +# CONFIG_ARCH_CDB89712 is not set +# CONFIG_ARCH_CLEP7312 is not set +# CONFIG_ARCH_EDB7211 is not set +# CONFIG_ARCH_P720T is not set +# CONFIG_ARCH_FORTUNET is not set +# CONFIG_ARCH_EP7211 is not set +# CONFIG_ARCH_EP7212 is not set +# CONFIG_ARCH_ACORN is not set +# CONFIG_FOOTBRIDGE is not set +# CONFIG_FOOTBRIDGE_HOST is not set +# CONFIG_FOOTBRIDGE_ADDIN is not set +CONFIG_CPU_32=y +# CONFIG_CPU_26 is not set + +# +# Processor Type +# +# CONFIG_CPU_32v3 is not set +CONFIG_CPU_32v4=y +# CONFIG_CPU_ARM610 is not set +# CONFIG_CPU_ARM710 is not set +# CONFIG_CPU_ARM720T is not set +CONFIG_CPU_ARM920T=y +CONFIG_CPU_ARM920_CPU_IDLE=y +CONFIG_CPU_ARM920_I_CACHE_ON=y +CONFIG_CPU_ARM920_D_CACHE_ON=y +# CONFIG_CPU_ARM920_WRITETHROUGH is not set +# CONFIG_CPU_ARM922T is not set +# CONFIG_CPU_ARM926T is not set +# CONFIG_CPU_ARM1020 is not set +# CONFIG_CPU_SA110 is not set +# CONFIG_CPU_SA1100 is not set +# CONFIG_ARM_THUMB is not set +# CONFIG_DISCONTIGMEM is not set + +# +# General setup +# +# CONFIG_PCI is not set +# CONFIG_ISA is not set +# CONFIG_ISA_DMA is not set +# CONFIG_HOTPLUG is not set +# CONFIG_PCMCIA is not set +CONFIG_NET=y +CONFIG_SYSVIPC=y +CONFIG_BSD_PROCESS_ACCT=y +CONFIG_SYSCTL=y + +# +# At least one math emulation must be selected +# +CONFIG_FPE_NWFPE=y +# CONFIG_FPE_FASTFPE is not set +# CONFIG_VFP is not set +CONFIG_KCORE_ELF=y +# CONFIG_KCORE_AOUT is not set +CONFIG_BINFMT_AOUT=y +CONFIG_BINFMT_ELF=y +CONFIG_BINFMT_MISC=y +# CONFIG_PM is not set +# CONFIG_ARTHUR is not set +# CONFIG_LEDS is not set +CONFIG_ALIGNMENT_TRAP=y + +# +# Parallel port support +# +# CONFIG_PARPORT is not set + +# +# Memory Technology Devices (MTD) +# +# CONFIG_MTD is not set + +# +# Plug and Play configuration +# +# CONFIG_PNP is not set +# CONFIG_ISAPNP is not set + +# +# Block devices +# +# CONFIG_BLK_DEV_FD is not set +# CONFIG_BLK_DEV_XD is not set +# CONFIG_PARIDE is not set +# CONFIG_BLK_CPQ_DA is not set +# CONFIG_BLK_CPQ_CISS_DA is not set +# CONFIG_BLK_DEV_DAC960 is not set +CONFIG_BLK_DEV_LOOP=y +# CONFIG_BLK_DEV_NBD is not set +CONFIG_BLK_DEV_RAM=y +CONFIG_BLK_DEV_RAM_SIZE=4096 +# CONFIG_BLK_DEV_INITRD is not set + +# +# Multi-device support (RAID and LVM) +# +# CONFIG_MD is not set +# CONFIG_BLK_DEV_MD is not set +# CONFIG_MD_LINEAR is not set +# CONFIG_MD_RAID0 is not set +# CONFIG_MD_RAID1 is not set +# CONFIG_MD_RAID5 is not set +# CONFIG_MD_MULTIPATH is not set +# CONFIG_BLK_DEV_LVM is not set + +# +# Networking options +# +CONFIG_PACKET=y +# CONFIG_PACKET_MMAP is not set +# CONFIG_NETLINK_DEV is not set +# CONFIG_NETFILTER is not set +# CONFIG_FILTER is not set +CONFIG_UNIX=y +CONFIG_INET=y +# CONFIG_IP_MULTICAST is not set +# CONFIG_IP_ADVANCED_ROUTER is not set +# CONFIG_IP_PNP is not set +# CONFIG_NET_IPIP is not set +# CONFIG_NET_IPGRE is not set +# CONFIG_INET_ECN is not set +# CONFIG_SYN_COOKIES is not set + +# +# +# +# CONFIG_IPX is not set +# CONFIG_ATALK is not set +# CONFIG_DECNET is not set +# CONFIG_BRIDGE is not set + +# +# QoS and/or fair queueing +# +# CONFIG_NET_SCHED is not set + +# +# Network device support +# +CONFIG_NETDEVICES=y + +# +# ARCnet devices +# +# CONFIG_ARCNET is not set +# CONFIG_DUMMY is not set +# CONFIG_BONDING is not set +# CONFIG_EQUALIZER is not set +# CONFIG_TUN is not set + +# +# Ethernet (10 or 100Mbit) +# +CONFIG_NET_ETHERNET=y +# CONFIG_ARM_AM79C961A is not set +CONFIG_SMC91C111=y +# CONFIG_SUNLANCE is not set +# CONFIG_SUNBMAC is not set +# CONFIG_SUNQE is not set +# CONFIG_SUNGEM is not set +# CONFIG_NET_VENDOR_3COM is not set +# CONFIG_LANCE is not set +# CONFIG_NET_VENDOR_SMC is not set +# CONFIG_NET_VENDOR_RACAL is not set +# CONFIG_NET_ISA is not set +# CONFIG_NET_PCI is not set +# CONFIG_NET_POCKET is not set + +# +# Ethernet (1000 Mbit) +# +# CONFIG_ACENIC is not set +# CONFIG_DL2K is not set +# CONFIG_MYRI_SBUS is not set +# CONFIG_NS83820 is not set +# CONFIG_HAMACHI is not set +# CONFIG_YELLOWFIN is not set +# CONFIG_SK98LIN is not set +# CONFIG_FDDI is not set +# CONFIG_PLIP is not set +# CONFIG_PPP is not set +# CONFIG_SLIP is not set + +# +# Wireless LAN (non-hamradio) +# +# CONFIG_NET_RADIO is not set + +# +# Token Ring devices +# +# CONFIG_TR is not set +# CONFIG_NET_FC is not set + +# +# Wan interfaces +# +# CONFIG_WAN is not set + +# +# Amateur Radio support +# +# CONFIG_HAMRADIO is not set + +# +# IrDA (infrared) support +# +# CONFIG_IRDA is not set + +# +# ATA/IDE/MFM/RLL support +# +CONFIG_IDE=y + +# +# IDE, ATA and ATAPI Block devices +# +CONFIG_BLK_DEV_IDE=y + +# +# Please see Documentation/ide.txt for help/info on IDE drives +# +# CONFIG_BLK_DEV_HD_IDE is not set +# CONFIG_BLK_DEV_HD is not set +CONFIG_BLK_DEV_IDEDISK=y +CONFIG_IDEDISK_MULTI_MODE=y +# CONFIG_BLK_DEV_IDEDISK_VENDOR is not set +# CONFIG_BLK_DEV_IDEDISK_FUJITSU is not set +# CONFIG_BLK_DEV_IDEDISK_IBM is not set +# CONFIG_BLK_DEV_IDEDISK_MAXTOR is not set +# CONFIG_BLK_DEV_IDEDISK_QUANTUM is not set +# CONFIG_BLK_DEV_IDEDISK_SEAGATE is not set +# CONFIG_BLK_DEV_IDEDISK_WD is not set +# CONFIG_BLK_DEV_COMMERIAL is not set +# CONFIG_BLK_DEV_TIVO is not set +# CONFIG_BLK_DEV_IDECS is not set +# CONFIG_BLK_DEV_IDECD is not set +# CONFIG_BLK_DEV_IDETAPE is not set +# CONFIG_BLK_DEV_IDEFLOPPY is not set +# CONFIG_BLK_DEV_IDESCSI is not set + +# +# IDE chipset support/bugfixes +# +# CONFIG_BLK_DEV_CMD640 is not set +# CONFIG_BLK_DEV_CMD640_ENHANCED is not set +# CONFIG_BLK_DEV_ISAPNP is not set +# CONFIG_IDE_CHIPSETS is not set +# CONFIG_IDEDMA_AUTO is not set +# CONFIG_DMA_NONPCI is not set +# CONFIG_BLK_DEV_IDE_MODES is not set +# CONFIG_BLK_DEV_ATARAID is not set +# CONFIG_BLK_DEV_ATARAID_PDC is not set +# CONFIG_BLK_DEV_ATARAID_HPT is not set + +# +# SCSI support +# +# CONFIG_SCSI is not set + +# +# I2O device support +# +# CONFIG_I2O is not set +# CONFIG_I2O_BLOCK is not set +# CONFIG_I2O_LAN is not set +# CONFIG_I2O_SCSI is not set +# CONFIG_I2O_PROC is not set + +# +# ISDN subsystem +# +# CONFIG_ISDN is not set + +# +# Input core support +# +# CONFIG_INPUT is not set +# CONFIG_INPUT_KEYBDEV is not set +# CONFIG_INPUT_MOUSEDEV is not set +# CONFIG_INPUT_JOYDEV is not set +# CONFIG_INPUT_EVDEV is not set + +# +# Character devices +# +# CONFIG_VT is not set +# CONFIG_SERIAL is not set +# CONFIG_SERIAL_EXTENDED is not set +# CONFIG_SERIAL_NONSTANDARD is not set +# CONFIG_ARM_DCC is not set + +# +# Serial drivers +# +# CONFIG_SERIAL_ANAKIN is not set +# CONFIG_SERIAL_ANAKIN_CONSOLE is not set +# CONFIG_SERIAL_AMBA is not set +# CONFIG_SERIAL_AMBA_CONSOLE is not set +# CONFIG_SERIAL_CLPS711X is not set +# CONFIG_SERIAL_CLPS711X_CONSOLE is not set +# CONFIG_SERIAL_21285 is not set +# CONFIG_SERIAL_21285_OLD is not set +# CONFIG_SERIAL_21285_CONSOLE is not set +# CONFIG_SERIAL_UART00 is not set +# CONFIG_SERIAL_UART00_CONSOLE is not set +# CONFIG_SERIAL_SA1100 is not set +# CONFIG_SERIAL_SA1100_CONSOLE is not set +# CONFIG_SERIAL_OMAHA is not set +# CONFIG_SERIAL_OMAHA_CONSOLE is not set +# CONFIG_SERIAL_8250 is not set +# CONFIG_SERIAL_8250_CONSOLE is not set +# CONFIG_SERIAL_8250_EXTENDED is not set +# CONFIG_SERIAL_8250_MANY_PORTS is not set +# CONFIG_SERIAL_8250_SHARE_IRQ is not set +# CONFIG_SERIAL_8250_DETECT_IRQ is not set +# CONFIG_SERIAL_8250_MULTIPORT is not set +# CONFIG_SERIAL_8250_HUB6 is not set +CONFIG_UNIX98_PTYS=y +CONFIG_UNIX98_PTY_COUNT=256 + +# +# I2C support +# +CONFIG_I2C=y +CONFIG_I2C_ALGOBIT=y +# CONFIG_I2C_PHILIPSPAR is not set +# CONFIG_I2C_ELV is not set +# CONFIG_I2C_VELLEMAN is not set +CONFIG_I2C_OMAHA=y +# CONFIG_I2C_ALGOPCF is not set +CONFIG_I2C_CHARDEV=y +CONFIG_I2C_PROC=y + +# +# L3 serial bus support +# +# CONFIG_L3 is not set +# CONFIG_L3_ALGOBIT is not set +# CONFIG_L3_BIT_SA1100_GPIO is not set + +# +# Other L3 adapters +# +# CONFIG_L3_SA1111 is not set +# CONFIG_BIT_SA1100_GPIO is not set + +# +# Mice +# +# CONFIG_BUSMOUSE is not set +# CONFIG_MOUSE is not set + +# +# Joysticks +# +# CONFIG_INPUT_GAMEPORT is not set + +# +# Input core support is needed for gameports +# + +# +# Input core support is needed for joysticks +# +# CONFIG_QIC02_TAPE is not set + +# +# Watchdog Cards +# +CONFIG_WATCHDOG=y +# CONFIG_WATCHDOG_NOWAYOUT is not set +# CONFIG_SOFT_WATCHDOG is not set +# CONFIG_WDT is not set +# CONFIG_WDTPCI is not set +# CONFIG_PCWATCHDOG is not set +# CONFIG_ACQUIRE_WDT is not set +# CONFIG_ADVANTECH_WDT is not set +# CONFIG_21285_WATCHDOG is not set +# CONFIG_977_WATCHDOG is not set +# CONFIG_SA1100_WATCHDOG is not set +CONFIG_OMAHA_WATCHDOG=y +# CONFIG_EUROTECH_WDT is not set +# CONFIG_IB700_WDT is not set +# CONFIG_I810_TCO is not set +# CONFIG_MIXCOMWD is not set +# CONFIG_60XX_WDT is not set +# CONFIG_W83877F_WDT is not set +# CONFIG_MACHZ_WDT is not set +# CONFIG_INTEL_RNG is not set +# CONFIG_NVRAM is not set +# CONFIG_RTC is not set +CONFIG_OMAHA_RTC=y +# CONFIG_DTLK is not set +# CONFIG_R3964 is not set +# CONFIG_APPLICOM is not set + +# +# Ftape, the floppy tape device driver +# +# CONFIG_FTAPE is not set +# CONFIG_AGP is not set +# CONFIG_DRM is not set + +# +# Multimedia devices +# +# CONFIG_VIDEO_DEV is not set + +# +# File systems +# +# CONFIG_QUOTA is not set +# CONFIG_AUTOFS_FS is not set +# CONFIG_AUTOFS4_FS is not set +# CONFIG_REISERFS_FS is not set +# CONFIG_REISERFS_CHECK is not set +# CONFIG_REISERFS_PROC_INFO is not set +# CONFIG_ADFS_FS is not set +# CONFIG_ADFS_FS_RW is not set +# CONFIG_AFFS_FS is not set +# CONFIG_HFS_FS is not set +# CONFIG_BFS_FS is not set +# CONFIG_EXT3_FS is not set +# CONFIG_JBD is not set +# CONFIG_JBD_DEBUG is not set +CONFIG_FAT_FS=y +CONFIG_MSDOS_FS=y +CONFIG_UMSDOS_FS=y +CONFIG_VFAT_FS=y +# CONFIG_EFS_FS is not set +# CONFIG_JFFS_FS is not set +# CONFIG_JFFS2_FS is not set +# CONFIG_CRAMFS is not set +CONFIG_TMPFS=y +CONFIG_RAMFS=y +# CONFIG_ISO9660_FS is not set +# CONFIG_JOLIET is not set +# CONFIG_ZISOFS is not set +CONFIG_MINIX_FS=y +# CONFIG_VXFS_FS is not set +# CONFIG_NTFS_FS is not set +# CONFIG_NTFS_RW is not set +# CONFIG_HPFS_FS is not set +CONFIG_PROC_FS=y +# CONFIG_DEVFS_FS is not set +# CONFIG_DEVFS_MOUNT is not set +# CONFIG_DEVFS_DEBUG is not set +CONFIG_DEVPTS_FS=y +# CONFIG_QNX4FS_FS is not set +# CONFIG_QNX4FS_RW is not set +# CONFIG_ROMFS_FS is not set +CONFIG_EXT2_FS=y +# CONFIG_SYSV_FS is not set +# CONFIG_UDF_FS is not set +# CONFIG_UDF_RW is not set +# CONFIG_UFS_FS is not set +# CONFIG_UFS_FS_WRITE is not set + +# +# Network File Systems +# +# CONFIG_CODA_FS is not set +# CONFIG_INTERMEZZO_FS is not set +CONFIG_NFS_FS=y +CONFIG_NFS_V3=y +# CONFIG_ROOT_NFS is not set +# CONFIG_NFSD is not set +# CONFIG_NFSD_V3 is not set +CONFIG_SUNRPC=y +CONFIG_LOCKD=y +CONFIG_LOCKD_V4=y +# CONFIG_SMB_FS is not set +# CONFIG_NCP_FS is not set +# CONFIG_NCPFS_PACKET_SIGNING is not set +# CONFIG_NCPFS_IOCTL_LOCKING is not set +# CONFIG_NCPFS_STRONG is not set +# CONFIG_NCPFS_NFS_NS is not set +# CONFIG_NCPFS_OS2_NS is not set +# CONFIG_NCPFS_SMALLDOS is not set +# CONFIG_NCPFS_NLS is not set +# CONFIG_NCPFS_EXTRAS is not set +# CONFIG_ZISOFS_FS is not set +# CONFIG_ZLIB_FS_INFLATE is not set + +# +# Partition Types +# +CONFIG_PARTITION_ADVANCED=y +# CONFIG_ACORN_PARTITION is not set +# CONFIG_OSF_PARTITION is not set +# CONFIG_AMIGA_PARTITION is not set +# CONFIG_ATARI_PARTITION is not set +# CONFIG_MAC_PARTITION is not set +CONFIG_MSDOS_PARTITION=y +# CONFIG_BSD_DISKLABEL is not set +# CONFIG_MINIX_SUBPARTITION is not set +# CONFIG_SOLARIS_X86_PARTITION is not set +# CONFIG_UNIXWARE_DISKLABEL is not set +# CONFIG_LDM_PARTITION is not set +# CONFIG_SGI_PARTITION is not set +# CONFIG_ULTRIX_PARTITION is not set +# CONFIG_SUN_PARTITION is not set +# CONFIG_SMB_NLS is not set +CONFIG_NLS=y + +# +# Native Language Support +# +CONFIG_NLS_DEFAULT="iso8859-1" +# CONFIG_NLS_CODEPAGE_437 is not set +# CONFIG_NLS_CODEPAGE_737 is not set +# CONFIG_NLS_CODEPAGE_775 is not set +# CONFIG_NLS_CODEPAGE_850 is not set +# CONFIG_NLS_CODEPAGE_852 is not set +# CONFIG_NLS_CODEPAGE_855 is not set +# CONFIG_NLS_CODEPAGE_857 is not set +# CONFIG_NLS_CODEPAGE_860 is not set +# CONFIG_NLS_CODEPAGE_861 is not set +# CONFIG_NLS_CODEPAGE_862 is not set +# CONFIG_NLS_CODEPAGE_863 is not set +# CONFIG_NLS_CODEPAGE_864 is not set +# CONFIG_NLS_CODEPAGE_865 is not set +# CONFIG_NLS_CODEPAGE_866 is not set +# CONFIG_NLS_CODEPAGE_869 is not set +# CONFIG_NLS_CODEPAGE_936 is not set +# CONFIG_NLS_CODEPAGE_950 is not set +# CONFIG_NLS_CODEPAGE_932 is not set +# CONFIG_NLS_CODEPAGE_949 is not set +# CONFIG_NLS_CODEPAGE_874 is not set +# CONFIG_NLS_ISO8859_8 is not set +# CONFIG_NLS_CODEPAGE_1250 is not set +# CONFIG_NLS_CODEPAGE_1251 is not set +# CONFIG_NLS_ISO8859_1 is not set +# CONFIG_NLS_ISO8859_2 is not set +# CONFIG_NLS_ISO8859_3 is not set +# CONFIG_NLS_ISO8859_4 is not set +# CONFIG_NLS_ISO8859_5 is not set +# CONFIG_NLS_ISO8859_6 is not set +# CONFIG_NLS_ISO8859_7 is not set +# CONFIG_NLS_ISO8859_9 is not set +# CONFIG_NLS_ISO8859_13 is not set +# CONFIG_NLS_ISO8859_14 is not set +# CONFIG_NLS_ISO8859_15 is not set +# CONFIG_NLS_KOI8_R is not set +# CONFIG_NLS_KOI8_U is not set +# CONFIG_NLS_UTF8 is not set + +# +# Multimedia Capabilities Port drivers +# +# CONFIG_MCP is not set +# CONFIG_MCP_SA1100 is not set +# CONFIG_MCP_UCB1200 is not set +# CONFIG_MCP_UCB1200_AUDIO is not set +# CONFIG_MCP_UCB1200_TS is not set + +# +# Omaha /proc drivers +# +CONFIG_OMAHA_MISC=y + +# +# USB support +# +# CONFIG_USB is not set + +# +# USB Controllers +# +# CONFIG_USB_UHCI is not set +# CONFIG_USB_UHCI_ALT is not set +# CONFIG_USB_OHCI is not set +# CONFIG_USB_OHCI_SA1111 is not set + +# +# USB Device Class drivers +# +# CONFIG_USB_AUDIO is not set +# CONFIG_USB_BLUETOOTH is not set + +# +# SCSI support is needed for USB Storage +# +# CONFIG_USB_STORAGE is not set +# CONFIG_USB_STORAGE_DEBUG is not set +# CONFIG_USB_STORAGE_DATAFAB is not set +# CONFIG_USB_STORAGE_FREECOM is not set +# CONFIG_USB_STORAGE_ISD200 is not set +# CONFIG_USB_STORAGE_DPCM is not set +# CONFIG_USB_STORAGE_HP8200e is not set +# CONFIG_USB_STORAGE_SDDR09 is not set +# CONFIG_USB_STORAGE_JUMPSHOT is not set +# CONFIG_USB_ACM is not set +# CONFIG_USB_PRINTER is not set + +# +# USB Human Interface Devices (HID) +# + +# +# Input core support is needed for USB HID +# + +# +# USB Imaging devices +# +# CONFIG_USB_DC2XX is not set +# CONFIG_USB_MDC800 is not set +# CONFIG_USB_SCANNER is not set +# CONFIG_USB_MICROTEK is not set +# CONFIG_USB_HPUSBSCSI is not set + +# +# USB Multimedia devices +# + +# +# Video4Linux support is needed for USB Multimedia device support +# + +# +# USB Network adaptors +# +# CONFIG_USB_PEGASUS is not set +# CONFIG_USB_KAWETH is not set +# CONFIG_USB_CATC is not set +# CONFIG_USB_CDCETHER is not set +# CONFIG_USB_USBNET is not set + +# +# USB port drivers +# +# CONFIG_USB_USS720 is not set + +# +# USB Serial Converter support +# +# CONFIG_USB_SERIAL is not set +# CONFIG_USB_SERIAL_GENERIC is not set +# CONFIG_USB_SERIAL_BELKIN is not set +# CONFIG_USB_SERIAL_WHITEHEAT is not set +# CONFIG_USB_SERIAL_DIGI_ACCELEPORT is not set +# CONFIG_USB_SERIAL_EMPEG is not set +# CONFIG_USB_SERIAL_FTDI_SIO is not set +# CONFIG_USB_SERIAL_VISOR is not set +# CONFIG_USB_SERIAL_IPAQ is not set +# CONFIG_USB_SERIAL_IR is not set +# CONFIG_USB_SERIAL_EDGEPORT is not set +# CONFIG_USB_SERIAL_KEYSPAN_PDA is not set +# CONFIG_USB_SERIAL_KEYSPAN is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XA is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XB is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA18X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19W is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA49W is not set +# CONFIG_USB_SERIAL_MCT_U232 is not set +# CONFIG_USB_SERIAL_KLSI is not set +# CONFIG_USB_SERIAL_PL2303 is not set +# CONFIG_USB_SERIAL_CYBERJACK is not set +# CONFIG_USB_SERIAL_XIRCOM is not set +# CONFIG_USB_SERIAL_OMNINET is not set + +# +# USB Miscellaneous drivers +# +# CONFIG_USB_RIO500 is not set + +# +# Kernel hacking +# +# CONFIG_FRAME_POINTER is not set +# CONFIG_DEBUG_USER is not set +# CONFIG_DEBUG_INFO is not set +# CONFIG_NO_PGT_CACHE is not set +# CONFIG_DEBUG_KERNEL is not set +# CONFIG_DEBUG_SLAB is not set +# CONFIG_MAGIC_SYSRQ is not set +# CONFIG_DEBUG_SPINLOCK is not set +# CONFIG_DEBUG_WAITQ is not set +# CONFIG_DEBUG_BUGVERBOSE is not set +# CONFIG_DEBUG_ERRORS is not set +# CONFIG_DEBUG_LL is not set +# CONFIG_DEBUG_DC21285_PORT is not set +# CONFIG_DEBUG_CLPS711X_UART2 is not set diff -urN orig/arch/arm/def-configs/riscstation linux/arch/arm/def-configs/riscstation --- orig/arch/arm/def-configs/riscstation Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/def-configs/riscstation Fri Dec 20 17:03:01 2002 @@ -0,0 +1,766 @@ +# +# Automatically generated by make menuconfig: don't edit +# +CONFIG_ARM=y +# CONFIG_EISA is not set +# CONFIG_SBUS is not set +# CONFIG_MCA is not set +CONFIG_UID16=y +CONFIG_RWSEM_GENERIC_SPINLOCK=y +# CONFIG_RWSEM_XCHGADD_ALGORITHM is not set +# CONFIG_GENERIC_BUST_SPINLOCK is not set +# CONFIG_GENERIC_ISA_DMA is not set + +# +# Code maturity level options +# +CONFIG_EXPERIMENTAL=y +# CONFIG_OBSOLETE is not set + +# +# Loadable module support +# +# CONFIG_MODULES is not set +# CONFIG_MODVERSIONS is not set +# CONFIG_KMOD is not set + +# +# System Type +# +# CONFIG_ARCH_ANAKIN is not set +# CONFIG_ARCH_ARCA5K is not set +# CONFIG_ARCH_CLPS7500 is not set +# CONFIG_ARCH_CLPS711X is not set +# CONFIG_ARCH_CO285 is not set +# CONFIG_ARCH_EBSA110 is not set +# CONFIG_ARCH_CAMELOT is not set +# CONFIG_ARCH_FOOTBRIDGE is not set +# CONFIG_ARCH_INTEGRATOR is not set +# CONFIG_ARCH_L7200 is not set +# CONFIG_ARCH_MX1ADS is not set +# CONFIG_ARCH_RPC is not set +CONFIG_ARCH_RISCSTATION=y +# CONFIG_ARCH_SA1100 is not set +# CONFIG_ARCH_SHARK is not set + +# +# Archimedes/A5000 Implementations +# +# CONFIG_ARCH_ARC is not set +# CONFIG_ARCH_A5K is not set + +# +# Footbridge Implementations +# +# CONFIG_ARCH_CATS is not set +# CONFIG_ARCH_PERSONAL_SERVER is not set +# CONFIG_ARCH_EBSA285_ADDIN is not set +# CONFIG_ARCH_EBSA285_HOST is not set +# CONFIG_ARCH_NETWINDER is not set + +# +# SA11x0 Implementations +# +# CONFIG_SA1100_ASSABET is not set +# CONFIG_ASSABET_NEPONSET is not set +# CONFIG_SA1100_ADSBITSY is not set +# CONFIG_SA1100_BRUTUS is not set +# CONFIG_SA1100_CERF is not set +# CONFIG_SA1100_H3100 is not set +# CONFIG_SA1100_H3600 is not set +# CONFIG_SA1100_H3800 is not set +# CONFIG_SA1100_H3XXX is not set +# CONFIG_SA1100_EXTENEX1 is not set +# CONFIG_SA1100_FLEXANET is not set +# CONFIG_SA1100_FREEBIRD is not set +# CONFIG_SA1100_FRODO is not set +# CONFIG_SA1100_GRAPHICSCLIENT is not set +# CONFIG_SA1100_GRAPHICSMASTER is not set +# CONFIG_SA1100_BADGE4 is not set +# CONFIG_SA1100_JORNADA720 is not set +# CONFIG_SA1100_HUW_WEBPANEL is not set +# CONFIG_SA1100_ITSY is not set +# CONFIG_SA1100_LART is not set +# CONFIG_SA1100_NANOENGINE is not set +# CONFIG_SA1100_OMNIMETER is not set +# CONFIG_SA1100_PANGOLIN is not set +# CONFIG_SA1100_PLEB is not set +# CONFIG_SA1100_PT_SYSTEM3 is not set +# CONFIG_SA1100_SHANNON is not set +# CONFIG_SA1100_SHERMAN is not set +# CONFIG_SA1100_SIMPAD is not set +# CONFIG_SA1100_PFS168 is not set +# CONFIG_SA1100_VICTOR is not set +# CONFIG_SA1100_XP860 is not set +# CONFIG_SA1100_YOPY is not set +# CONFIG_SA1100_USB is not set +# CONFIG_SA1100_USB_NETLINK is not set +# CONFIG_SA1100_USB_CHAR is not set +# CONFIG_H3600_SLEEVE is not set + +# +# CLPS711X/EP721X Implementations +# +# CONFIG_ARCH_AUTCPU12 is not set +# CONFIG_ARCH_CDB89712 is not set +# CONFIG_ARCH_CLEP7312 is not set +# CONFIG_ARCH_EDB7211 is not set +# CONFIG_ARCH_P720T is not set +# CONFIG_ARCH_FORTUNET is not set +# CONFIG_ARCH_EP7211 is not set +# CONFIG_ARCH_EP7212 is not set +# CONFIG_ARCH_ACORN is not set +# CONFIG_FOOTBRIDGE is not set +# CONFIG_FOOTBRIDGE_HOST is not set +# CONFIG_FOOTBRIDGE_ADDIN is not set +CONFIG_CPU_32=y +# CONFIG_CPU_26 is not set +CONFIG_CPU_32v3=y +# CONFIG_CPU_32v4 is not set +# CONFIG_CPU_ARM610 is not set +CONFIG_CPU_ARM710=y +# CONFIG_CPU_ARM720T is not set +# CONFIG_CPU_ARM920T is not set +# CONFIG_CPU_ARM922T is not set +# CONFIG_CPU_ARM926T is not set +# CONFIG_CPU_ARM1020 is not set +# CONFIG_CPU_SA110 is not set +# CONFIG_CPU_SA1100 is not set +# CONFIG_ARM_THUMB is not set +# CONFIG_DISCONTIGMEM is not set + +# +# General setup +# +# CONFIG_PCI is not set +CONFIG_ISA=y +# CONFIG_ISA_DMA is not set +# CONFIG_HOTPLUG is not set +# CONFIG_PCMCIA is not set +CONFIG_NET=y +CONFIG_SYSVIPC=y +# CONFIG_BSD_PROCESS_ACCT is not set +CONFIG_SYSCTL=y +CONFIG_FPE_NWFPE=y +# CONFIG_FPE_FASTFPE is not set +CONFIG_KCORE_ELF=y +# CONFIG_KCORE_AOUT is not set +CONFIG_BINFMT_AOUT=y +CONFIG_BINFMT_ELF=y +# CONFIG_BINFMT_MISC is not set +# CONFIG_PM is not set +# CONFIG_ARTHUR is not set +CONFIG_CMDLINE="ip=dhcp" +# CONFIG_ALIGNMENT_TRAP is not set + +# +# Parallel port support +# +# CONFIG_PARPORT is not set + +# +# Plug and Play configuration +# +# CONFIG_PNP is not set +# CONFIG_ISAPNP is not set + +# +# Block devices +# +# CONFIG_BLK_DEV_FD is not set +# CONFIG_BLK_DEV_XD is not set +# CONFIG_PARIDE is not set +# CONFIG_BLK_CPQ_DA is not set +# CONFIG_BLK_CPQ_CISS_DA is not set +# CONFIG_BLK_DEV_DAC960 is not set +# CONFIG_BLK_DEV_LOOP is not set +# CONFIG_BLK_DEV_NBD is not set +CONFIG_BLK_DEV_RAM=y +CONFIG_BLK_DEV_RAM_SIZE=4096 +# CONFIG_BLK_DEV_INITRD is not set + +# +# Multi-device support (RAID and LVM) +# +# CONFIG_MD is not set +# CONFIG_BLK_DEV_MD is not set +# CONFIG_MD_LINEAR is not set +# CONFIG_MD_RAID0 is not set +# CONFIG_MD_RAID1 is not set +# CONFIG_MD_RAID5 is not set +# CONFIG_MD_MULTIPATH is not set +# CONFIG_BLK_DEV_LVM is not set + +# +# Networking options +# +# CONFIG_PACKET is not set +# CONFIG_NETLINK_DEV is not set +# CONFIG_NETFILTER is not set +# CONFIG_FILTER is not set +CONFIG_UNIX=y +CONFIG_INET=y +# CONFIG_IP_MULTICAST is not set +# CONFIG_IP_ADVANCED_ROUTER is not set +# CONFIG_IP_PNP is not set +# CONFIG_NET_IPIP is not set +# CONFIG_NET_IPGRE is not set +# CONFIG_ARPD is not set +# CONFIG_INET_ECN is not set +# CONFIG_SYN_COOKIES is not set +# CONFIG_IPV6 is not set +# CONFIG_KHTTPD is not set +# CONFIG_ATM is not set +# CONFIG_VLAN_8021Q is not set +# CONFIG_IPX is not set +# CONFIG_ATALK is not set +# CONFIG_DECNET is not set +# CONFIG_BRIDGE is not set +# CONFIG_X25 is not set +# CONFIG_LAPB is not set +# CONFIG_LLC is not set +# CONFIG_NET_DIVERT is not set +# CONFIG_ECONET is not set +# CONFIG_WAN_ROUTER is not set +# CONFIG_NET_FASTROUTE is not set +# CONFIG_NET_HW_FLOWCONTROL is not set + +# +# QoS and/or fair queueing +# +# CONFIG_NET_SCHED is not set + +# +# Network device support +# +CONFIG_NETDEVICES=y + +# +# ARCnet devices +# +# CONFIG_ARCNET is not set +# CONFIG_DUMMY is not set +# CONFIG_BONDING is not set +# CONFIG_EQUALIZER is not set +# CONFIG_TUN is not set +# CONFIG_ETHERTAP is not set + +# +# Ethernet (10 or 100Mbit) +# +CONFIG_NET_ETHERNET=y +# CONFIG_ARM_AM79C961A is not set +# CONFIG_ARM_ETHER1 is not set +# CONFIG_ARM_ETHER3 is not set +# CONFIG_ARM_ETHERH is not set +# CONFIG_SUNLANCE is not set +# CONFIG_SUNBMAC is not set +# CONFIG_SUNQE is not set +# CONFIG_SUNGEM is not set +# CONFIG_NET_VENDOR_3COM is not set +# CONFIG_LANCE is not set +# CONFIG_NET_VENDOR_SMC is not set +# CONFIG_NET_VENDOR_RACAL is not set +# CONFIG_AT1700 is not set +# CONFIG_DEPCA is not set +# CONFIG_HP100 is not set +# CONFIG_NET_ISA is not set +CONFIG_NET_PCI=y +# CONFIG_PCNET32 is not set +# CONFIG_ADAPTEC_STARFIRE is not set +# CONFIG_AC3200 is not set +# CONFIG_APRICOT is not set +# CONFIG_CS89x0 is not set +# CONFIG_TULIP is not set +# CONFIG_DM9102 is not set +# CONFIG_EEPRO100 is not set +# CONFIG_LNE390 is not set +# CONFIG_FEALNX is not set +# CONFIG_NATSEMI is not set +# CONFIG_NE2K_PCI is not set +# CONFIG_NE3210 is not set +# CONFIG_ES3210 is not set +# CONFIG_8139CP is not set +# CONFIG_8139TOO is not set +# CONFIG_8139TOO_PIO is not set +# CONFIG_8139TOO_TUNE_TWISTER is not set +# CONFIG_8139TOO_8129 is not set +# CONFIG_8139_NEW_RX_RESET is not set +# CONFIG_SIS900 is not set +# CONFIG_EPIC100 is not set +# CONFIG_SUNDANCE is not set +# CONFIG_VIA_RHINE is not set +# CONFIG_VIA_RHINE_MMIO is not set +# CONFIG_WINBOND_840 is not set +# CONFIG_NET_POCKET is not set + +# +# Ethernet (1000 Mbit) +# +# CONFIG_ACENIC is not set +# CONFIG_DL2K is not set +# CONFIG_MYRI_SBUS is not set +# CONFIG_NS83820 is not set +# CONFIG_HAMACHI is not set +# CONFIG_YELLOWFIN is not set +# CONFIG_SK98LIN is not set +# CONFIG_FDDI is not set +# CONFIG_HIPPI is not set +# CONFIG_PLIP is not set +# CONFIG_PPP is not set +# CONFIG_SLIP is not set + +# +# Wireless LAN (non-hamradio) +# +# CONFIG_NET_RADIO is not set + +# +# Token Ring devices +# +# CONFIG_TR is not set +# CONFIG_NET_FC is not set +# CONFIG_RCPCI is not set +# CONFIG_SHAPER is not set + +# +# Wan interfaces +# +# CONFIG_WAN is not set + +# +# Amateur Radio support +# +# CONFIG_HAMRADIO is not set + +# +# IrDA (infrared) support +# +# CONFIG_IRDA is not set + +# +# ATA/IDE/MFM/RLL support +# +CONFIG_IDE=y + +# +# IDE, ATA and ATAPI Block devices +# +CONFIG_BLK_DEV_IDE=y +# CONFIG_BLK_DEV_HD_IDE is not set +# CONFIG_BLK_DEV_HD is not set +CONFIG_BLK_DEV_IDEDISK=y +# CONFIG_IDEDISK_MULTI_MODE is not set +# CONFIG_BLK_DEV_IDEDISK_VENDOR is not set +# CONFIG_BLK_DEV_IDEDISK_FUJITSU is not set +# CONFIG_BLK_DEV_IDEDISK_IBM is not set +# CONFIG_BLK_DEV_IDEDISK_MAXTOR is not set +# CONFIG_BLK_DEV_IDEDISK_QUANTUM is not set +# CONFIG_BLK_DEV_IDEDISK_SEAGATE is not set +# CONFIG_BLK_DEV_IDEDISK_WD is not set +# CONFIG_BLK_DEV_COMMERIAL is not set +# CONFIG_BLK_DEV_TIVO is not set +# CONFIG_BLK_DEV_IDECS is not set +CONFIG_BLK_DEV_IDECD=y +# CONFIG_BLK_DEV_IDETAPE is not set +# CONFIG_BLK_DEV_IDEFLOPPY is not set +# CONFIG_BLK_DEV_IDESCSI is not set +# CONFIG_BLK_DEV_CMD640 is not set +# CONFIG_BLK_DEV_CMD640_ENHANCED is not set +# CONFIG_BLK_DEV_ISAPNP is not set +CONFIG_BLK_DEV_IDE_RISCSTATION=y +# CONFIG_IDE_CHIPSETS is not set +# CONFIG_IDEDMA_AUTO is not set +# CONFIG_DMA_NONPCI is not set +# CONFIG_BLK_DEV_IDE_MODES is not set +# CONFIG_BLK_DEV_ATARAID is not set +# CONFIG_BLK_DEV_ATARAID_PDC is not set +# CONFIG_BLK_DEV_ATARAID_HPT is not set + +# +# SCSI support +# +# CONFIG_SCSI is not set + +# +# I2O device support +# +# CONFIG_I2O is not set +# CONFIG_I2O_BLOCK is not set +# CONFIG_I2O_LAN is not set +# CONFIG_I2O_SCSI is not set +# CONFIG_I2O_PROC is not set + +# +# ISDN subsystem +# +# CONFIG_ISDN is not set + +# +# Input core support +# +# CONFIG_INPUT is not set +# CONFIG_INPUT_KEYBDEV is not set +# CONFIG_INPUT_MOUSEDEV is not set +# CONFIG_INPUT_JOYDEV is not set +# CONFIG_INPUT_EVDEV is not set + +# +# Character devices +# +CONFIG_VT=y +CONFIG_VT_CONSOLE=y +CONFIG_SERIAL=y +CONFIG_SERIAL_CONSOLE=y +# CONFIG_SERIAL_EXTENDED is not set +# CONFIG_SERIAL_NONSTANDARD is not set + +# +# Serial drivers +# +# CONFIG_SERIAL_ANAKIN is not set +# CONFIG_SERIAL_ANAKIN_CONSOLE is not set +# CONFIG_SERIAL_AMBA is not set +# CONFIG_SERIAL_AMBA_CONSOLE is not set +# CONFIG_SERIAL_CLPS711X is not set +# CONFIG_SERIAL_CLPS711X_CONSOLE is not set +# CONFIG_SERIAL_21285 is not set +# CONFIG_SERIAL_21285_OLD is not set +# CONFIG_SERIAL_21285_CONSOLE is not set +# CONFIG_SERIAL_UART00 is not set +# CONFIG_SERIAL_UART00_CONSOLE is not set +# CONFIG_SERIAL_SA1100 is not set +# CONFIG_SERIAL_SA1100_CONSOLE is not set +# CONFIG_SERIAL_8250 is not set +# CONFIG_SERIAL_8250_CONSOLE is not set +# CONFIG_SERIAL_8250_EXTENDED is not set +# CONFIG_SERIAL_8250_MANY_PORTS is not set +# CONFIG_SERIAL_8250_SHARE_IRQ is not set +# CONFIG_SERIAL_8250_DETECT_IRQ is not set +# CONFIG_SERIAL_8250_MULTIPORT is not set +# CONFIG_SERIAL_8250_HUB6 is not set +CONFIG_UNIX98_PTYS=y +CONFIG_UNIX98_PTY_COUNT=256 + +# +# I2C support +# +# CONFIG_I2C is not set + +# +# L3 serial bus support +# +# CONFIG_L3 is not set +# CONFIG_L3_ALGOBIT is not set +# CONFIG_L3_BIT_SA1100_GPIO is not set +# CONFIG_L3_SA1111 is not set +# CONFIG_BIT_SA1100_GPIO is not set + +# +# Mice +# +# CONFIG_BUSMOUSE is not set +CONFIG_MOUSE=y +# CONFIG_PSMOUSE is not set +# CONFIG_82C710_MOUSE is not set +# CONFIG_PC110_PAD is not set +CONFIG_7K5MOUSE=y +# +# Joysticks +# +# CONFIG_INPUT_GAMEPORT is not set +# CONFIG_QIC02_TAPE is not set + +# +# Watchdog Cards +# +# CONFIG_WATCHDOG is not set +# CONFIG_INTEL_RNG is not set +# CONFIG_NVRAM is not set +CONFIG_RTC=y +# CONFIG_DTLK is not set +# CONFIG_R3964 is not set +# CONFIG_APPLICOM is not set + +# +# Ftape, the floppy tape device driver +# +# CONFIG_FTAPE is not set +# CONFIG_AGP is not set +# CONFIG_DRM is not set + +# +# Multimedia devices +# +# CONFIG_VIDEO_DEV is not set + +# +# File systems +# +# CONFIG_QUOTA is not set +# CONFIG_AUTOFS_FS is not set +# CONFIG_AUTOFS4_FS is not set +# CONFIG_REISERFS_FS is not set +# CONFIG_REISERFS_CHECK is not set +# CONFIG_REISERFS_PROC_INFO is not set +CONFIG_ADFS_FS=y +# CONFIG_ADFS_FS_RW is not set +# CONFIG_AFFS_FS is not set +# CONFIG_HFS_FS is not set +# CONFIG_BFS_FS is not set +CONFIG_EXT3_FS=y +CONFIG_JBD=y +# CONFIG_JBD_DEBUG is not set +CONFIG_FAT_FS=y +CONFIG_MSDOS_FS=y +# CONFIG_UMSDOS_FS is not set +CONFIG_VFAT_FS=y +# CONFIG_EFS_FS is not set +# CONFIG_JFFS_FS is not set +# CONFIG_JFFS2_FS is not set +CONFIG_CRAMFS=y +CONFIG_TMPFS=y +# CONFIG_RAMFS is not set +# CONFIG_ISO9660_FS is not set +# CONFIG_JOLIET is not set +# CONFIG_ZISOFS is not set +# CONFIG_MINIX_FS is not set +# CONFIG_VXFS_FS is not set +# CONFIG_NTFS_FS is not set +# CONFIG_NTFS_RW is not set +# CONFIG_HPFS_FS is not set +CONFIG_PROC_FS=y +# CONFIG_DEVFS_FS is not set +# CONFIG_DEVFS_MOUNT is not set +# CONFIG_DEVFS_DEBUG is not set +CONFIG_DEVPTS_FS=y +# CONFIG_QNX4FS_FS is not set +# CONFIG_QNX4FS_RW is not set +CONFIG_ROMFS_FS=y +CONFIG_EXT2_FS=y +# CONFIG_SYSV_FS is not set +# CONFIG_UDF_FS is not set +# CONFIG_UDF_RW is not set +# CONFIG_UFS_FS is not set +# CONFIG_UFS_FS_WRITE is not set + +# +# Network File Systems +# +# CONFIG_CODA_FS is not set +# CONFIG_INTERMEZZO_FS is not set +# CONFIG_NFS_FS is not set +# CONFIG_NFS_V3 is not set +# CONFIG_ROOT_NFS is not set +# CONFIG_NFSD is not set +# CONFIG_NFSD_V3 is not set +# CONFIG_SUNRPC is not set +# CONFIG_LOCKD is not set +# CONFIG_SMB_FS is not set +# CONFIG_NCP_FS is not set +# CONFIG_NCPFS_PACKET_SIGNING is not set +# CONFIG_NCPFS_IOCTL_LOCKING is not set +# CONFIG_NCPFS_STRONG is not set +# CONFIG_NCPFS_NFS_NS is not set +# CONFIG_NCPFS_OS2_NS is not set +# CONFIG_NCPFS_SMALLDOS is not set +# CONFIG_NCPFS_NLS is not set +# CONFIG_NCPFS_EXTRAS is not set +# CONFIG_ZISOFS_FS is not set +CONFIG_ZLIB_FS_INFLATE=y + +# +# Partition Types +# +CONFIG_PARTITION_ADVANCED=y +CONFIG_ACORN_PARTITION=y +CONFIG_ACORN_PARTITION_EESOX=y +CONFIG_ACORN_PARTITION_ICS=y +CONFIG_ACORN_PARTITION_ADFS=y +CONFIG_ACORN_PARTITION_POWERTEC=y +CONFIG_ACORN_PARTITION_RISCIX=y +# CONFIG_OSF_PARTITION is not set +# CONFIG_AMIGA_PARTITION is not set +# CONFIG_ATARI_PARTITION is not set +# CONFIG_MAC_PARTITION is not set +CONFIG_MSDOS_PARTITION=y +# CONFIG_BSD_DISKLABEL is not set +# CONFIG_MINIX_SUBPARTITION is not set +# CONFIG_SOLARIS_X86_PARTITION is not set +# CONFIG_UNIXWARE_DISKLABEL is not set +# CONFIG_LDM_PARTITION is not set +# CONFIG_SGI_PARTITION is not set +# CONFIG_ULTRIX_PARTITION is not set +# CONFIG_SUN_PARTITION is not set +# CONFIG_SMB_NLS is not set +CONFIG_NLS=y + +# +# Native Language Support +# +CONFIG_NLS_DEFAULT="iso8859-1" +# CONFIG_NLS_CODEPAGE_437 is not set +# CONFIG_NLS_CODEPAGE_737 is not set +# CONFIG_NLS_CODEPAGE_775 is not set +# CONFIG_NLS_CODEPAGE_850 is not set +# CONFIG_NLS_CODEPAGE_852 is not set +# CONFIG_NLS_CODEPAGE_855 is not set +# CONFIG_NLS_CODEPAGE_857 is not set +# CONFIG_NLS_CODEPAGE_860 is not set +# CONFIG_NLS_CODEPAGE_861 is not set +# CONFIG_NLS_CODEPAGE_862 is not set +# CONFIG_NLS_CODEPAGE_863 is not set +# CONFIG_NLS_CODEPAGE_864 is not set +# CONFIG_NLS_CODEPAGE_865 is not set +# CONFIG_NLS_CODEPAGE_866 is not set +# CONFIG_NLS_CODEPAGE_869 is not set +# CONFIG_NLS_CODEPAGE_936 is not set +# CONFIG_NLS_CODEPAGE_950 is not set +# CONFIG_NLS_CODEPAGE_932 is not set +# CONFIG_NLS_CODEPAGE_949 is not set +# CONFIG_NLS_CODEPAGE_874 is not set +# CONFIG_NLS_ISO8859_8 is not set +# CONFIG_NLS_CODEPAGE_1250 is not set +# CONFIG_NLS_CODEPAGE_1251 is not set +# CONFIG_NLS_ISO8859_1 is not set +# CONFIG_NLS_ISO8859_2 is not set +# CONFIG_NLS_ISO8859_3 is not set +# CONFIG_NLS_ISO8859_4 is not set +# CONFIG_NLS_ISO8859_5 is not set +# CONFIG_NLS_ISO8859_6 is not set +# CONFIG_NLS_ISO8859_7 is not set +# CONFIG_NLS_ISO8859_9 is not set +# CONFIG_NLS_ISO8859_13 is not set +# CONFIG_NLS_ISO8859_14 is not set +# CONFIG_NLS_ISO8859_15 is not set +# CONFIG_NLS_KOI8_R is not set +# CONFIG_NLS_KOI8_U is not set +# CONFIG_NLS_UTF8 is not set + +# +# Console drivers +# + +# +# Frame-buffer support +# +CONFIG_FB=y +CONFIG_DUMMY_CONSOLE=y +CONFIG_FB_ACORN=y +# CONFIG_FB_ANAKIN is not set +# CONFIG_FB_CLPS711X is not set +# CONFIG_FB_SA1100 is not set +# CONFIG_FB_CYBER2000 is not set +# CONFIG_FB_VIRTUAL is not set +# CONFIG_FBCON_ADVANCED is not set +CONFIG_FBCON_MFB=y +CONFIG_FBCON_CFB2=y +CONFIG_FBCON_CFB4=y +CONFIG_FBCON_CFB8=y +# CONFIG_FBCON_FONTWIDTH8_ONLY is not set +# CONFIG_FBCON_FONTS is not set +CONFIG_FONT_8x8=y +CONFIG_FONT_8x16=y +CONFIG_FONT_ACORN_8x8=y + +# +# Sound +# +# CONFIG_SOUND is not set + +# +# Multimedia Capabilities Port drivers +# +# CONFIG_MCP is not set +# CONFIG_MCP_SA1100 is not set +# CONFIG_MCP_UCB1200 is not set +# CONFIG_MCP_UCB1200_AUDIO is not set +# CONFIG_MCP_UCB1200_TS is not set + +# +# USB support +# +# CONFIG_USB is not set +# CONFIG_USB_UHCI is not set +# CONFIG_USB_UHCI_ALT is not set +# CONFIG_USB_OHCI is not set +# CONFIG_USB_OHCI_SA1111 is not set +# CONFIG_USB_AUDIO is not set +# CONFIG_USB_BLUETOOTH is not set +# CONFIG_USB_STORAGE is not set +# CONFIG_USB_STORAGE_DEBUG is not set +# CONFIG_USB_STORAGE_DATAFAB is not set +# CONFIG_USB_STORAGE_FREECOM is not set +# CONFIG_USB_STORAGE_ISD200 is not set +# CONFIG_USB_STORAGE_DPCM is not set +# CONFIG_USB_STORAGE_HP8200e is not set +# CONFIG_USB_STORAGE_SDDR09 is not set +# CONFIG_USB_STORAGE_JUMPSHOT is not set +# CONFIG_USB_ACM is not set +# CONFIG_USB_PRINTER is not set +# CONFIG_USB_DC2XX is not set +# CONFIG_USB_MDC800 is not set +# CONFIG_USB_SCANNER is not set +# CONFIG_USB_MICROTEK is not set +# CONFIG_USB_HPUSBSCSI is not set +# CONFIG_USB_PEGASUS is not set +# CONFIG_USB_KAWETH is not set +# CONFIG_USB_CATC is not set +# CONFIG_USB_CDCETHER is not set +# CONFIG_USB_USBNET is not set +# CONFIG_USB_USS720 is not set + +# +# USB Serial Converter support +# +# CONFIG_USB_SERIAL is not set +# CONFIG_USB_SERIAL_GENERIC is not set +# CONFIG_USB_SERIAL_BELKIN is not set +# CONFIG_USB_SERIAL_WHITEHEAT is not set +# CONFIG_USB_SERIAL_DIGI_ACCELEPORT is not set +# CONFIG_USB_SERIAL_EMPEG is not set +# CONFIG_USB_SERIAL_FTDI_SIO is not set +# CONFIG_USB_SERIAL_VISOR is not set +# CONFIG_USB_SERIAL_IPAQ is not set +# CONFIG_USB_SERIAL_IR is not set +# CONFIG_USB_SERIAL_EDGEPORT is not set +# CONFIG_USB_SERIAL_KEYSPAN_PDA is not set +# CONFIG_USB_SERIAL_KEYSPAN is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XA is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA28XB is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19 is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA18X is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA19W is not set +# CONFIG_USB_SERIAL_KEYSPAN_USA49W is not set +# CONFIG_USB_SERIAL_MCT_U232 is not set +# CONFIG_USB_SERIAL_KLSI is not set +# CONFIG_USB_SERIAL_PL2303 is not set +# CONFIG_USB_SERIAL_CYBERJACK is not set +# CONFIG_USB_SERIAL_XIRCOM is not set +# CONFIG_USB_SERIAL_OMNINET is not set +# CONFIG_USB_RIO500 is not set + +# +# Bluetooth support +# +# CONFIG_BLUEZ is not set + +# +# Kernel hacking +# +CONFIG_FRAME_POINTER=y +# CONFIG_DEBUG_USER is not set +# CONFIG_DEBUG_INFO is not set +# CONFIG_NO_PGT_CACHE is not set +# CONFIG_DEBUG_KERNEL is not set +# CONFIG_DEBUG_SLAB is not set +# CONFIG_MAGIC_SYSRQ is not set +# CONFIG_DEBUG_SPINLOCK is not set +# CONFIG_DEBUG_WAITQ is not set +# CONFIG_DEBUG_BUGVERBOSE is not set +# CONFIG_DEBUG_ERRORS is not set +# CONFIG_DEBUG_LL is not set +# CONFIG_DEBUG_DC21285_PORT is not set +# CONFIG_DEBUG_CLPS711X_UART2 is not set diff -urN orig/arch/arm/def-configs/sherman linux/arch/arm/def-configs/sherman --- orig/arch/arm/def-configs/sherman Mon Sep 3 14:14:39 2001 +++ linux/arch/arm/def-configs/sherman Sat May 18 14:49:36 2002 @@ -47,6 +47,9 @@ # # General setup # +CONFIG_ZBOOT_ROM=y +CONFIG_ZBOOT_ROM_TEXT=0x00050000 +CONFIG_ZBOOT_ROM_BSS=0xc0200000 # CONFIG_NET is not set # CONFIG_SYSVIPC is not set # CONFIG_BSD_PROCESS_ACCT is not set diff -urN orig/arch/arm/def-configs/victor linux/arch/arm/def-configs/victor --- orig/arch/arm/def-configs/victor Wed May 31 13:12:12 2000 +++ linux/arch/arm/def-configs/victor Sat May 18 14:49:52 2002 @@ -46,6 +46,9 @@ # # General setup # +CONFIG_ZBOOT_ROM=y +CONFIG_ZBOOT_ROM_TEXT=0x00002000 +CONFIG_ZBOOT_ROM_BSS=0xc0200000 # CONFIG_NET is not set # CONFIG_SYSVIPC is not set # CONFIG_BSD_PROCESS_ACCT is not set diff -urN orig/arch/arm/fastfpe/CPDO.S linux/arch/arm/fastfpe/CPDO.S --- orig/arch/arm/fastfpe/CPDO.S Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/fastfpe/CPDO.S Fri Dec 28 17:39:50 2001 @@ -0,0 +1,682 @@ +/* +The FP structure has 4 words reserved for each register, the first is used just +for the sign in bit 31, the second and third are for the mantissa (unsigned +integer, high 32 bit first) and the fourth is the exponent (signed integer). +The mantissa is always normalized. + +If the exponent is 0x80000000, that is the most negative value, the number +represented is 0 and both mantissa words are also 0. + +If the exponent is 0x7fffffff, that is the biggest positive value, the number +represented is infinity if the high 32 mantissa bit are also 0, otherwise it is +a NaN. The low 32 mantissa bit are 0 if the number represented is infinity. + +Decimal and packed decimal numbers are not supported yet. + +The parameters to these functions are r0=destination pointer, r1 and r2 +source pointers. r4 is the instruction. They may use r0-r8 and r14. They return +to fastfpe_next, except CPDO_rnf_core which expects the return address in r14. +*/ + +/*---------------------------------------------------------------------------*/ + + .globl CPDO_adf +CPDO_adf: + ldmia r1,{r1,r3,r5,r7} + ldmia r2,{r2,r4,r6,r8} + + cmp r7,#0x7fffffff + cmpne r8,#0x7fffffff + beq CPDO_adf_extra + + cmp r1,r2 + bne CPDO_suf_s + +CPDO_adf_s: + subs r2,r7,r8 + bge CPDO_adf_2nd + + mov r7,r8 + rsb r2,r2,#0 + cmp r2,#32 + ble CPDO_adf_1st2 + + sub r2,r2,#32 + cmp r2,#32 + movgt r2,#32 + mov r5,r3,lsr r2 + mov r3,#0 + b CPDO_adf_add + +CPDO_adf_1st2: + rsb r8,r2,#32 + mov r5,r5,lsr r2 + orr r5,r5,r3,lsl r8 + mov r3,r3,lsr r2 @ 1. op normalized + b CPDO_adf_add + +CPDO_adf_2nd: + cmp r2,#32 + ble CPDO_adf_2nd2 + + sub r2,r2,#32 + cmp r2,#32 + movgt r2,#32 + mov r6,r4,lsr r2 + mov r4,#0 + b CPDO_adf_add + +CPDO_adf_2nd2: + rsb r8,r2,#32 + mov r6,r6,lsr r2 + orr r6,r6,r4,lsl r8 + mov r4,r4,lsr r2 @ 2. op normalized + +CPDO_adf_add: + adds r5,r5,r6 + adcs r3,r3,r4 @ do addition + bcc CPDO_adf_end + + add r7,r7,#1 + movs r3,r3,rrx + mov r5,r5,rrx @ correct for overflow + +CPDO_adf_end: + cmp r7,#0x20000000 + bge CPDO_inf + + stmia r0,{r1,r3,r5,r7} + b fastfpe_next + +CPDO_adf_extra: + cmp r7,#0x7fffffff @ was it the 1st ? + bne CPDO_infnan_2 @ no it was the 2nd + cmp r8,#0x7fffffff @ if 1st, 2nd too ? + bne CPDO_infnan_1 @ no only 1st + cmp r3,#0 + cmpeq r4,#0 + bne CPDO_nan_12 + b CPDO_inf + +/*---------------------------------------------------------------------------*/ + +CPDO_infnan_1: + stmia r0,{r1,r3,r5,r7} + b fastfpe_next + +CPDO_infnan_2: + stmia r0,{r2,r4,r6,r8} + b fastfpe_next + +CPDO_nan_12: + orr r2,r3,r4 + b CPDO_inf_1 + +CPDO_nan: + mov r2,#0x40000000 @ create non signalling NaN + b CPDO_inf_1 + +CPDO_inf: + mov r2,#0 +CPDO_inf_1: + mov r3,#0 + mov r4,#0x7fffffff +CPDO_store_1234: + stmia r0,{r1,r2,r3,r4} + b fastfpe_next + +CPDO_zero: + mov r1,#0 +CPDO_zero_1: + mov r2,#0 + mov r3,#0 + mov r4,#0x80000000 + stmia r0,{r1,r2,r3,r4} + b fastfpe_next + +/*---------------------------------------------------------------------------*/ + + .globl CPDO_suf +CPDO_suf: + ldmia r1,{r1,r3,r5,r7} + ldmia r2,{r2,r4,r6,r8} + +CPDO_suf_l: + cmp r7,#0x7fffffff + cmpne r8,#0x7fffffff + beq CPDO_suf_extra + + cmp r1,r2 + bne CPDO_adf_s + +CPDO_suf_s: + subs r2,r7,r8 @ determine greater number + bgt CPDO_suf_2nd @ first number is greater + blt CPDO_suf_1st @ second number is greater + cmp r3,r4 @ also mantissa is important + cmpeq r5,r6 + bhi CPDO_suf_2nd @ first number is greater + beq CPDO_zero + +CPDO_suf_1st: + eor r1,r1,#0x80000000 @ second number is greater, invert sign + mov r7,r8 + rsb r2,r2,#0 + cmp r2,#32 + ble CPDO_suf_1st2 + + sub r2,r2,#32 + cmp r2,#32 + movgt r2,#32 + mov r5,r3,lsr r2 + mov r3,#0 + b CPDO_suf_1st_sub + +CPDO_suf_1st2: + rsb r8,r2,#32 + mov r5,r5,lsr r2 + orr r5,r5,r3,lsl r8 + mov r3,r3,lsr r2 @ 1. op normalized + +CPDO_suf_1st_sub: + subs r5,r6,r5 @ do subtraction + sbc r3,r4,r3 + b CPDO_suf_norm + +CPDO_suf_2nd: + cmp r2,#32 + ble CPDO_suf_2nd2 + + sub r2,r2,#32 + cmp r2,#32 + movgt r2,#32 + mov r6,r4,lsr r2 + mov r4,#0 + b CPDO_suf_2nd_sub + +CPDO_suf_2nd2: + rsb r8,r2,#32 + mov r6,r6,lsr r2 + orr r6,r6,r4,lsl r8 + mov r4,r4,lsr r2 @ 2. op normalized + +CPDO_suf_2nd_sub: + subs r5,r5,r6 + sbc r3,r3,r4 @ do subtraction + +CPDO_suf_norm: + teq r3,#0 @ normalize 32bit + moveq r3,r5 + moveq r5,#0 + subeq r7,r7,#32 + + cmp r3,#0x00010000 @ 16bit + movcc r3,r3,lsl#16 + orrcc r3,r3,r5,lsr#16 + movcc r5,r5,lsl#16 + subcc r7,r7,#16 + + cmp r3,#0x01000000 @ 8bit + movcc r3,r3,lsl#8 + orrcc r3,r3,r5,lsr#24 + movcc r5,r5,lsl#8 + subcc r7,r7,#8 + + cmp r3,#0x10000000 @ 4bit + movcc r3,r3,lsl#4 + orrcc r3,r3,r5,lsr#28 + movcc r5,r5,lsl#4 + subcc r7,r7,#4 + + cmp r3,#0x40000000 @ 2bit + movcc r3,r3,lsl#2 + orrcc r3,r3,r5,lsr#30 + movcc r5,r5,lsl#2 + subcc r7,r7,#2 + + cmp r3,#0x80000000 @ 1bit + movcc r3,r3,lsl#1 + orrcc r3,r3,r5,lsr#31 + movcc r5,r5,lsl#1 + subcc r7,r7,#1 + + cmp r7,#0xe0000000 + ble CPDO_zero_1 + + stmia r0,{r1,r3,r5,r7} + b fastfpe_next + +CPDO_suf_extra: + cmp r7,#0x7fffffff @ was it the 1st ? + eorne r2,r2,#0x80000000 @ change sign, might have been INF + bne CPDO_infnan_2 @ no it was the 2nd + cmp r8,#0x7fffffff @ if 1st, 2nd too ? + bne CPDO_infnan_1 @ no only 1st + cmp r3,#0 + cmpeq r4,#0 + bne CPDO_nan_12 + b CPDO_nan @ here is difference with adf ! + +/*---------------------------------------------------------------------------*/ + + .globl CPDO_rsf +CPDO_rsf: + mov r3,r2 + ldmia r1,{r2,r4,r6,r8} + ldmia r3,{r1,r3,r5,r7} + b CPDO_suf_l + +/*---------------------------------------------------------------------------*/ + + .globl CPDO_muf +CPDO_muf: + ldmia r1,{r1,r3,r5,r7} + ldmia r2,{r2,r4,r6,r8} + + cmp r7,#0x7fffffff + cmpne r8,#0x7fffffff + beq CPDO_muf_extra + + eor r1,r1,r2 + adds r8,r7,r8 + bvs CPDO_zero_1 + + umull r7,r2,r3,r4 + umull r14,r3,r6,r3 + adds r7,r7,r3 @ r2|r7|r14 = r2|r7|#0 + #0|r3|r14 + adc r2,r2,#0 + umull r4,r3,r5,r4 + adds r14,r14,r4 @ r2|r7|r14 += #0|r3|r4 + adcs r7,r7,r3 + adc r2,r2,#0 + umull r4,r3,r5,r6 + adds r14,r14,r3 @ r2|r7|r14 += #0|#0|r3 + adcs r7,r7,#0 + adcs r2,r2,#0 + + bpl CPDO_muf_norm + + add r8,r8,#1 + b CPDO_muf_end + +CPDO_muf_norm: + adds r14,r14,r14 + adcs r7,r7,r7 + adcs r2,r2,r2 + +CPDO_muf_end: + cmp r8,#0x20000000 + bge CPDO_inf + cmp r8,#0xe0000000 + ble CPDO_zero_1 + stmia r0,{r1,r2,r7,r8} + b fastfpe_next + +CPDO_muf_extra: + cmp r7,#0x7fffffff @ was it the first? + bne CPDO_muf_extra_2nd @ no, so it was the second + cmp r8,#0x7fffffff @ yes, second too? + bne CPDO_muf_extra_1st @ no, only first + orr r3,r3,r4 @ if both inf -> inf, otherwise nan + eor r1,r1,r2 @ sign for the inf case + b CPDO_infnan_1 + +CPDO_muf_extra_1st: + cmp r3,#0 @ is it a nan? + bne CPDO_infnan_1 + cmp r8,#0x80000000 @ is the second 0? + beq CPDO_nan + eor r1,r1,r2 @ correct sign for inf + b CPDO_inf + +CPDO_muf_extra_2nd: + cmp r4,#0 @ is it a nan? + bne CPDO_infnan_2 + cmp r7,#0x80000000 @ is the first 0? + beq CPDO_nan + eor r1,r1,r2 @ correct sign for inf + b CPDO_inf + +/*---------------------------------------------------------------------------*/ + + .globl CPDO_dvf +CPDO_dvf: + ldmia r1,{r1,r3,r5,r7} + ldmia r2,{r2,r4,r6,r8} + +CPDO_dvf_l: + cmp r7,#0x7fffffff + cmpne r8,#0x7fffffff + beq CPDO_dvf_extra + cmp r8,#0x80000000 + beq CPDO_dvf_by0 + + eor r1,r1,r2 + cmp r7,#0x80000000 + beq CPDO_zero_1 + + sub r8,r7,r8 + + mov r2,#0 + mov r7,#1 + + cmp r3,r4 + cmpeq r5,r6 + bcs CPDO_dvf_loop_ + + sub r8,r8,#1 + +CPDO_dvf_loop: + adds r5,r5,r5 + adcs r3,r3,r3 + bcs CPDO_dvf_anyway +CPDO_dvf_loop_: + subs r5,r5,r6 + sbcs r3,r3,r4 + bcs CPDO_dvf_okay + + adds r5,r5,r6 + adc r3,r3,r4 + adds r7,r7,r7 + adcs r2,r2,r2 + bcc CPDO_dvf_loop + b CPDO_dvf_end + +CPDO_dvf_anyway: + adcs r7,r7,r7 + adcs r2,r2,r2 + bcs CPDO_dvf_end + subs r5,r5,r6 + sbc r3,r3,r4 + b CPDO_dvf_loop + +CPDO_dvf_okay: + adcs r7,r7,r7 + adcs r2,r2,r2 + bcc CPDO_dvf_loop + +CPDO_dvf_end: + b CPDO_muf_end + +CPDO_dvf_by0: + cmp R7,#0x80000000 + beq CPDO_nan @ first also 0 -> nan + eor r1,r1,r2 @ otherwise calculatesign for inf + b CPDO_inf + +CPDO_dvf_extra: + cmp r7,#0x7fffffff @ was it the first? + bne CPDO_dvf_extra_2nd @ no, so it was the second + cmp r8,#0x7fffffff @ yes, second too? + bne CPDO_dvf_extra_1st @ no, only first + orrs r3,r3,r4 + beq CPDO_nan @ if both inf -> create nan + b CPDO_nan_12 @ otherwise keep nan + +CPDO_dvf_extra_1st: + eor r1,r1,r2 @ correct sign for inf + b CPDO_infnan_1 + +CPDO_dvf_extra_2nd: + cmp r4,#0 @ is it a nan? + bne CPDO_infnan_2 + eor r1,r1,r2 @ correct sign for zero + b CPDO_zero_1 + +/*---------------------------------------------------------------------------*/ + + .globl CPDO_rdf +CPDO_rdf: + mov r3,r2 + ldmia r1,{r2,r4,r6,r8} + ldmia r3,{r1,r3,r5,r7} + b CPDO_dvf_l + +/*---------------------------------------------------------------------------*/ + + .globl CPDO_rmf +CPDO_rmf: + b fastfpe_next + +/*---------------------------------------------------------------------------*/ + + + +/*---------------------------------------------------------------------------*/ + + .globl CPDO_mvf +CPDO_mvf: + ldmia r2,{r1,r2,r3,r4} + stmia r0,{r1,r2,r3,r4} + b fastfpe_next + +/*---------------------------------------------------------------------------*/ + + .globl CPDO_mnf +CPDO_mnf: + ldmia r2,{r1,r2,r3,r4} + eor r1,r1,#0x80000000 + stmia r0,{r1,r2,r3,r4} + b fastfpe_next + +/*---------------------------------------------------------------------------*/ + + .globl CPDO_abs +CPDO_abs: + ldmia r2,{r1,r2,r3,r4} + bic r1,r1,#0x80000000 + stmia r0,{r1,r2,r3,r4} + b fastfpe_next + +/*---------------------------------------------------------------------------*/ + + .globl CPDO_sqt +CPDO_sqt: + ldmia r2,{r1,r2,r3,r4} + cmp r1,#0 + bne CPDO_nan + cmp r4,#0x7fffffff + beq CPDO_store_1234 + + tst r4,r4,lsr#1 @carry=exponent bit 0 + bcc CPDO_sqt_exponenteven + adds r3,r3,r3 + adcs r2,r2,r2 @carry is needed in loop! +CPDO_sqt_exponenteven: + mov r4,r4,asr #1 + str r4,[r0,#12] + + mov r4,#0x80000000 + mov r5,#0 + sub r2,r2,#0x80000000 + + mov r8,#0x40000000 + mov r14,#0x80000000 + + mov r1,#1 + b CPDO_sqt_loop1_first +CPDO_sqt_loop1: + adds r3,r3,r3 + adcs r2,r2,r2 +CPDO_sqt_loop1_first: + add r6,r4,r8,lsr r1 @r7 const = r5 + bcs CPDO_sqt_loop1_1 + cmp r2,r6 + cmpeq r3,r5 @r5 for r7 + bcc CPDO_sqt_loop1_0 +CPDO_sqt_loop1_1: + orr r4,r4,r14,lsr r1 + subs r3,r3,r5 @r5 for r7 + sbc r2,r2,r6 +CPDO_sqt_loop1_0: + add r1,r1,#1 + cmp r1,#30 + ble CPDO_sqt_loop1 + + adds r3,r3,r3 + adcs r2,r2,r2 + bcs CPDO_sqt_between_1 + adds r7,r5,#0x80000000 + adc r6,r4,#0 + cmp r2,r6 + cmpeq r3,r7 + bcc CPDO_sqt_between_0 +CPDO_sqt_between_1: + orr r4,r4,#0x00000001 + subs r3,r3,r5 + sbc r2,r2,r4 + subs r3,r3,#0x80000000 + sbc r2,r2,#0 +CPDO_sqt_between_0: + mov r1,#0 + +CPDO_sqt_loop2: + adds r3,r3,r3 + adcs r2,r2,r2 + bcs CPDO_sqt_loop2_1 + adds r7,r5,r8,lsr r1 + adc r6,r4,#0 + cmp r2,r6 + cmpeq r3,r7 + bcc CPDO_sqt_loop2_0 +CPDO_sqt_loop2_1: + orr r5,r5,r14,lsr r1 + subs r3,r3,r5 + sbc r2,r2,r4 + subs r3,r3,r8,lsr r1 + sbc r2,r2,#0 +CPDO_sqt_loop2_0: + add r1,r1,#1 + cmp r1,#30 + ble CPDO_sqt_loop2 + + adds r3,r3,r3 + adcs r2,r2,r2 + bcs CPDO_sqt_after_1 + cmp r2,r6 + cmpeq r3,r7 + bcc CPDO_sqt_after_0 +CPDO_sqt_after_1: + orr r5,r5,#0x00000001 +CPDO_sqt_after_0: + + mov r1,#0 + stmia r0,{r1,r4,r5} + b fastfpe_next + +/*---------------------------------------------------------------------------*/ + + .globl CPDO_rnd +CPDO_rnd: + ldmia r2,{r1,r2,r3,r5} + bl CPDO_rnd_core + +CPDO_rnd_store: + stmia r0,{r1,r2,r3,r5} + b fastfpe_next + +/*---------------------------------------------------------------------------*/ + + .globl CPDO_rnd_core +CPDO_rnd_core: + and r4,r4,#0x00000060 + add pc,pc,r4,lsr#3 + mov r0,r0 + b CPDO_rnd_N + b CPDO_rnd_P + b CPDO_rnd_M + b CPDO_rnd_Z + +CPDO_rnd_N: + cmp r5,#-1 + blt CPDO_rnd_zero + cmp r5,#63 + movge pc,r14 + mov r4,#0x40000000 + cmp r5,#31 + bge CPDO_rnd_N_2 + + adds r2,r2,r4,lsr r5 + bcc CPDO_rnd_end + b CPDO_rnd_end_norm + +CPDO_rnd_N_2: +CPDO_rnd_P_2: + sub r6,r5,#32 + adds r3,r3,r4,ror r6 @ror ist needed to handle a -1 correctly + adcs r2,r2,#0 + bcc CPDO_rnd_end + b CPDO_rnd_end_norm + +CPDO_rnd_P: + tst r1,#0x80000000 + bne CPDO_rnd_M_entry +CPDO_rnd_P_entry: + cmp r5,#0 + blt CPDO_rnd_P_small + cmp r5,#63 + movge pc,r14 + mov r4,#0x7fffffff + cmp r5,#32 + bge CPDO_rnd_P_2 + + adds r3,r3,#0xffffffff + adcs r2,r2,r4,lsr r5 + bcc CPDO_rnd_end + b CPDO_rnd_end_norm + +CPDO_rnd_P_small: + cmp r5,#0x80000000 + moveq pc,r14 + b CPDO_rnd_one + +CPDO_rnd_M: + tst r1,#0x80000000 + bne CPDO_rnd_P_entry +CPDO_rnd_M_entry: + cmp r5,#0 + blt CPDO_rnd_zero + cmp r5,#63 + movge pc,r14 + + b CPDO_rnd_end + +CPDO_rnd_Z: + cmp r5,#0 + blt CPDO_rnd_zero + cmp r5,#63 + movge pc,r14 + b CPDO_rnd_end + +CPDO_rnd_end_norm: + add r5,r5,#1 + movs r2,r2,rrx + mov r3,r3,rrx +CPDO_rnd_end: + rsbs r4,r5,#31 + bmi CPDO_rnd_end_2 + mov r3,#0 + mov r2,r2,lsr r4 + mov r2,r2,lsl r4 + mov pc,r14 + +CPDO_rnd_end_2: + rsb r4,r5,#63 + mov r3,r3,lsr r4 + mov r3,r3,lsl r4 + mov pc,r14 + +CPDO_rnd_one: + mov r2,#0x80000000 + mov r3,#0 + mov r5,#0 + mov pc,r14 + +CPDO_rnd_zero: + mov r1,#0 + mov r2,#0 + mov r3,#0 + mov r5,#0x80000000 + mov pc,r14 + +/*---------------------------------------------------------------------------*/ diff -urN orig/arch/arm/fastfpe/CPDT.S linux/arch/arm/fastfpe/CPDT.S --- orig/arch/arm/fastfpe/CPDT.S Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/fastfpe/CPDT.S Thu Oct 24 13:58:10 2002 @@ -0,0 +1,430 @@ +/* +The FP structure has 4 words reserved for each register, the first is used just +for the sign in bit 31, the second and third are for the mantissa (unsigned +integer, high 32 bit first) and the fourth is the exponent (signed integer). +The mantissa is always normalized. + +If the exponent is 0x80000000, that is the most negative value, the number +represented is 0 and both mantissa words are also 0. + +If the exponent is 0x7fffffff, that is the biggest positive value, the number +represented is infinity if the high 32 mantissa bit are also 0, otherwise it is +a NaN. The low 32 mantissa bit are 0 if the number represented is infinity. + +Decimal and packed decimal numbers are not supported yet. +*/ + +/*---------------------------------------------------------------------------*/ + + .globl CPDT_load_single +CPDT_load_single: + ldr r1,[r6] + + and r2,r1,#0x80000000 @ r2 = sign + + mov r5,r1,lsr#23 + bics r5,r5,#0x100 + beq CPDT_ls_e0 @ exponent = 0; zero/denormalized + teq r5,#255 + beq CPDT_ls_e255 @ exponent = 255; infinity/NaN + + sub r5,r5,#127 @ r5 = exponent, remove normalized bias + + mov r3,r1,lsl#8 + orr r3,r3,#0x80000000 + mov r4,#0 @ r3,r4 = mantissa + + stmia r0,{r2-r5} + b fastfpe_next + +CPDT_ls_e0: + movs r3,r1,lsl#9 + beq CPDT_load_zero + + mov r5,#-127 + +CPDT_ls_e0_norm: + tst r3,#0x80000000 + subeq r5,r5,#1 + moveq r3,r3,lsl#1 + beq CPDT_ls_e0_norm + + mov r4,#0 + stmia r0,{r2-r5} + b fastfpe_next + +CPDT_ls_e255: + mov r3,r1,lsl#9 + mov r4,#0 + mov r5,#0x7fffffff + stmia r0,{r2-r5} + b fastfpe_next + +CPDT_load_zero: + mov r3,#0 + mov r4,#0 + mov r5,#0x80000000 + stmia r0,{r2-r5} + b fastfpe_next + +/*---------------------------------------------------------------------------*/ + + .globl CPDT_load_double +CPDT_load_double: + ldr r1,[r6] + ldr r6,[r6,#4] + + and r2,r1,#0x80000000 @ r2 = sign + + mov r5,r1,lsr#20 + bics r5,r5,#0x800 + beq CPDT_ld_e0 @ exponent = 0; zero/denormalized + add r4,r5,#1 + teq r4,#2048 + beq CPDT_ld_e2047 @ exponent = 2047; infinity/NaN + + add r5,r5,#1 + sub r5,r5,#1024 @ r5 = exponent, remove normalized bias + + mov r3,r1,lsl#11 + orr r3,r3,#0x80000000 + orr r3,r3,r6,lsr #21 + mov r4,r6,lsl#11 @ r3,r4 = mantissa + + stmia r0,{r2-r5} + b fastfpe_next + +CPDT_ld_e0: + mov r3,r1,lsl#12 + orr r3,r3,r6,lsr#20 + movs r4,r6,lsl#12 + teqeq r3,#0 + beq CPDT_load_zero + + mov r5,#1 + sub r5,r5,#1024 + +CPDT_ld_e0_norm: + tst r3,#0x80000000 + subeq r5,r5,#1 + moveqs r4,r4,lsl#1 + adceq r3,r3,r3 + beq CPDT_ld_e0_norm + + stmia r0,{r2-r5} + b fastfpe_next + +CPDT_ld_e2047: + mov r3,r1,lsl#12 + orr r3,r3,r6,lsr#1 + bic r6,r6,#0x80000000 + orr r3,r3,r6 @ to get all fraction bits ! + mov r4,#0 + mov r5,#0x7fffffff + stmia r0,{r2-r5} + b fastfpe_next + +/*---------------------------------------------------------------------------*/ + + .globl CPDT_load_extended +CPDT_load_extended: + ldr r1,[r6] + ldr r3,[r6,#4] + ldr r4,[r6,#8] + + and r2,r1,#0x80000000 + bics r5,r1,#0x80000000 + beq CPDT_le_e0 + add r1,r5,#1 + teq r4,#32768 + beq CPDT_le_e32767 + + add r5,r5,#1 + sub r5,r5,#16384 + + stmia r0,{r2-r5} + b fastfpe_next + +CPDT_le_e0: + teq r3,#0 + teqeq r4,#0 + beq CPDT_load_zero + + mov r5,#2 + sub r5,r5,#16384 + b CPDT_ld_e0_norm + +CPDT_le_e32767: + mov r3,r3,lsl#1 + orr r3,r3,r4,lsr#1 + bic r4,r4,#0x80000000 + orr r3,r3,r4 + mov r5,#0x7fffffff + stmia r0,{r2-r5} + b fastfpe_next + +/*---------------------------------------------------------------------------*/ + + .globl CPDT_load_decimal +CPDT_load_decimal: + + b fastfpe_next + +/*---------------------------------------------------------------------------*/ + + .globl CPDT_store_single +CPDT_store_single: + ldmia r0,{r1-r4} + + cmp r4,#-127 + ble CPDT_ss_e0 + cmp r4,#128 + bge CPDT_ss_e255 + + adds r2,r2,#1<<7 @ round to nearest + bcs CPDT_ss_rnd_ovfl @ very very seldom taken + +CPDT_ss_store: + add r4,r4,#127 + orr r1,r1,r4,lsl#23 + + bic r2,r2,#0x80000000 + orr r1,r1,r2,lsr#8 + + str r1,[r6] + b fastfpe_next + +CPDT_ss_rnd_ovfl: + add r4,r4,#1 + cmp r4,#128 + bge CPDT_ss_e255 + + mov r2,#0x80000000 + mov r3,#0 + b CPDT_ss_store + +CPDT_ss_e0: + cmp r4,#-150 + ble CPDT_ss_zero + + add r4,r4,#126 +CPDT_ss_unnormalize: + mov r2,r2,lsr#1 + adds r4,r4,#1 + bne CPDT_ss_unnormalize + + orr r1,r1,r2,lsr#8 + +CPDT_ss_zero: + str r1,[r6] + b fastfpe_next + +CPDT_ss_e255: + cmp r4,#0x7fffffff + bne CPDT_ss_inf + cmp r2,#0 + beq CPDT_ss_inf + + orr r1,r1,#0x00200000 @ for safety so that it is not INF + orr r1,r1,r2,lsr#9 @ get highest bit of mantissa + +CPDT_ss_inf: + orr r1,r1,#0x7f000000 + orr r1,r1,#0x00800000 + str r1,[r6] + b fastfpe_next + +/*---------------------------------------------------------------------------*/ + + .globl CPDT_store_double +CPDT_store_double: + ldmia r0,{r1-r4} + + cmp r4,#1024 @ this check has to be first, or + bge CPDT_sd_e2047 @ overflow can occur on second ! + add r0,r4,#3 + cmp r0,#-1023+3 @ cmp with -1023 + ble CPDT_sd_e0 + + adds r3,r3,#1<<10 @ round to nearest + adcs r2,r2,#0 + bcs CPDT_sd_rnd_ovfl @ very very seldom taken + +CPDT_sd_store: + sub r4,r4,#1 + add r4,r4,#1024 + orr r1,r1,r4,lsl#20 + + bic r2,r2,#0x80000000 + orr r1,r1,r2,lsr#11 + + mov r2,r2,lsl#21 + orr r2,r2,r3,lsr#11 + + stmia r6,{r1,r2} + b fastfpe_next + +CPDT_sd_rnd_ovfl: + add r4,r4,#1 + cmp r4,#1024 + bge CPDT_sd_e2047 + + mov r2,#0x80000000 + mov r3,#0 + b CPDT_sd_store + +CPDT_sd_e0: + add r0,r4,#1075-1024 + cmp r0,#-1024 + ble CPDT_sd_zero + + add r4,r4,#1024 + sub r4,r4,#2 +CPDT_sd_unnormalize: + movs r2,r2,lsr#1 + mov r3,r3,rrx + adds r4,r4,#1 + bne CPDT_sd_unnormalize + + orr r1,r1,r2,lsr#11 + mov r2,r2,lsl#21 + orr r2,r2,r3,lsr#11 + + stmia r6,{r1,r2} + b fastfpe_next + +CPDT_sd_zero: + mov r2,#0 + stmia r6,{r1,r2} + b fastfpe_next + +CPDT_sd_e2047: + cmp r4,#0x7fffffff + bne CPDT_sd_inf + cmp r2,#0 + beq CPDT_sd_inf + + orr r1,r1,#0x00040000 @ for safety so that it is not INF + orr r1,r1,r2,lsr#12 @ get highest bit of mantissa + +CPDT_sd_inf: + orr r1,r1,#0x7f000000 + orr r1,r1,#0x00f00000 + stmia r6,{r1,r2} + b fastfpe_next + +/*---------------------------------------------------------------------------*/ + + .globl CPDT_store_extended +CPDT_store_extended: + ldmia r0,{r1-r4} + + cmp r4,#16384 @ this check has to be first, or + bge CPDT_se_e32767 @ overflow can occur with second ! + add r0,r4,#63 + cmp r0,#-16383+63 + ble CPDT_se_e0 + + sub r4,r4,#1 + add r4,r4,#16384 + orr r1,r1,r4 + + stmia r6,{r1-r3} + b fastfpe_next + +CPDT_se_e0: + add r0,r4,#16446-16384 + cmp r0,#-16384 + ble CPDT_se_zero + + add r4,r4,#16384 + sub r4,r4,#2 +CPDT_se_unnormalize: + movs r2,r2,lsr#1 + mov r3,r3,rrx + adds r4,r4,#1 + bne CPDT_se_unnormalize + + stmia r6,{r1-r3} + b fastfpe_next + +CPDT_se_zero: + mov r2,#0 + mov r3,#0 + stmia r6,{r1-r3} + b fastfpe_next + +CPDT_se_e32767: + cmp r4,#0x7fffffff + bne CPDT_se_inf + cmp r2,#0 + beq CPDT_se_inf + + mov r2,r2,lsl#1 + orr r2,r2,#0x20000000 + +CPDT_se_inf: + orr r1,r1,#0x00007f00 + orr r1,r1,#0x000000ff + stmia r6,{r1-r3} + b fastfpe_next + +/*---------------------------------------------------------------------------*/ + + .globl CPDT_store_decimal +CPDT_store_decimal: + + b fastfpe_next + +/*---------------------------------------------------------------------------*/ + + .globl CPDT_sfm +CPDT_sfm: + add r2,r10,r0,lsr#8 + ldr r4,[r2,#0] + ldr r3,[r2,#4] + bic r3,r3,#0x80000000 + orr r3,r3,r4 + str r3,[r6],#4 + ldr r3,[r2,#8] + str r3,[r6],#4 + ldr r3,[r2,#12] + str r3,[r6],#4 + + add r0,r0,#1<<12 + and r0,r0,#7<<12 + subs r1,r1,#1 + bne CPDT_sfm + b fastfpe_next + +/*---------------------------------------------------------------------------*/ + + .globl CPDT_lfm +CPDT_lfm: + add r2,r10,r0,lsr#8 + ldr r4,[r6],#4 + and r3,r4,#0x80000000 + str r3,[r2,#0] + ldr r3,[r6],#4 + str r3,[r2,#8] + ldr r3,[r6],#4 + str r3,[r2,#12] + + cmp r3,#0x80000000 @ does the exp indicate zero? + biceq r4,r4,#0x80000000 @ if so, indicate 'denormalized' + beq CPDT_lfm_storer4 + cmp r3,#0x7fffffff @ does the exp indicate inf or NaN? + biceq r4,r4,#0x80000000 @ if so, indicate 'denormalized' + beq CPDT_lfm_storer4 + orrne r4,r4,#0x80000000 @ otherwise, set normalized bit + +CPDT_lfm_storer4: + str r4,[r2,#4] + + add r0,r0,#1<<12 + and r0,r0,#7<<12 + subs r1,r1,#1 + bne CPDT_lfm + b fastfpe_next + +/*---------------------------------------------------------------------------*/ diff -urN orig/arch/arm/fastfpe/CPRT.S linux/arch/arm/fastfpe/CPRT.S --- orig/arch/arm/fastfpe/CPRT.S Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/fastfpe/CPRT.S Thu Oct 24 17:47:42 2002 @@ -0,0 +1,185 @@ +/* +The FP structure has 4 words reserved for each register, the first is used +just +for the sign in bit 31, the second and third are for the mantissa (unsigned +integer, high 32 bit first) and the fourth is the exponent (signed integer). +The mantissa is always normalized. + +If the exponent is 0x80000000, that is the most negative value, the number +represented is 0 and both mantissa words are also 0. + +If the exponent is 0x7fffffff, that is the biggest positive value, the +number +represented is infinity if the high 32 mantissa bit are also 0, otherwise it +is +a NaN. The low 32 mantissa bit are 0 if the number represented is infinity. + +Decimal and packed decimal numbers are not supported yet. +*/ + +/*---------------------------------------------------------------------------*/ + + .text + .globl CPRT_flt +CPRT_flt: + add r0,r13,r0,lsr#10 + ldr r2,[r0] + mov r3,#0 + cmp r2,#0 + beq CPRT_flt_zero + + ands r0,r2,#0x80000000 + rsbne r2,r2,#0 + mov r4,#31 + + cmp r2,#0x00010000 + movcc r2,r2,lsl#16 + subcc r4,r4,#16 + + cmp r2,#0x01000000 + movcc r2,r2,lsl#8 + subcc r4,r4,#8 + + cmp r2,#0x10000000 + movcc r2,r2,lsl#4 + subcc r4,r4,#4 + + cmp r2,#0x40000000 + movcc r2,r2,lsl#2 + subcc r4,r4,#2 + + cmp r2,#0x80000000 + movcc r2,r2,lsl#1 + subcc r4,r4,#1 + + stmia r1,{r0,r2,r3,r4} + b fastfpe_next + +CPRT_flt_zero: + mov r0,#0 + mov r4,#0x80000000 + stmia r1,{r0,r2,r3,r4} + b fastfpe_next + +/*---------------------------------------------------------------------------*/ + + .globl CPRT_fix +CPRT_fix: + ldmia r2,{r1,r2,r3,r5} + bl CPDO_rnd_core + +CPRT_back: + add r0,r13,r0,lsr#10 + cmp r5,#0 + blt CPRT_int_zero + cmp r5,#30 + bgt CPRT_overflow + + rsb r5,r5,#31 + mov r2,r2,lsr r5 + tst r1,#0x80000000 + rsbne r2,r2,#0 + + str r2,[r0] + b fastfpe_next + +CPRT_int_zero: + mov r2,#0 + str r2,[r0] + b fastfpe_next + +CPRT_overflow: + mov r2,#0x80000000 + tst r1,#0x80000000 + subeq r2,r2,#1 + str r2,[r0] + b fastfpe_next + +/*---------------------------------------------------------------------------*/ + + .globl CPRT_wfs +CPRT_wfs: + b fastfpe_next + +/*---------------------------------------------------------------------------*/ + + .globl CPRT_rfs +CPRT_rfs: + add r0,r13,r0,lsr#10 + mov r1,#0x02000000 @ Software Emulation, not Acorn FPE + str r1,[r0] + b fastfpe_next + +/*---------------------------------------------------------------------------*/ + + .globl CPRT_cmf +CPRT_cmf: + ldmia r1,{r1,r3,r5,r7} + ldmia r2,{r2,r4,r6,r8} + +CPRT_cmf_e: + ldr r0,[r13,#16*4] + + cmp r7,#0x7fffffff + bic r0,r0,#0xf0000000 + + cmpeq r3,#0xffffffff + beq CPRT_cmf_unordered + cmp r8,#0x7fffffff + cmpeq r4,#0xffffffff + beq CPRT_cmf_unordered + + cmp r1,r2 + beq CPRT_cmf_equalsign + b CPRT_cmf_sign + +CPRT_cmf_equalsign: + cmp r7,r8 + beq CPRT_cmf_equalexponent + bgt CPRT_cmf_sign + b CPRT_cmf_signb + +CPRT_cmf_equalexponent: + cmp r3,r4 + cmpeq r5,r6 + beq CPRT_cmf_equal + bhi CPRT_cmf_sign + b CPRT_cmf_signb + +CPRT_cmf_sign: + cmp r7,#0x80000000 @ (0.0 == -0.0)? + cmpeq r7,r8 + beq CPRT_cmf_equal + tst r1,#0x80000000 + orreq r0,r0,#0x20000000 + orrne r0,r0,#0x80000000 + str r0,[r13,#16*4] + b fastfpe_next + +CPRT_cmf_signb: + tst r1,#0x80000000 + orrne r0,r0,#0x20000000 + orreq r0,r0,#0x80000000 + str r0,[r13,#16*4] + b fastfpe_next + +CPRT_cmf_equal: + orr r0,r0,#0x60000000 + str r0,[r13,#16*4] + b fastfpe_next + +CPRT_cmf_unordered: + orr r0,r0,#0x10000000 + str r0,[r13,#16*4] + b fastfpe_next + +/*---------------------------------------------------------------------------*/ + + .globl CPRT_cnf +CPRT_cnf: + ldmia r1,{r1,r3,r5,r7} + ldmia r2,{r2,r4,r6,r8} + eor r2,r2,#0x80000000 + b CPRT_cmf_e + +/*---------------------------------------------------------------------------*/ diff -urN orig/arch/arm/fastfpe/Makefile linux/arch/arm/fastfpe/Makefile --- orig/arch/arm/fastfpe/Makefile Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/fastfpe/Makefile Mon Feb 11 15:42:49 2002 @@ -0,0 +1,25 @@ +# +# linux/arch/arm/fastfpe/Makefile +# +# Copyright (C) Peter Teichmann +# + +O_TARGET := fast-math-emu.o + +obj-y := +obj-m := +obj-n := +obj- := + +fastfpe-objs := module.o entry.o CPDO.o CPRT.o CPDT.o + +list-multi := fastfpe.o + +obj-$(CONFIG_FPE_FASTFPE) += fastfpe.o + +USE_STANDARD_AS_RULE := true + +include $(TOPDIR)/Rules.make + +fastfpe.o: $(fastfpe-objs) + $(LD) -r -o $@ $(fastfpe-objs) diff -urN orig/arch/arm/fastfpe/entry.S linux/arch/arm/fastfpe/entry.S --- orig/arch/arm/fastfpe/entry.S Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/fastfpe/entry.S Fri Dec 28 17:39:51 2001 @@ -0,0 +1,295 @@ +/* +At entry the registers contain the following information: + +r14 return address for undefined exception return +r9 return address for return from exception +r13 user registers on stack, offset 0 up to offset 4*15 contains + registers r0..15, then the psr +r10 FP workspace 35 words (init, reg[8][4], fpsr, fpcr) + +*/ + +/*---------------------------------------------------------------------------*/ + + .data +fp_const: + .word 0, 0x00000000, 0, 0x80000000 @ 0 + .word 0, 0x80000000, 0, 0 @ 1 + .word 0, 0x80000000, 0, 1 @ 2 + .word 0, 0xc0000000, 0, 1 @ 3 + .word 0, 0x80000000, 0, 2 @ 4 + .word 0, 0xa0000000, 0, 2 @ 5 + .word 0, 0x80000000, 0, -1 @ 0.5 + .word 0, 0xa0000000, 0, 3 @ 10 +fp_undef: + .word 0 +fp_cond: + .word 0xf0f0 @ eq + .word 0x0f0f @ ne + .word 0xcccc @ cs + .word 0x3333 @ cc + .word 0xff00 @ mi + .word 0x00ff @ pl + .word 0xaaaa @ vs + .word 0x5555 @ vc + .word 0x0c0c @ hi + .word 0xf3f3 @ ls + .word 0xaa55 @ ge + .word 0x55aa @ lt + .word 0x0a05 @ gt + .word 0xf5fa @ le + .word 0xffff @ al + .word 0x0000 @ nv + +/*---------------------------------------------------------------------------*/ + + .text + .globl fastfpe_enter +fastfpe_enter: + ldr r4,=fp_undef + str r14,[r4] @ to free one register + add r10,r10,#4 @ to make the code simpler + ldr r4,[r13,#60] @ r4=saved PC + ldr r4,[r4,#-4] @ r4=trapped instruction + and r1,r4,#0x00000f00 @ r1=coprocessor << 8 +next_enter: + cmp r1,#1<<8 @ copro 1 ? + beq copro_1 + cmp r1,#2<<8 + movne pc,r14 + +copro_2: + and r1,r4,#0x0f000000 + cmp r1,#0x0c000000 @ CPDT with post indexing + cmpne r1,#0x0d000000 @ CPDT with pre indexing + beq CPDT_M_enter + mov pc,r14 + +copro_1: + and r1,r4,#0x0f000000 + cmp r1,#0x0e000000 @ CPDO + beq CPDO_CPRT_enter + cmp r1,#0x0c000000 @ CPDT with post indexing + cmpne r1,#0x0d000000 @ CPDT with pre indexing + beq CPDT_1_enter + mov pc,r14 + +/*---------------------------------------------------------------------------*/ + + .globl fastfpe_next +fastfpe_next: + ldr r5,[r13,#60] +next_after_cond: +__x1: + ldrt r4,[r5],#4 + + ldr r0,=fp_cond @ check condition of next instruction + ldr r1,[r13,#64] @ psr containing flags + mov r2,r4,lsr#28 + mov r1,r1,lsr#28 + ldr r0,[r0,r2,lsl#2] + mov r0,r0,lsr r1 + tst r0,#1 + beq next_after_cond @ must not necessarily have been an + @ FP instruction ! + and r1,r4,#0x0f000000 @ Test for copro instruction + cmp r1,#0x0c000000 + rsbgts r0,r1,#0x0e000000 @ cmpgt #0x0e000000,r1 + movlt pc,r9 @ next is no copro instruction, return + + ands r1,r4,#0x00000f00 @ r1 = coprocessor << 8 + cmpne r1,#3<<8 + movge pc,r9 @ copro = 0 or >=3, return + + str r5,[r13,#60] @ save updated pc + b next_enter + +/*---------------------------------------------------------------------------*/ + +undefined: + ldr r4,=fp_undef + ldr pc,[r4] + +/*---------------------------------------------------------------------------*/ + +CPDT_1_enter: + and r5,r4,#0x000f0000 @ r5=base register number << 16 + ldr r6,[r13,r5,lsr#14] @ r6=base address + cmp r5,#0x000f0000 @ base register = pc ? + addeq r6,r6,#4 + and r7,r4,#0x000000ff @ r7=offset value + + tst r4,#0x00800000 @ up or down? + addne r7,r6,r7,lsl#2 + subeq r7,r6,r7,lsl#2 @ r6=base address +/- offset + tst r4,#0x01000000 @ preindexing ? + movne r6,r7 + tst r4,#0x00200000 @ write back ? + cmpne r5,#0x000f0000 @ base register = pc ? + strne r7,[r13,r5,lsr#14] + + and r0,r4,#0x00007000 @ r0=fp register number << 12 + add r0,r10,r0,lsr#8 @ r0=address of fp register + mov r1,#0 + tst r4,#0x00008000 + orrne r1,r1,#1 @ T0 + tst r4,#0x00400000 + orrne r1,r1,#2 @ T1 + tst r4,#0x00100000 + orrne r1,r1,#4 @ L/S + + add pc,pc,r1,lsl#2 + mov r0,r0 + b CPDT_store_single @ these functions get + b CPDT_store_double @ r0=address of fp register + b CPDT_store_extended @ r6=address of data + b undefined @ CPDT_store_decimal + b CPDT_load_single + b CPDT_load_double + b CPDT_load_extended + b undefined @ CPDT_load_decimal + +/*---------------------------------------------------------------------------*/ + +CPDT_M_enter: + and r5,r4,#0x000f0000 @ r5=base register number << 16 + ldr r6,[r13,r5,lsr#14] @ r6=base address + cmp r5,#0x000f0000 @ base register = pc ? + addeq r6,r6,#4 + and r7,r4,#0x000000ff @ r7=offset value + + tst r4,#0x00800000 @ up or down? + addne r7,r6,r7,lsl#2 + subeq r7,r6,r7,lsl#2 @ r7=base address +/- offset + tst r4,#0x01000000 @ preindexing ? + movne r6,r7 + tst r4,#0x00200000 @ write back ? + cmpne r5,#0x000f0000 @ base register = pc ? + strne r7,[r13,r5,lsr#14] + + and r0,r4,#0x00007000 @ r0=fp register number << 12 + and r1,r4,#0x00008000 + mov r1,r1,lsr#15 @ N0 + and r2,r4,#0x00400000 + orrs r1,r1,r2,lsr#21 @ N1 + addeq r1,r1,#4 @ r1=register count + + tst r4,#0x00100000 @ load/store + beq CPDT_sfm + b CPDT_lfm + +/*---------------------------------------------------------------------------*/ + +CPDO_CPRT_enter: + tst r4,#0x00000010 + bne CPRT_enter + + and r0,r4,#0x00007000 + add r0,r10,r0,lsr#8 @ r0=address of Fd + and r1,r4,#0x00070000 + add r1,r10,r1,lsr#12 @ r1=address of Fn + tst r4,#0x00000008 + bne CPDO_const + and r2,r4,#0x00000007 + add r2,r10,r2,lsl#4 @ r2=address of Fm + +CPDO_constback: + and r3,r4,#0x00f00000 + tst r4,#0x00008000 + orrne r3,r3,#0x01000000 + + add pc,pc,r3,lsr#18 + mov r0,r0 + b CPDO_adf + b CPDO_muf + b CPDO_suf + b CPDO_rsf + b CPDO_dvf + b CPDO_rdf + b undefined + b undefined + b undefined @ CPDO_rmf + b CPDO_muf + b CPDO_dvf + b CPDO_rdf + b undefined + b undefined + b undefined + b undefined + b CPDO_mvf + b CPDO_mnf + b CPDO_abs + b CPDO_rnd + b CPDO_sqt + b undefined + b undefined + b undefined + b undefined + b undefined + b undefined + b undefined + b undefined + b undefined + b CPDO_rnd + b fastfpe_next + +CPDO_const: + ldr r2,=fp_const + and r3,r4,#0x00000007 + add r2,r2,r3,lsl#4 + b CPDO_constback + +/*---------------------------------------------------------------------------*/ + +CPRT_enter: + and r0,r4,#0x0000f000 @ r0=Rd<<12 + and r1,r4,#0x00070000 + add r1,r10,r1,lsr#12 @ r1=address of Fn + tst r4,#0x00000008 + bne CPRT_const + and r2,r4,#0x00000007 + add r2,r10,r2,lsl#4 @ r2=address of Fm + +CPRT_constback: + and r3,r4,#0x00f00000 + + add pc,pc,r3,lsr#18 + mov r0,r0 + b CPRT_flt + b CPRT_fix + b CPRT_wfs + b CPRT_rfs + b undefined + b undefined + b undefined + b undefined + b undefined + b CPRT_cmf + b undefined + b CPRT_cnf + b undefined + b CPRT_cmf + b undefined + b CPRT_cnf + +CPRT_const: + ldr r2,=fp_const + and r3,r4,#0x00000007 + add r2,r2,r3,lsl#4 + b CPRT_constback + +/*---------------------------------------------------------------------------*/ + + @ The fetch of the next instruction to emulate could fault + + .section .fixup,"ax" + .align +__f1: + mov pc,r9 + .previous + .section __ex_table,"a" + .align 3 + .long __x1,__f1 + .previous + +/*---------------------------------------------------------------------------*/ diff -urN orig/arch/arm/fastfpe/module.c linux/arch/arm/fastfpe/module.c --- orig/arch/arm/fastfpe/module.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/fastfpe/module.c Fri Dec 28 17:39:51 2001 @@ -0,0 +1,78 @@ +/* + Fast Floating Point Emulator + (c) Peter Teichmann + + This program is free software; you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation; either version 2 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program; if not, write to the Free Software + Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. +*/ + +#include +#include +#include +#include +#include +#include + +#ifndef MODULE +#define kern_fp_enter fp_enter + +extern char fpe_type[]; +#endif + +static void (*orig_fp_enter)(void); /* old kern_fp_enter value */ +extern void (*kern_fp_enter)(void); /* current FP handler */ +extern void fastfpe_enter(void); /* forward declarations */ + +#ifdef MODULE +/* + * Return 0 if we can be unloaded. This can only happen if + * kern_fp_enter is still pointing at fastfpe_enter + */ +static int fpe_unload(void) +{ + return (kern_fp_enter == fastfpe_enter) ? 0 : 1; +} +#endif + +static int __init fpe_init(void) +{ +#ifdef MODULE + if (!mod_member_present(&__this_module, can_unload)) + return -EINVAL; + __this_module.can_unload = fpe_unload; +#else + if (fpe_type[0] && strcmp(fpe_type, "fastfpe")) + return 0; +#endif + + printk("Fast Floating Point Emulator V0.9 (c) Peter Teichmann.\n"); + + /* Save pointer to the old FP handler and then patch ourselves in */ + orig_fp_enter = kern_fp_enter; + kern_fp_enter = fastfpe_enter; + + return 0; +} + +static void __exit fpe_exit(void) +{ + /* Restore the values we saved earlier. */ + kern_fp_enter = orig_fp_enter; +} + +module_init(fpe_init); +module_exit(fpe_exit); + +MODULE_AUTHOR("Peter Teichmann "); +MODULE_DESCRIPTION("Fast floating point emulator with full precision"); diff -urN orig/arch/arm/kernel/Makefile linux/arch/arm/kernel/Makefile --- orig/arch/arm/kernel/Makefile Mon Aug 5 13:29:44 2002 +++ linux/arch/arm/kernel/Makefile Wed Jan 22 23:46:12 2003 @@ -18,6 +18,7 @@ O_OBJS_rpc = dma-rpc.o O_OBJS_footbridge = dma-footbridge.o isa.o O_OBJS_l7200 = fiq.o +O_OBJS_riscstation = fiq.o dma-riscstation.o pci-ftvpci = plx90x0.o pci-footbridge = dec21285.o @@ -43,7 +44,8 @@ no-irq-arch := $(CONFIG_ARCH_INTEGRATOR) $(CONFIG_ARCH_CLPS711X) \ $(CONFIG_FOOTBRIDGE) $(CONFIG_ARCH_EBSA110) \ $(CONFIG_ARCH_SA1100) $(CONFIG_ARCH_CAMELOT) \ - $(CONFIG_ARCH_MX1ADS) + $(CONFIG_ARCH_MX1ADS) $(CONFIG_ARCH_OMAHA) \ + $(CONFIG_ARCH_AT91RM9200DK) ifneq ($(findstring y,$(no-irq-arch)),y) obj-y += irq-arch.o @@ -51,6 +53,7 @@ obj-$(CONFIG_ARCH_ACORN) += ecard.o fiq.o time-acorn.o obj-$(CONFIG_ARCH_CLPS7500) += time-acorn.o +obj-$(CONFIG_ARCH_RISCSTATION) += time-acorn.o obj-$(CONFIG_DEBUG_LL) += debug-$(PROCESSOR).o obj-$(CONFIG_MODULES) += armksyms.o obj-$(CONFIG_ARTHUR) += arthur.o diff -urN orig/arch/arm/kernel/arch.c linux/arch/arm/kernel/arch.c --- orig/arch/arm/kernel/arch.c Mon Sep 3 14:14:39 2001 +++ linux/arch/arm/kernel/arch.c Sun Sep 15 20:18:44 2002 @@ -20,7 +20,7 @@ unsigned int vram_size; -#ifdef CONFIG_ARCH_ACORN +#if defined(CONFIG_ARCH_ACORN) || defined(CONFIG_ARCH_RISCSTATION) unsigned int memc_ctrl_reg; unsigned int number_mfm_drives; @@ -49,7 +49,8 @@ __tagtable(ATAG_ACORN, parse_tag_acorn); -#ifdef CONFIG_ARCH_RPC + +#if defined(CONFIG_ARCH_RPC) || defined(CONFIG_ARCH_RISCSTATION) static void __init fixup_riscpc(struct machine_desc *desc, struct param_struct *unusd, char **cmdline, struct meminfo *mi) @@ -61,7 +62,9 @@ } extern void __init rpc_map_io(void); +extern void __init riscstation_map_io(void); +#ifdef CONFIG_ARCH_RPC MACHINE_START(RISCPC, "Acorn-RiscPC") MAINTAINER("Russell King") BOOT_MEM(0x10000000, 0x03000000, 0xe0000000) @@ -73,6 +76,20 @@ INITIRQ(genarch_init_irq) MACHINE_END #endif + +#ifdef CONFIG_ARCH_RISCSTATION +/* TODO = check all parameters */ +MACHINE_START(RISCSTATION, "RiscStation-RS7500") + MAINTAINER("Ben Dooks, Vincent Sanders") + BOOT_MEM(0x10000000, 0x03000000, 0xe0000000) + BOOT_PARAMS(0x10000100) + FIXUP(fixup_riscpc) + MAPIO(riscstation_map_io) + INITIRQ(genarch_init_irq) +MACHINE_END +#endif + +#endif /* CONFIG_ARCH_RPC | CONFIG_ARCH_RISCSTATION */ #ifdef CONFIG_ARCH_ARC MACHINE_START(ARCHIMEDES, "Acorn-Archimedes") MAINTAINER("Dave Gilbert") diff -urN orig/arch/arm/kernel/armksyms.c linux/arch/arm/kernel/armksyms.c --- orig/arch/arm/kernel/armksyms.c Mon Aug 5 13:29:44 2002 +++ linux/arch/arm/kernel/armksyms.c Sat May 18 21:17:48 2002 @@ -62,9 +62,11 @@ extern void __muldi3(void); extern void __ucmpdi2(void); extern void __udivdi3(void); +extern void __umoddi3(void); extern void __udivmoddi4(void); extern void __udivsi3(void); extern void __umodsi3(void); +extern void abort(void); extern void ret_from_exception(void); extern void fpundefinstr(void); @@ -189,6 +191,7 @@ EXPORT_SYMBOL_NOVERS(memmove); EXPORT_SYMBOL_NOVERS(memcmp); EXPORT_SYMBOL_NOVERS(memscan); +EXPORT_SYMBOL_NOVERS(memchr); EXPORT_SYMBOL_NOVERS(__memzero); /* user mem (segment) */ @@ -228,6 +231,7 @@ EXPORT_SYMBOL_NOVERS(__muldi3); EXPORT_SYMBOL_NOVERS(__ucmpdi2); EXPORT_SYMBOL_NOVERS(__udivdi3); +EXPORT_SYMBOL_NOVERS(__umoddi3); EXPORT_SYMBOL_NOVERS(__udivmoddi4); EXPORT_SYMBOL_NOVERS(__udivsi3); EXPORT_SYMBOL_NOVERS(__umodsi3); diff -urN orig/arch/arm/kernel/arthur.c linux/arch/arm/kernel/arthur.c --- orig/arch/arm/kernel/arthur.c Mon Oct 1 23:10:02 2001 +++ linux/arch/arm/kernel/arthur.c Wed Sep 11 18:48:21 2002 @@ -43,20 +43,48 @@ 24, 25, 26, 27, 28, 29, 30, 31 }; -static unsigned long linux_to_arthur_signals[32] = { - 0, -1, ARTHUR_SIGINT, -1, - ARTHUR_SIGILL, 5, ARTHUR_SIGABRT, 7, - ARTHUR_SIGFPE, 9, ARTHUR_SIGUSR1, ARTHUR_SIGSEGV, - ARTHUR_SIGUSR2, 13, 14, ARTHUR_SIGTERM, - 16, 17, 18, 19, - 20, 21, 22, 23, - 24, 25, 26, 27, - 28, 29, 30, 31 +/* + * Linux to Arthur signal map. + */ +static unsigned long arthur_invmap[32] = { + [0] = 0, + [SIGHUP] = -1, + [SIGINT] = ARTHUR_SIGINT, + [SIGQUIT] = -1, + [SIGILL] = ARTHUR_SIGILL, + [SIGTRAP] = 5, + [SIGABRT] = ARTHUR_SIGABRT, + [SIGBUS] = 7, + [SIGFPE] = ARTHUR_SIGFPE, + [SIGKILL] = 9, + [SIGUSR1] = ARTHUR_SIGUSR1, + [SIGSEGV] = ARTHUR_SIGSEGV, + [SIGUSR2] = ARTHUR_SIGUSR2, + [SIGPIPE] = 13, + [SIGALRM] = 14, + [SIGTERM] = ARTHUR_SIGTERM, + [SIGSTKFLT] = 16, + [SIGCHLD] = 17, + [SIGCONT] = 18, + [SIGSTOP] = 19, + [SIGTSTP] = 20, + [SIGTTIN] = 21, + [SIGTTOU] = 22, + [SIGURG] = 23, + [SIGXCPU] = 24, + [SIGXFSZ] = 25, + [SIGVTALRM] = 26, + [SIGPROF] = 27, + [SIGWINCH] = 28, + [SIGIO] = 29, + [SIGPWR] = 30, + [SIGSYS] = 31 }; static void arthur_lcall7(int nr, struct pt_regs *regs) { struct siginfo info; + info.si_signo = SIGSWI; info.si_errno = nr; /* Bounce it to the emulator */ @@ -64,13 +92,13 @@ } static struct exec_domain arthur_exec_domain = { - "Arthur", /* name */ - arthur_lcall7, - PER_RISCOS, PER_RISCOS, - arthur_to_linux_signals, - linux_to_arthur_signals, - THIS_MODULE, - NULL /* Nothing after this in the list. */ + .name = "Arthur", /* name */ + .handler = arthur_lcall7, + .pers_low = PER_RISCOS, + .pers_high = PER_RISCOS, + .signal_map = arthur_to_linux_signals, + .signal_invmap = arthur_invmap, + .module = THIS_MODULE, }; /* @@ -78,15 +106,18 @@ * processes are using it. */ -int __init arthur_init(void) +static int __init arthur_init(void) { return register_exec_domain(&arthur_exec_domain); } -void __exit arthur_exit(void) +static void __exit arthur_exit(void) { unregister_exec_domain(&arthur_exec_domain); } module_init(arthur_init); module_exit(arthur_exit); + +MODULE_AUTHOR("Philip Blundell"); +MODULE_LICENSE("GPL"); diff -urN orig/arch/arm/kernel/calls.S linux/arch/arm/kernel/calls.S --- orig/arch/arm/kernel/calls.S Mon Aug 5 13:29:44 2002 +++ linux/arch/arm/kernel/calls.S Sat Feb 22 11:24:58 2003 @@ -1,7 +1,7 @@ /* - * linux/arch/arm/lib/calls.h + * linux/arch/arm/kernel/calls.S * - * Copyright (C) 1995-1998 Russell King + * Copyright (C) 1995-2003 Russell King * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as @@ -10,7 +10,7 @@ * This file is included twice in entry-common.S */ #ifndef NR_syscalls -#define NR_syscalls 256 +#define NR_syscalls 288 #else __syscall_start: @@ -21,13 +21,13 @@ .long SYMBOL_NAME(sys_write) /* 5 */ .long SYMBOL_NAME(sys_open) .long SYMBOL_NAME(sys_close) - .long SYMBOL_NAME(sys_waitpid) + .long SYMBOL_NAME(sys_ni_syscall) /* was sys_waitpid */ .long SYMBOL_NAME(sys_creat) .long SYMBOL_NAME(sys_link) /* 10 */ .long SYMBOL_NAME(sys_unlink) .long SYMBOL_NAME(sys_execve_wrapper) .long SYMBOL_NAME(sys_chdir) - .long SYMBOL_NAME(sys_time) + .long SYMBOL_NAME(sys_time) /* used by libc4 */ .long SYMBOL_NAME(sys_mknod) /* 15 */ .long SYMBOL_NAME(sys_chmod) .long SYMBOL_NAME(sys_lchown16) @@ -36,15 +36,15 @@ .long SYMBOL_NAME(sys_lseek) /* 20 */ .long SYMBOL_NAME(sys_getpid) .long SYMBOL_NAME(sys_mount) - .long SYMBOL_NAME(sys_oldumount) + .long SYMBOL_NAME(sys_oldumount) /* used by libc4 */ .long SYMBOL_NAME(sys_setuid16) .long SYMBOL_NAME(sys_getuid16) /* 25 */ .long SYMBOL_NAME(sys_stime) .long SYMBOL_NAME(sys_ptrace) - .long SYMBOL_NAME(sys_alarm) + .long SYMBOL_NAME(sys_alarm) /* used by libc4 */ .long SYMBOL_NAME(sys_ni_syscall) /* was sys_fstat */ .long SYMBOL_NAME(sys_pause) -/* 30 */ .long SYMBOL_NAME(sys_utime) +/* 30 */ .long SYMBOL_NAME(sys_utime) /* used by libc4 */ .long SYMBOL_NAME(sys_ni_syscall) /* was sys_stty */ .long SYMBOL_NAME(sys_ni_syscall) /* was sys_getty */ .long SYMBOL_NAME(sys_access) @@ -62,7 +62,7 @@ /* 45 */ .long SYMBOL_NAME(sys_brk) .long SYMBOL_NAME(sys_setgid16) .long SYMBOL_NAME(sys_getgid16) - .long SYMBOL_NAME(sys_signal) + .long SYMBOL_NAME(sys_ni_syscall) /* was sys_signal */ .long SYMBOL_NAME(sys_geteuid16) /* 50 */ .long SYMBOL_NAME(sys_getegid16) .long SYMBOL_NAME(sys_acct) @@ -82,29 +82,29 @@ /* 65 */ .long SYMBOL_NAME(sys_getpgrp) .long SYMBOL_NAME(sys_setsid) .long SYMBOL_NAME(sys_sigaction) - .long SYMBOL_NAME(sys_sgetmask) - .long SYMBOL_NAME(sys_ssetmask) + .long SYMBOL_NAME(sys_ni_syscall) /* was sys_sgetmask */ + .long SYMBOL_NAME(sys_ni_syscall) /* was sys_ssetmask */ /* 70 */ .long SYMBOL_NAME(sys_setreuid16) .long SYMBOL_NAME(sys_setregid16) .long SYMBOL_NAME(sys_sigsuspend_wrapper) .long SYMBOL_NAME(sys_sigpending) .long SYMBOL_NAME(sys_sethostname) /* 75 */ .long SYMBOL_NAME(sys_setrlimit) - .long SYMBOL_NAME(sys_old_getrlimit) + .long SYMBOL_NAME(sys_old_getrlimit) /* used by libc4 */ .long SYMBOL_NAME(sys_getrusage) .long SYMBOL_NAME(sys_gettimeofday) .long SYMBOL_NAME(sys_settimeofday) /* 80 */ .long SYMBOL_NAME(sys_getgroups16) .long SYMBOL_NAME(sys_setgroups16) - .long SYMBOL_NAME(old_select) + .long SYMBOL_NAME(old_select) /* used by libc4 */ .long SYMBOL_NAME(sys_symlink) .long SYMBOL_NAME(sys_ni_syscall) /* was sys_lstat */ /* 85 */ .long SYMBOL_NAME(sys_readlink) .long SYMBOL_NAME(sys_uselib) .long SYMBOL_NAME(sys_swapon) .long SYMBOL_NAME(sys_reboot) - .long SYMBOL_NAME(old_readdir) -/* 90 */ .long SYMBOL_NAME(old_mmap) + .long SYMBOL_NAME(old_readdir) /* used by libc4 */ +/* 90 */ .long SYMBOL_NAME(old_mmap) /* used by libc4 */ .long SYMBOL_NAME(sys_munmap) .long SYMBOL_NAME(sys_truncate) .long SYMBOL_NAME(sys_ftruncate) @@ -240,23 +240,37 @@ .long SYMBOL_NAME(sys_ni_syscall) /* Security */ .long SYMBOL_NAME(sys_gettid) /* 225 */ .long SYMBOL_NAME(sys_readahead) - .long SYMBOL_NAME(sys_ni_syscall) /* sys_setxattr */ - .long SYMBOL_NAME(sys_ni_syscall) /* sys_lsetxattr */ - .long SYMBOL_NAME(sys_ni_syscall) /* sys_fsetxattr */ - .long SYMBOL_NAME(sys_ni_syscall) /* sys_getxattr */ -/* 230 */ .long SYMBOL_NAME(sys_ni_syscall) /* sys_lgetxattr */ - .long SYMBOL_NAME(sys_ni_syscall) /* sys_fgetxattr */ - .long SYMBOL_NAME(sys_ni_syscall) /* sys_listxattr */ - .long SYMBOL_NAME(sys_ni_syscall) /* sys_llistxattr */ - .long SYMBOL_NAME(sys_ni_syscall) /* sys_flistxattr */ -/* 235 */ .long SYMBOL_NAME(sys_ni_syscall) /* sys_removexattr */ - .long SYMBOL_NAME(sys_ni_syscall) /* sys_lremovexattr */ - .long SYMBOL_NAME(sys_ni_syscall) /* sys_fremovexattr */ + .long SYMBOL_NAME(sys_ni_syscall) /* setxattr */ + .long SYMBOL_NAME(sys_ni_syscall) /* lsetxattr */ + .long SYMBOL_NAME(sys_ni_syscall) /* fsetxattr */ + .long SYMBOL_NAME(sys_ni_syscall) /* getxattr */ +/* 230 */ .long SYMBOL_NAME(sys_ni_syscall) /* lgetxattr */ + .long SYMBOL_NAME(sys_ni_syscall) /* fgetxattr */ + .long SYMBOL_NAME(sys_ni_syscall) /* listxattr */ + .long SYMBOL_NAME(sys_ni_syscall) /* llistxattr */ + .long SYMBOL_NAME(sys_ni_syscall) /* flistxattr */ +/* 235 */ .long SYMBOL_NAME(sys_ni_syscall) /* removexattr */ + .long SYMBOL_NAME(sys_ni_syscall) /* lremovexattr */ + .long SYMBOL_NAME(sys_ni_syscall) /* fremovexattr */ .long SYMBOL_NAME(sys_tkill) - /* - * Please check 2.5 _before_ adding calls here, - * and copy changes to rmk@arm.linux.org.uk. Thanks. - */ + .long SYMBOL_NAME(sys_ni_syscall) /* sendfile64 */ +/* 240 */ .long SYMBOL_NAME(sys_ni_syscall) /* futex */ + .long SYMBOL_NAME(sys_ni_syscall) /* sched_setaffinity */ + .long SYMBOL_NAME(sys_ni_syscall) /* sched_getaffinity */ + .long SYMBOL_NAME(sys_ni_syscall) /* io_setup */ + .long SYMBOL_NAME(sys_ni_syscall) /* io_destroy */ +/* 245 */ .long SYMBOL_NAME(sys_ni_syscall) /* io_getevents */ + .long SYMBOL_NAME(sys_ni_syscall) /* io_submit */ + .long SYMBOL_NAME(sys_ni_syscall) /* io_cancel */ + .long SYMBOL_NAME(sys_ni_syscall) /* exit_group */ + .long SYMBOL_NAME(sys_ni_syscall) /* lookup_dcookie */ +/* 250 */ .long SYMBOL_NAME(sys_ni_syscall) /* epoll_create */ + .long SYMBOL_NAME(sys_ni_syscall) /* epoll_ctl */ + .long SYMBOL_NAME(sys_ni_syscall) /* epoll_wait */ + .long SYMBOL_NAME(sys_ni_syscall) /* remap_file_pages */ + .long SYMBOL_NAME(sys_ni_syscall) /* set_thread_area */ +/* 255 */ .long SYMBOL_NAME(sys_ni_syscall) /* get_thread_area */ + .long SYMBOL_NAME(sys_ni_syscall) /* set_tid_address */ __syscall_end: .rept NR_syscalls - (__syscall_end - __syscall_start) / 4 diff -urN orig/arch/arm/kernel/compat.c linux/arch/arm/kernel/compat.c --- orig/arch/arm/kernel/compat.c Mon Aug 5 13:29:44 2002 +++ linux/arch/arm/kernel/compat.c Thu Sep 5 20:56:38 2002 @@ -33,18 +33,35 @@ return tag; } -static void __init build_tag_list(struct param_struct *params, void *taglist, int mem_init) +static void __init build_tag_list(struct param_struct *params, void *taglist) { struct tag *tag = taglist; - printk(KERN_DEBUG "Converting old-style param struct to taglist\n"); - if (params->u1.s.page_size != PAGE_SIZE) { printk(KERN_WARNING "Warning: bad configuration page, " "trying to continue\n"); return; } + printk(KERN_DEBUG "Converting old-style param struct to taglist\n"); + +#ifdef CONFIG_ARCH_NETWINDER + if (params->u1.s.nr_pages != 0x02000 && + params->u1.s.nr_pages != 0x04000 && + params->u1.s.nr_pages != 0x08000 && + params->u1.s.nr_pages != 0x10000) { + printk(KERN_WARNING "Warning: bad NeTTrom parameters " + "detected, using defaults\n"); + + params->u1.s.nr_pages = 0x1000; /* 16MB */ + params->u1.s.ramdisk_size = 0; + params->u1.s.flags = FLAG_READONLY; + params->u1.s.initrd_start = 0; + params->u1.s.initrd_size = 0; + params->u1.s.rd_start = 0; + } +#endif + tag->hdr.tag = ATAG_CORE; tag->hdr.size = tag_size(tag_core); tag->u.core.flags = params->u1.s.flags & FLAG_READONLY; @@ -76,17 +93,15 @@ tag->hdr.size = tag_size(tag_revision); tag->u.revision.rev = params->u1.s.system_rev; - if (mem_init) { #ifdef CONFIG_ARCH_ACORN - if (machine_is_riscpc()) { - int i; - for (i = 0; i < 4; i++) - tag = memtag(tag, PHYS_OFFSET + (i << 26), - params->u1.s.pages_in_bank[i] * PAGE_SIZE); - } else + if (machine_is_riscpc()) { + int i; + for (i = 0; i < 4; i++) + tag = memtag(tag, PHYS_OFFSET + (i << 26), + params->u1.s.pages_in_bank[i] * PAGE_SIZE); + } else #endif - tag = memtag(tag, PHYS_OFFSET, params->u1.s.nr_pages * PAGE_SIZE); - } + tag = memtag(tag, PHYS_OFFSET, params->u1.s.nr_pages * PAGE_SIZE); #ifdef CONFIG_FOOTBRIDGE if (params->u1.s.mem_fclk_21285) { @@ -97,6 +112,23 @@ } #endif +#ifdef CONFIG_ARCH_EBSA285 + if (machine_is_ebsa285()) { + tag = tag_next(tag); + tag->hdr.tag = ATAG_VIDEOTEXT; + tag->hdr.size = tag_size(tag_videotext); + tag->u.videotext.x = params->u1.s.video_x; + tag->u.videotext.y = params->u1.s.video_y; + tag->u.videotext.video_page = 0; + tag->u.videotext.video_mode = 0; + tag->u.videotext.video_cols = params->u1.s.video_num_cols; + tag->u.videotext.video_ega_bx = 0; + tag->u.videotext.video_lines = params->u1.s.video_num_rows; + tag->u.videotext.video_isvga = 1; + tag->u.videotext.video_points = 8; + } +#endif + #ifdef CONFIG_ARCH_ACORN tag = tag_next(tag); tag->hdr.tag = ATAG_ACORN; @@ -114,14 +146,22 @@ strcpy(tag->u.cmdline.cmdline, params->commandline); tag = tag_next(tag); - tag->hdr.tag = 0; + tag->hdr.tag = ATAG_NONE; tag->hdr.size = 0; memmove(params, taglist, ((int)tag) - ((int)taglist) + sizeof(struct tag_header)); } -void __init convert_to_tag_list(struct param_struct *params, int mem_init) +void __init convert_to_tag_list(struct tag *tags) +{ + struct param_struct *params = (struct param_struct *)tags; + build_tag_list(params, ¶ms->u2); +} + +void __init squash_mem_tags(struct tag *tag) { - build_tag_list(params, ¶ms->u2, mem_init); + for (; tag->hdr.size; tag = tag_next(tag)) + if (tag->hdr.tag == ATAG_MEM) + tag->hdr.tag = ATAG_NONE; } diff -urN orig/arch/arm/kernel/debug-armv.S linux/arch/arm/kernel/debug-armv.S --- orig/arch/arm/kernel/debug-armv.S Mon Aug 5 13:29:44 2002 +++ linux/arch/arm/kernel/debug-armv.S Wed Jan 22 23:46:12 2003 @@ -20,9 +20,20 @@ * printk isn't working). For DEBUGGING ONLY!!! Do not leave * references to these in a production kernel! */ -#if defined(CONFIG_ARCH_RPC) +#if defined(CONFIG_ARCH_RPC) || defined(CONFIG_ARCH_RISCSTATION) .macro addruart,rx - mov \rx, #0xe0000000 + +/* note, errors can occur in the head code BEFORE the memory map has been + * setup (and therefore the mapping for the IO) so the #if statement allows + * the address of the uart to be changed to the physical address instead of + * the logical + * + * (bjd) + */ + mrc p15, 0, \rx, c1, c0 + tst \rx, #1 @ MMU enabled? + movne \rx, #0xe0000000 + moveq \rx, #0x03000000 orr \rx, \rx, #0x00010000 orr \rx, \rx, #0x00000fe0 .endm @@ -66,7 +77,7 @@ tst \rd, #0x10 beq 1001b .endm - + #elif defined(CONFIG_ARCH_SHARK) .macro addruart,rx mov \rx, #0xe0000000 @@ -287,6 +298,27 @@ tst \rd, #1 << 3 @ UARTFLGUBUSY - 1 when busy bne 1001b .endm +#elif defined(CONFIG_ARCH_AT91RM9200DK) + + .macro addruart,rx + ldr \rx, =AT91C_BASE_DBGU @ DBGU base + .endm + + .macro senduart,rd,rx + strb \rd, [\rx, #DBGU_THR] + .endm + + .macro waituart,rd,rx +1001: ldr \rd, [\rx, #DBGU_CSR] @ UARTFLAG + tst \rd, #AT91C_US_TXRDY @ UARTFLGUTXFF - 1 when full + beq 1001b + .endm + + .macro busyuart,rd,rx +1001: ldr \rd, [\rx, #DBGU_CSR] @ UARTFLG + tst \rd, #AT91C_US_TXRDY @ UARTFLGUBUSY - 1 when busy + beq 1001b + .endm #elif defined(CONFIG_ARCH_CLPS711X) @@ -322,6 +354,21 @@ bne 1001b 1002: .endm +#elif defined(CONFIG_ARCH_OMAHA) + +#include + + .macro addruart,rx + .endm + + .macro senduart,rd,rx + .endm + + .macro waituart,rd,rx + .endm + + .macro busyuart,rd,rx + .endm #elif defined(CONFIG_ARCH_ANAKIN) @@ -365,7 +412,7 @@ tst \rx, #1 @ MMU enabled? ldr \rx, =EXC_UART00_BASE @ physical base address orrne \rx, \rx, #0xff000000 @ virtual base - orrne \rx, \rx, #0x00f00000 + orrne \rx, \rx, #0x00f00000 .endm .macro senduart,rd,rx diff -urN orig/arch/arm/kernel/dma-arc.c linux/arch/arm/kernel/dma-arc.c --- orig/arch/arm/kernel/dma-arc.c Mon Aug 5 13:29:44 2002 +++ linux/arch/arm/kernel/dma-arc.c Sun Sep 15 12:54:03 2002 @@ -34,14 +34,14 @@ extern void fdc1772_setupdma(unsigned int count,unsigned int addr); unsigned long flags; DPRINTK("enable_dma fdc1772 data read\n"); - save_flags(flags); - clf(); + local_save_flags(flags); + __clf(); memcpy ((void *)0x1c, (void *)&fdc1772_dma_read, &fdc1772_dma_read_end - &fdc1772_dma_read); fdc1772_setupdma(dma->buf.length, dma->buf.address); /* Sets data pointer up */ enable_fiq(FIQ_FLOPPYDATA); - restore_flags(flags); + local_irq_restore(flags); } break; @@ -50,14 +50,14 @@ extern void fdc1772_setupdma(unsigned int count,unsigned int addr); unsigned long flags; DPRINTK("enable_dma fdc1772 data write\n"); - save_flags(flags); - clf(); + local_save_flags(flags); + __clf(); memcpy ((void *)0x1c, (void *)&fdc1772_dma_write, &fdc1772_dma_write_end - &fdc1772_dma_write); fdc1772_setupdma(dma->buf.length, dma->buf.address); /* Sets data pointer up */ enable_fiq(FIQ_FLOPPYDATA; - restore_flags(flags); + local_irq_restore(flags); } break; default: @@ -81,14 +81,14 @@ DPRINTK("arc_floppy_cmdend_enable_dma\n"); /*printk("enable_dma fdc1772 command end FIQ\n");*/ - save_flags(flags); - clf(); + local_save_flags(flags); + __clf(); /* B fdc1772_comendhandler */ *((unsigned int *)0x1c)=0xea000000 | (((unsigned int)fdc1772_comendhandler-(0x1c+8))/4); - restore_flags(flags); + local_irq_restore(flags); } static int arc_floppy_cmdend_get_dma_residue(dmach_t channel, dma_t *dma) @@ -106,23 +106,23 @@ } static struct dma_ops arc_floppy_data_dma_ops = { - type: "FIQDMA", - enable: arc_floppy_data_enable_dma, - disable: arc_disable_dma, - residue: arc_floppy_data_get_dma_residue, + .type = "FIQDMA", + .enable = arc_floppy_data_enable_dma, + .disable = arc_disable_dma, + .residue = arc_floppy_data_get_dma_residue, }; static struct dma_ops arc_floppy_cmdend_dma_ops = { - type: "FIQCMD", - enable: arc_floppy_cmdend_enable_dma, - disable: arc_disable_dma, - residue: arc_floppy_cmdend_get_dma_residue, + .type = "FIQCMD", + .enable = arc_floppy_cmdend_enable_dma, + .disable = arc_disable_dma, + .residue = arc_floppy_cmdend_get_dma_residue, }; #endif #ifdef CONFIG_ARCH_A5K static struct fiq_handler fh = { - name: "floppydata" + .name = "floppydata", }; static int a5k_floppy_get_dma_residue(dmach_t channel, dma_t *dma) @@ -168,10 +168,10 @@ } static struct dma_ops a5k_floppy_dma_ops = { - type: "FIQDMA", - enable: a5k_floppy_enable_dma, - disable: a5k_floppy_disable_dma, - residue: a5k_floppy_get_dma_residue, + .type = "FIQDMA", + .enable = a5k_floppy_enable_dma, + .disable = a5k_floppy_disable_dma, + .residue = a5k_floppy_get_dma_residue, }; #endif @@ -183,9 +183,9 @@ } static struct dma_ops sound_dma_ops = { - type: "VIRTUAL", - enable: sound_enable_disable_dma, - disable: sound_enable_disable_dma, + .type = "VIRTUAL", + .enable = sound_enable_disable_dma, + .disable = sound_enable_disable_dma, }; void __init arch_dma_init(dma_t *dma) diff -urN orig/arch/arm/kernel/dma-footbridge.c linux/arch/arm/kernel/dma-footbridge.c --- orig/arch/arm/kernel/dma-footbridge.c Tue Oct 3 20:08:18 2000 +++ linux/arch/arm/kernel/dma-footbridge.c Sun Sep 15 12:55:39 2002 @@ -35,10 +35,10 @@ } static struct dma_ops fb_dma_ops = { - type: "fb", - request: fb_dma_request, - enable: fb_dma_enable, - disable: fb_dma_disable, + .type = "fb", + .request = fb_dma_request, + .enable = fb_dma_enable, + .disable = fb_dma_disable, }; #endif diff -urN orig/arch/arm/kernel/dma-isa.c linux/arch/arm/kernel/dma-isa.c --- orig/arch/arm/kernel/dma-isa.c Fri Oct 26 16:45:50 2001 +++ linux/arch/arm/kernel/dma-isa.c Thu Sep 5 20:22:19 2002 @@ -133,10 +133,10 @@ } static struct dma_ops isa_dma_ops = { - type: "ISA", - enable: isa_enable_dma, - disable: isa_disable_dma, - residue: isa_get_dma_residue, + .type = "ISA", + .enable = isa_enable_dma, + .disable = isa_disable_dma, + .residue = isa_get_dma_residue, }; static struct resource dma_resources[] = { diff -urN orig/arch/arm/kernel/dma-riscstation.c linux/arch/arm/kernel/dma-riscstation.c --- orig/arch/arm/kernel/dma-riscstation.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/kernel/dma-riscstation.c Sun Sep 15 20:18:44 2002 @@ -0,0 +1,113 @@ +/* + * linux/arch/arm/kernel/dma-riscstation.c + * + * Copyright (C) 1998 Russell King + * Copyright (C) 2002 Ben Dooks / Simtec Electronics + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + * + * DMA functions specific to RiscStatiun architecture + * + * sliced down version of the RiscPC DMA code by RMK + */ +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include + +#include +#include + +static struct fiq_handler fh = { + name: "floppydma" +}; + +static void floppy_enable_dma(dmach_t channel, dma_t *dma) +{ + void *fiqhandler_start; + unsigned int fiqhandler_length; + struct pt_regs regs; + + if (dma->dma_mode == DMA_MODE_READ) { + extern unsigned char floppy_fiqin_start, floppy_fiqin_end; + fiqhandler_start = &floppy_fiqin_start; + fiqhandler_length = &floppy_fiqin_end - &floppy_fiqin_start; + } else { + extern unsigned char floppy_fiqout_start, floppy_fiqout_end; + fiqhandler_start = &floppy_fiqout_start; + fiqhandler_length = &floppy_fiqout_end - &floppy_fiqout_start; + } + + regs.ARM_r9 = dma->buf.length; + regs.ARM_r10 = (unsigned long)dma->buf.address; + regs.ARM_fp = FLOPPYDMA_BASE; + + if (claim_fiq(&fh)) { + printk("floppydma: couldn't claim FIQ.\n"); + return; + } + + set_fiq_handler(fiqhandler_start, fiqhandler_length); + set_fiq_regs(®s); + enable_fiq(dma->dma_irq); +} + +static void floppy_disable_dma(dmach_t channel, dma_t *dma) +{ + disable_fiq(dma->dma_irq); + release_fiq(&fh); +} + +static int floppy_get_residue(dmach_t channel, dma_t *dma) +{ + struct pt_regs regs; + get_fiq_regs(®s); + return regs.ARM_r9; +} + +static struct dma_ops floppy_dma_ops = { + type: "FIQDMA", + enable: floppy_enable_dma, + disable: floppy_disable_dma, + residue: floppy_get_residue, +}; + +/* + * This is virtual DMA - we don't need anything here. + */ +static void sound_enable_disable_dma(dmach_t channel, dma_t *dma) +{ +} + +static struct dma_ops sound_dma_ops = { + type: "VIRTUAL", + enable: sound_enable_disable_dma, + disable: sound_enable_disable_dma, +}; + +void __init arch_dma_init(dma_t *dma) +{ +#if 0 + iomd_writeb(0, IOMD_IO0CR); + iomd_writeb(0, IOMD_IO1CR); + iomd_writeb(0, IOMD_IO2CR); + iomd_writeb(0, IOMD_IO3CR); + + iomd_writeb(0xa0, IOMD_DMATCR); +#endif + + dma[DMA_VIRTUAL_FLOPPY].dma_irq = FIQ_FLOPPYDATA; + dma[DMA_VIRTUAL_FLOPPY].d_ops = &floppy_dma_ops; + dma[DMA_VIRTUAL_SOUND].d_ops = &sound_dma_ops; +} diff -urN orig/arch/arm/kernel/dma-rpc.c linux/arch/arm/kernel/dma-rpc.c --- orig/arch/arm/kernel/dma-rpc.c Thu Feb 22 11:24:58 2001 +++ linux/arch/arm/kernel/dma-rpc.c Sat Feb 22 11:40:58 2003 @@ -91,76 +91,40 @@ sg->length |= flags; } -static inline void iomd_setup_dma_a(struct scatterlist *sg, dma_t *dma) -{ - iomd_writel(sg->dma_address, dma->dma_base + CURA); - iomd_writel(sg->length, dma->dma_base + ENDA); -} - -static inline void iomd_setup_dma_b(struct scatterlist *sg, dma_t *dma) -{ - iomd_writel(sg->dma_address, dma->dma_base + CURB); - iomd_writel(sg->length, dma->dma_base + ENDB); -} - static void iomd_dma_handle(int irq, void *dev_id, struct pt_regs *regs) { dma_t *dma = (dma_t *)dev_id; - unsigned int status = 0, no_buffer = dma->sg == NULL; + unsigned long base = dma->dma_base; do { - switch (dma->state) { - case state_prog_a: - iomd_get_next_sg(&dma->cur_sg, dma); - iomd_setup_dma_a(&dma->cur_sg, dma); - dma->state = state_wait_a; - - case state_wait_a: - status = iomd_readb(dma->dma_base + ST); - switch (status & (DMA_ST_OFL|DMA_ST_INT|DMA_ST_AB)) { - case DMA_ST_OFL|DMA_ST_INT: - iomd_get_next_sg(&dma->cur_sg, dma); - iomd_setup_dma_a(&dma->cur_sg, dma); - break; - - case DMA_ST_INT: - iomd_get_next_sg(&dma->cur_sg, dma); - iomd_setup_dma_b(&dma->cur_sg, dma); - dma->state = state_wait_b; - break; - - case DMA_ST_OFL|DMA_ST_INT|DMA_ST_AB: - iomd_setup_dma_b(&dma->cur_sg, dma); - dma->state = state_wait_b; - break; - } + unsigned int status; + + status = iomd_readb(base + ST); + if (!(status & DMA_ST_INT)) + return; + + if (status & DMA_ST_OFL && !dma->sg) + break; + + iomd_get_next_sg(&dma->cur_sg, dma); + + switch (status & (DMA_ST_OFL | DMA_ST_AB)) { + case DMA_ST_OFL: /* OIA */ + case DMA_ST_AB: /* .IB */ + iomd_writel(dma->cur_sg.dma_address, base + CURA); + iomd_writel(dma->cur_sg.length, base + ENDA); break; - case state_wait_b: - status = iomd_readb(dma->dma_base + ST); - switch (status & (DMA_ST_OFL|DMA_ST_INT|DMA_ST_AB)) { - case DMA_ST_OFL|DMA_ST_INT|DMA_ST_AB: - iomd_get_next_sg(&dma->cur_sg, dma); - iomd_setup_dma_b(&dma->cur_sg, dma); - break; - - case DMA_ST_INT|DMA_ST_AB: - iomd_get_next_sg(&dma->cur_sg, dma); - iomd_setup_dma_a(&dma->cur_sg, dma); - dma->state = state_wait_a; - break; - - case DMA_ST_OFL|DMA_ST_INT: - iomd_setup_dma_a(&dma->cur_sg, dma); - dma->state = state_wait_a; - break; - } + case DMA_ST_OFL | DMA_ST_AB: /* OIB */ + case 0: /* .IA */ + iomd_writel(dma->cur_sg.dma_address, base + CURB); + iomd_writel(dma->cur_sg.length, base + ENDB); break; } - } while (dma->sg && (status & DMA_ST_INT)); + } while (1); - if (no_buffer) - disable_irq(irq); + iomd_writeb(0, base + CR); + disable_irq(irq); } static int iomd_request_dma(dmach_t channel, dma_t *dma) @@ -194,7 +158,6 @@ } iomd_writeb(DMA_CR_C, dma_base + CR); - dma->state = state_prog_a; } if (dma->dma_mode == DMA_MODE_READ) @@ -207,11 +170,15 @@ static void iomd_disable_dma(dmach_t channel, dma_t *dma) { unsigned long dma_base = dma->dma_base; + unsigned long flags; unsigned int ctrl; - disable_irq(dma->dma_irq); + local_irq_save(flags); ctrl = iomd_readb(dma_base + CR); + if (ctrl & DMA_CR_E) + disable_irq(dma->dma_irq); iomd_writeb(ctrl & ~DMA_CR_E, dma_base + CR); + local_irq_restore(flags); } static int iomd_set_dma_speed(dmach_t channel, dma_t *dma, int cycle) @@ -257,16 +224,16 @@ } static struct dma_ops iomd_dma_ops = { - type: "IOMD", - request: iomd_request_dma, - free: iomd_free_dma, - enable: iomd_enable_dma, - disable: iomd_disable_dma, - setspeed: iomd_set_dma_speed, + .type = "IOMD", + .request = iomd_request_dma, + .free = iomd_free_dma, + .enable = iomd_enable_dma, + .disable = iomd_disable_dma, + .setspeed = iomd_set_dma_speed, }; static struct fiq_handler fh = { - name: "floppydma" + .name = "floppydma" }; static void floppy_enable_dma(dmach_t channel, dma_t *dma) @@ -313,10 +280,10 @@ } static struct dma_ops floppy_dma_ops = { - type: "FIQDMA", - enable: floppy_enable_dma, - disable: floppy_disable_dma, - residue: floppy_get_residue, + .type = "FIQDMA", + .enable = floppy_enable_dma, + .disable = floppy_disable_dma, + .residue = floppy_get_residue, }; /* @@ -327,9 +294,9 @@ } static struct dma_ops sound_dma_ops = { - type: "VIRTUAL", - enable: sound_enable_disable_dma, - disable: sound_enable_disable_dma, + .type = "VIRTUAL", + .enable = sound_enable_disable_dma, + .disable = sound_enable_disable_dma, }; void __init arch_dma_init(dma_t *dma) diff -urN orig/arch/arm/kernel/dma.c linux/arch/arm/kernel/dma.c --- orig/arch/arm/kernel/dma.c Mon Aug 5 13:29:44 2002 +++ linux/arch/arm/kernel/dma.c Sun Sep 15 16:08:52 2002 @@ -119,6 +119,10 @@ { dma_t *dma = dma_chan + channel; + if (dma->active) + printk(KERN_ERR "dma%d: altering DMA SG while " + "DMA active\n", channel); + dma->sg = sg; dma->sgcount = nr_sg; dma->using_sg = 1; @@ -217,6 +221,14 @@ BUG(); } +/* + * Is the specified DMA channel active? + */ +int dma_channel_active(dmach_t channel) +{ + return dma_chan[channel].active; +} + void set_dma_page(dmach_t channel, char pagenr) { printk(KERN_ERR "dma%d: trying to set_dma_page\n", channel); @@ -272,6 +284,7 @@ GLOBAL_ALIAS(set_dma_sg, get_dma_residue); GLOBAL_ALIAS(set_dma_speed, get_dma_residue); GLOBAL_ALIAS(init_dma, get_dma_residue); +GLOBAL_ALIAS(dma_channel_active, get_dma_residue); #endif @@ -286,3 +299,6 @@ EXPORT_SYMBOL(get_dma_residue); EXPORT_SYMBOL(set_dma_sg); EXPORT_SYMBOL(set_dma_speed); +EXPORT_SYMBOL(dma_channel_active); + +EXPORT_SYMBOL(dma_spin_lock); diff -urN orig/arch/arm/kernel/ecard.c linux/arch/arm/kernel/ecard.c --- orig/arch/arm/kernel/ecard.c Mon Sep 3 14:14:40 2001 +++ linux/arch/arm/kernel/ecard.c Thu Feb 6 11:30:47 2003 @@ -38,6 +38,8 @@ #include #include #include +#include +#include #include #include @@ -420,7 +422,7 @@ } static struct notifier_block ecard_reboot_notifier = { - notifier_call: ecard_reboot, + .notifier_call = ecard_reboot, }; diff -urN orig/arch/arm/kernel/entry-armv.S linux/arch/arm/kernel/entry-armv.S --- orig/arch/arm/kernel/entry-armv.S Mon Aug 5 13:29:44 2002 +++ linux/arch/arm/kernel/entry-armv.S Fri Feb 21 15:36:03 2003 @@ -187,11 +187,10 @@ .macro get_irqnr_and_base, irqnr, irqstat, base, tmp mov r4, #0xe0000000 - orr r4, r4, #0x20 mov \irqstat, #0x0C - strb \irqstat, [r4] @outb(0x0C, 0x20) /* Poll command */ - ldrb \irqnr, [r4] @irq = inb(0x20) & 7 + strb \irqstat, [r4, #0x20] @outb(0x0C, 0x20) /* Poll command */ + ldrb \irqnr, [r4, #0x20] @irq = inb(0x20) & 7 and \irqstat, \irqnr, #0x80 teq \irqstat, #0 beq 43f @@ -199,8 +198,8 @@ teq \irqnr, #2 bne 44f 43: mov \irqstat, #0x0C - strb \irqstat, [r4, #0x80] @outb(0x0C, 0xA0) /* Poll command */ - ldrb \irqnr, [r4, #0x80] @irq = (inb(0xA0) & 7) + 8 + strb \irqstat, [r4, #0xa0] @outb(0x0C, 0xA0) /* Poll command */ + ldrb \irqnr, [r4, #0xa0] @irq = (inb(0xA0) & 7) + 8 and \irqstat, \irqnr, #0x80 teq \irqstat, #0 beq 44f @@ -453,7 +452,23 @@ .macro irq_prio_table .endm +#elif defined(CONFIG_ARCH_AT91RM9200DK) +#include + .macro disable_fiq + .endm + + .macro get_irqnr_and_base, irqnr, irqstat, base, tmp + ldr \base, =AT91C_VA_BASE_AIC @ IO Base add vir + ldr \irqnr, [\base, #AIC_IVR] + ldr \irqstat, [\base, #AIC_ISR] @ read interrupt source number + str \tmp, [\base, #AIC_EOICR] + teq \irqstat, #0 +1001: + .endm + + .macro irq_prio_table + .endm #elif defined(CONFIG_ARCH_MX1ADS) .macro disable_fiq @@ -479,6 +494,32 @@ .macro irq_prio_table .endm +#elif defined(CONFIG_ARCH_OMAHA) + + .macro disable_fiq + .endm + + .macro get_irqnr_and_base, irqnr, irqstat, base, tmp + + /* Read all interrupts pending... */ + ldr \irqnr, =IO_ADDRESS(PLAT_PERIPHERAL_BASE) + OMAHA_INTPND + ldr \irqstat, [\irqnr] /* INTPND */ + + /* All pending irqs are now in \irqstat */ + mov \irqnr, #0 +1001: tst \irqstat, #1 + bne 1002f + add \irqnr, \irqnr, #1 + mov \irqstat, \irqstat, lsr #1 + cmp \irqnr, #MAXIRQNUM + bcc 1001b +1002: /* EQ will be set if we reach MAXIRQNUM */ + + .endm + + .macro irq_prio_table + .endm + #elif defined(CONFIG_ARCH_CLPS711X) #include @@ -611,16 +652,6 @@ and r2, r6, #31 @ int mode b SYMBOL_NAME(bad_mode) -#if defined CONFIG_FPE_NWFPE || defined CONFIG_FPE_FASTFPE - /* The FPE is always present */ - .equ fpe_not_present, 0 -#else -wfs_mask_data: .word 0x0e200110 @ WFS/RFS - .word 0x0fef0fff - .word 0x0d000100 @ LDF [sp]/STF [sp] - .word 0x0d000100 @ LDF [fp]/STF [fp] - .word 0x0f000f00 - /* We get here if an undefined instruction happens and the floating * point emulator is not present. If the offending instruction was * a WFS, we just perform a normal return as if we had emulated the @@ -628,30 +659,7 @@ * to run so that the emulator module proper can be loaded. --philb */ fpe_not_present: - adr r10, wfs_mask_data - ldmia r10, {r4, r5, r6, r7, r8} - ldr r10, [sp, #S_PC] @ Load PC - sub r10, r10, #4 - mask_pc r10, r10 - ldrt r10, [r10] @ get instruction - and r5, r10, r5 - teq r5, r4 @ Is it WFS? - moveq pc, r9 - and r5, r10, r8 - teq r5, r6 @ Is it LDF/STF on sp or fp? - teqne r5, r7 - movne pc, lr - tst r10, #0x00200000 @ Does it have WB - moveq pc, r9 - and r4, r10, #255 @ get offset - and r6, r10, #0x000f0000 - tst r10, #0x00800000 @ +/- - ldr r5, [sp, r6, lsr #14] @ Load reg - rsbeq r4, r4, #0 - add r5, r5, r4, lsl #2 - str r5, [sp, r6, lsr #14] @ Save reg - mov pc, r9 -#endif + mov pc, lr /* * SVC mode handlers @@ -703,11 +711,13 @@ @ routine called with r0 = irq number, r1 = struct pt_regs * @ adrsvc ne, lr, 1b - bne do_IRQ + bne asm_do_IRQ ldr r0, [sp, #S_PSR] @ irqs are already disabled msr spsr, r0 ldmia sp, {r0 - pc}^ @ load r0 - pc, cpsr + .ltorg + .align 5 __und_svc: sub sp, sp, #S_FRAME_SIZE stmia sp, {r0 - r12} @ save r0 - r12 @@ -807,11 +817,13 @@ @ @ routine called with r0 = irq number, r1 = struct pt_regs * @ - bne do_IRQ + bne asm_do_IRQ mov why, #0 get_current_task tsk b ret_to_user + .ltorg + .align 5 __und_usr: sub sp, sp, #S_FRAME_SIZE @ Allocate frame size in one go stmia sp, {r0 - r12} @ Save r0 - r12 @@ -822,6 +834,8 @@ stmdb r8, {sp, lr}^ @ Save user sp, lr alignment_trap r4, r7, __temp_und zero_fp + tst r6, #T_BIT @ Thumb mode + bne fpundefinstr adrsvc al, r9, ret_from_exception @ r9 = normal FP return adrsvc al, lr, fpundefinstr @ lr = undefined instr return @@ -859,8 +873,9 @@ * This is the return code to user mode for abort handlers */ ENTRY(ret_from_exception) - get_current_task tsk + disable_irq r1 mov why, #0 + get_current_task tsk b ret_to_user .data @@ -909,7 +924,9 @@ @ @ now branch to the relevent MODE handling routine @ - mov r13, #I_BIT | MODE_SVC + mrs r13, cpsr + bic r13, r13, #MODE_MASK + orr r13, r13, #I_BIT | MODE_SVC msr spsr_c, r13 @ switch to SVC_32 mode and lr, lr, #15 @@ -950,7 +967,9 @@ @ @ now branch to the relevent MODE handling routine @ - mov r13, #I_BIT | MODE_SVC + mrs r13, cpsr + bic r13, r13, #MODE_MASK + orr r13, r13, #I_BIT | MODE_SVC msr spsr_c, r13 @ switch to SVC_32 mode and lr, lr, #15 @@ -992,7 +1011,9 @@ @ @ now branch to the relevent MODE handling routine @ - mov r13, #I_BIT | MODE_SVC + mrs r13, cpsr + bic r13, r13, #MODE_MASK + orr r13, r13, #I_BIT | MODE_SVC msr spsr_c, r13 @ switch to SVC_32 mode ands lr, lr, #15 @@ -1033,7 +1054,9 @@ @ @ now branch to the relevent MODE handling routine @ - mov r13, #I_BIT | MODE_SVC + mrs r13, cpsr + bic r13, r13, #MODE_MASK + orr r13, r13, #I_BIT | MODE_SVC msr spsr_c, r13 @ switch to SVC_32 mode and lr, lr, #15 diff -urN orig/arch/arm/kernel/entry-common.S linux/arch/arm/kernel/entry-common.S --- orig/arch/arm/kernel/entry-common.S Wed Feb 27 14:24:47 2002 +++ linux/arch/arm/kernel/entry-common.S Fri Feb 21 15:36:40 2003 @@ -34,6 +34,7 @@ * stack. */ ret_fast_syscall: + disable_irq r1 @ ensure IRQs are disabled ldr r1, [tsk, #TSK_NEED_RESCHED] ldr r2, [tsk, #TSK_SIGPENDING] teq r1, #0 @ need_resched || sigpending @@ -45,28 +46,35 @@ * Ok, we need to do extra processing, enter the slow path. */ slow: str r0, [sp, #S_R0+S_OFF]! @ returned r0 - b 1f + teq r1, #0 + beq 1f /* * "slow" syscall return path. "why" tells us if this was a real syscall. */ reschedule: bl SYMBOL_NAME(schedule) +ret_disable_irq: + disable_irq r1 @ ensure IRQs are disabled ENTRY(ret_to_user) ret_slow_syscall: ldr r1, [tsk, #TSK_NEED_RESCHED] ldr r2, [tsk, #TSK_SIGPENDING] -1: teq r1, #0 @ need_resched => schedule() + teq r1, #0 @ need_resched => schedule() bne reschedule - teq r2, #0 @ sigpending => do_signal() - blne __do_signal +1: teq r2, #0 @ sigpending => do_signal() + bne __do_signal +restore: restore_user_regs __do_signal: + enable_irq r1 mov r0, #0 @ NULL 'oldset' mov r1, sp @ 'regs' mov r2, why @ 'syscall' - b SYMBOL_NAME(do_signal) @ note the bl above sets lr + bl SYMBOL_NAME(do_signal) @ note the bl above sets lr + disable_irq r1 @ ensure IRQs are disabled + b restore /* * This is how we return from a fork. __switch_to will be calling us @@ -79,11 +87,11 @@ ldr ip, [tsk, #TSK_PTRACE] @ check for syscall tracing mov why, #1 tst ip, #PT_TRACESYS @ are we tracing syscalls? - beq ret_slow_syscall + beq ret_disable_irq mov r1, sp mov r0, #1 @ trace exit [IP = 1] bl SYMBOL_NAME(syscall_trace) - b ret_slow_syscall + b ret_disable_irq #include "calls.S" @@ -130,7 +138,7 @@ ldr ip, [ip] mcr p15, 0, ip, c1, c0 @ update control register #endif - enable_irqs ip + enable_irq ip str r4, [sp, #-S_OFF]! @ push fifth arg @@ -174,7 +182,7 @@ mov r1, sp mov r0, #1 @ trace exit [IP = 1] bl SYMBOL_NAME(syscall_trace) - b ret_slow_syscall + b ret_disable_irq .align 5 #ifdef CONFIG_ALIGNMENT_TRAP diff -urN orig/arch/arm/kernel/entry-header.S linux/arch/arm/kernel/entry-header.S --- orig/arch/arm/kernel/entry-header.S Mon Aug 5 13:29:44 2002 +++ linux/arch/arm/kernel/entry-header.S Wed Feb 12 13:10:21 2003 @@ -41,14 +41,14 @@ @ @ Stack format (ensured by USER_* and SVC_*) @ -#define S_FRAME_SIZE 72 #ifdef CONFIG_CPU_32 +#define S_FRAME_SIZE 72 #define S_OLD_R0 68 #define S_PSR 64 #else +#define S_FRAME_SIZE 68 #define S_OLD_R0 64 #define S_PSR 60 -#define S_PC 60 #endif #define S_PC 60 @@ -70,6 +70,24 @@ #define S_OFF 8 #ifdef CONFIG_CPU_32 + .macro set_cpsr_c, reg, mode +#if 1 + /* broken binutils */ + mov \reg, \mode + msr cpsr_c, \reg +#else + msr cpsr_c, \mode +#endif + .endm + + .macro disable_irq, temp + set_cpsr_c \temp, #I_BIT | MODE_SVC + .endm + + .macro enable_irq, temp + set_cpsr_c \temp, #MODE_SVC + .endm + .macro save_user_regs sub sp, sp, #S_FRAME_SIZE stmia sp, {r0 - r12} @ Calling r0 - r12 @@ -81,21 +99,23 @@ str r0, [sp, #S_OLD_R0] @ Save OLD_R0 .endm +/* + * Must be called with IRQs already disabled. + */ .macro restore_user_regs - ldr r0, [sp, #S_PSR] @ Get calling cpsr - mov ip, #I_BIT | MODE_SVC - msr cpsr_c, ip @ disable IRQs - msr spsr, r0 @ save in spsr_svc - ldr lr, [sp, #S_PC] @ Get PC - ldmia sp, {r0 - lr}^ @ Get calling r0 - lr + ldr r1, [sp, #S_PSR] @ Get calling cpsr + ldr lr, [sp, #S_PC]! @ Get PC + msr spsr, r1 @ save in spsr_svc + ldmdb sp, {r0 - lr}^ @ Get calling r0 - lr mov r0, r0 - add sp, sp, #S_FRAME_SIZE + add sp, sp, #S_FRAME_SIZE - S_PC movs pc, lr @ return & move spsr_svc into cpsr .endm +/* + * Must be called with IRQs already disabled. + */ .macro fast_restore_user_regs - mov ip, #I_BIT | MODE_SVC - msr cpsr_c, ip @ disable IRQs ldr r1, [sp, #S_OFF + S_PSR] @ get calling cpsr ldr lr, [sp, #S_OFF + S_PC]! @ get pc msr spsr, r1 @ save in spsr_svc @@ -108,11 +128,6 @@ .macro mask_pc, rd, rm .endm - .macro enable_irqs, temp - mov \temp, #MODE_SVC - msr cpsr_c, \temp - .endm - .macro get_current_task, rd mov \rd, sp, lsr #13 mov \rd, \rd, lsl #13 @@ -136,9 +151,9 @@ #else .macro save_user_regs - str r0, [sp, #-4]! - str lr, [sp, #-4]! - sub sp, sp, #15*4 + sub sp, sp, #S_FRAME_SIZE + str r0, [sp, #S_OLD_R0] + str lr, [sp, #S_PC] stmia sp, {r0 - lr}^ mov r0, r0 .endm @@ -146,17 +161,16 @@ .macro restore_user_regs ldmia sp, {r0 - lr}^ mov r0, r0 - ldr lr, [sp, #15*4] - add sp, sp, #15*4+8 + ldr lr, [sp, #S_PC] + add sp, sp, #S_FRAME_SIZE movs pc, lr .endm .macro fast_restore_user_regs - add sp, sp, #S_OFF - ldmib sp, {r1 - lr}^ + add sp, sp, #S_OFF + S_PC + ldmdb sp, {r1 - lr}^ mov r0, r0 - ldr lr, [sp, #15*4] - add sp, sp, #15*4+8 + ldr lr, [sp], #S_FRAME_SIZE - S_PC movs pc, lr .endm @@ -164,7 +178,11 @@ bic \rd, \rm, #PCMASK .endm - .macro enable_irqs, temp + .macro disable_irq, temp + teqp pc, #0x08000003 + .endm + + .macro enable_irq, temp teqp pc, #0x00000003 .endm diff -urN orig/arch/arm/kernel/fiq.c linux/arch/arm/kernel/fiq.c --- orig/arch/arm/kernel/fiq.c Wed Jul 4 19:53:31 2001 +++ linux/arch/arm/kernel/fiq.c Thu Sep 5 20:51:08 2002 @@ -86,8 +86,8 @@ } static struct fiq_handler default_owner = { - name: "default", - fiq_op: fiq_def_op, + .name = "default", + .fiq_op = fiq_def_op, }; static struct fiq_handler *current_fiq = &default_owner; diff -urN orig/arch/arm/kernel/ftv-pci.c linux/arch/arm/kernel/ftv-pci.c --- orig/arch/arm/kernel/ftv-pci.c Tue Oct 3 20:08:18 2000 +++ linux/arch/arm/kernel/ftv-pci.c Sun Sep 15 12:55:57 2002 @@ -44,8 +44,8 @@ /* ftv host-specific stuff */ struct hw_pci ftv_pci __initdata = { - init: plx90x0_init, - swizzle: ftv_swizzle, - map_irq: ftv_map_irq, + .init = plx90x0_init, + .swizzle = ftv_swizzle, + .map_irq = ftv_map_irq, }; diff -urN orig/arch/arm/kernel/head-armv.S linux/arch/arm/kernel/head-armv.S --- orig/arch/arm/kernel/head-armv.S Fri Oct 26 16:45:50 2001 +++ linux/arch/arm/kernel/head-armv.S Wed Sep 11 11:56:28 2002 @@ -52,17 +52,22 @@ .endm /* - * Kernel startup entry point. + * Kernel startup entry point. + * --------------------------- * - * The rules are: - * r0 - should be 0 - * r1 - unique architecture number - * MMU - off - * I-cache - on or off - * D-cache - off + * This is normally called from the decompressor code. The requirements + * are: MMU = off, D-cache = off, I-cache = dont care, r0 = 0, + * r1 = machine nr. * - * See linux/arch/arm/tools/mach-types for the complete list of numbers - * for r1. + * This code is mostly position independent, so if you link the kernel at + * 0xc0008000, you call this at __pa(0xc0008000). + * + * See linux/arch/arm/tools/mach-types for the complete list of machine + * numbers for r1. + * + * We're trying to keep crap to a minimum; DO NOT add any machine specific + * crap here - that's what the boot loader (or in extreme, well justified + * circumstances, zImage) is for. */ .section ".text.init",#alloc,#execinstr .type stext, #function @@ -143,7 +148,6 @@ .type __switch_data, %object __switch_data: .long __mmap_switched - .long SYMBOL_NAME(compat) .long SYMBOL_NAME(__bss_start) .long SYMBOL_NAME(_end) .long SYMBOL_NAME(processor_id) @@ -151,28 +155,33 @@ .long SYMBOL_NAME(cr_alignment) .long SYMBOL_NAME(init_task_union)+8192 +/* + * Enable the MMU. This completely changes the structure of the visible + * memory space. You will not be able to trace execution through this. + * If you have an enquiry about this, *please* check the linux-arm-kernel + * mailing list archives BEFORE sending another post to the list. + */ .type __ret, %function __ret: ldr lr, __switch_data mcr p15, 0, r0, c1, c0 - mov r0, r0 + mrc p15, 0, r0, c1, c0, 0 @ read it back. mov r0, r0 mov r0, r0 mov pc, lr - /* - * This code follows on after the page - * table switch and jump above. - * - * r0 = processor control register - * r1 = machine ID - * r9 = processor ID - */ +/* + * The following fragment of code is executed with the MMU on, and uses + * absolute addresses; this is not position independent. + * + * r0 = processor control register + * r1 = machine ID + * r9 = processor ID + */ .align 5 __mmap_switched: adr r3, __switch_data + 4 - ldmia r3, {r2, r4, r5, r6, r7, r8, sp}@ r2 = compat + ldmia r3, {r4, r5, r6, r7, r8, sp}@ r2 = compat @ sp = stack pointer - str r12, [r2] mov fp, #0 @ Clear BSS (and zero fp) 1: cmp r4, r5 diff -urN orig/arch/arm/kernel/irq-arch.c linux/arch/arm/kernel/irq-arch.c --- orig/arch/arm/kernel/irq-arch.c Mon Sep 3 14:14:40 2001 +++ linux/arch/arm/kernel/irq-arch.c Thu Feb 6 11:31:26 2003 @@ -20,6 +20,8 @@ #include #include #include +#include +#include #include #include diff -urN orig/arch/arm/kernel/irq.c linux/arch/arm/kernel/irq.c --- orig/arch/arm/kernel/irq.c Sat Jul 21 10:46:32 2001 +++ linux/arch/arm/kernel/irq.c Wed Feb 12 17:12:18 2003 @@ -27,6 +27,8 @@ #include #include #include +#include +#include #include #include @@ -46,6 +48,7 @@ static volatile unsigned long irq_err_count; static spinlock_t irq_controller_lock; +static LIST_HEAD(irq_pending); struct irqdesc irq_desc[NR_IRQS]; void (*init_arch_irq)(void) __initdata = NULL; @@ -61,17 +64,21 @@ * disable_irq - disable an irq and wait for completion * @irq: Interrupt to disable * - * Disable the selected interrupt line. + * Disable the selected interrupt line. We do this lazily. * - * This function may be called - with care - from IRQ context. + * This function may be called from IRQ context. */ void disable_irq(unsigned int irq) { + struct irqdesc *desc = irq_desc + irq; unsigned long flags; spin_lock_irqsave(&irq_controller_lock, flags); - irq_desc[irq].enabled = 0; - irq_desc[irq].mask(irq); + if (!desc->disable_depth++) { +#ifndef CONFIG_CPU_SA1100 + desc->mask(irq); +#endif + } spin_unlock_irqrestore(&irq_controller_lock, flags); } @@ -79,19 +86,37 @@ * enable_irq - enable interrupt handling on an irq * @irq: Interrupt to enable * - * Re-enables the processing of interrupts on this IRQ line + * Re-enables the processing of interrupts on this IRQ line. + * Note that this may call the interrupt handler, so you may + * get unexpected results if you hold IRQs disabled. * * This function may be called from IRQ context. */ void enable_irq(unsigned int irq) { + struct irqdesc *desc = irq_desc + irq; unsigned long flags; spin_lock_irqsave(&irq_controller_lock, flags); - irq_desc[irq].probing = 0; - irq_desc[irq].triggered = 0; - irq_desc[irq].enabled = 1; - irq_desc[irq].unmask(irq); + if (!desc->disable_depth) { + printk("enable_irq(%u) unbalanced from %p\n", irq, + __builtin_return_address(0)); + } else if (!--desc->disable_depth) { + desc->probing = 0; + desc->unmask(irq); + + /* + * If the interrupt is waiting to be processed, + * try to re-run it. We can't directly run it + * from here since the caller might be in an + * interrupt-protected region. + */ + if (desc->pending) { + desc->pending = 0; + if (list_empty(&desc->pend)) + list_add(&desc->pend, &irq_pending); + } + } spin_unlock_irqrestore(&irq_controller_lock, flags); } @@ -128,7 +153,7 @@ * a large number if IRQs to appear in the same jiffie with the * same instruction pointer (or within 2 instructions). */ -static void check_irq_lock(struct irqdesc *desc, int irq, struct pt_regs *regs) +static int check_irq_lock(struct irqdesc *desc, int irq, struct pt_regs *regs) { unsigned long instr_ptr = instruction_pointer(regs); @@ -137,95 +162,198 @@ desc->lck_cnt += 1; if (desc->lck_cnt > MAX_IRQ_CNT) { - printk(KERN_ERR "IRQ LOCK: IRQ%d is locking the system, disabled\n", irq); - disable_irq(irq); + if (!desc->lck_warned++) + printk(KERN_ERR "IRQ LOCK: IRQ%d is locking the system, disabled\n", irq); + mod_timer(&desc->lck_timer, jiffies + 10*HZ); + return 1; } } else { desc->lck_cnt = 0; desc->lck_pc = instruction_pointer(regs); desc->lck_jif = jiffies; + if (desc->lck_warned < 0) + desc->lck_warned ++; } + return 0; +} + +static void +__do_irq(unsigned int irq, struct irqaction *action, struct pt_regs *regs) +{ + unsigned int status; + + spin_unlock(&irq_controller_lock); + + if (!(action->flags & SA_INTERRUPT)) + local_irq_enable(); + + status = 0; + do { + status |= action->flags; + action->handler(irq, action->dev_id, regs); + action = action->next; + } while (action); + + if (status & SA_SAMPLE_RANDOM) + add_interrupt_randomness(irq); + + spin_lock_irq(&irq_controller_lock); } /* * do_IRQ handles all normal device IRQ's */ -asmlinkage void do_IRQ(int irq, struct pt_regs * regs) +void do_IRQ(int irq, struct pt_regs * regs) { - struct irqdesc * desc; - struct irqaction * action; - int cpu; + struct irqdesc *desc = irq_desc + irq; - irq = fixup_irq(irq); + desc->triggered = 1; /* - * Some hardware gives randomly wrong interrupts. Rather - * than crashing, do something sensible. + * Acknowledge and clear the IRQ, but (if its + * a level-based IRQ, don't mask it) */ - if (irq >= NR_IRQS) - goto bad_irq; + desc->mask_ack(irq); - desc = irq_desc + irq; + /* + * If we're currently running this IRQ, or its disabled, + * we shouldn't process the IRQ. Instead, turn on the + * hardware masks. + */ + if (desc->running || desc->disable_depth) + goto running; - spin_lock(&irq_controller_lock); - desc->mask_ack(irq); - spin_unlock(&irq_controller_lock); + /* + * Mark the IRQ currently in progress. + */ + desc->running = 1; - cpu = smp_processor_id(); - irq_enter(cpu, irq); - kstat.irqs[cpu][irq]++; - desc->triggered = 1; + kstat.irqs[smp_processor_id()][irq]++; - /* Return with this interrupt masked if no action */ - action = desc->action; + do { + struct irqaction *action; - if (action) { - int status = 0; + action = desc->action; + if (!action) + break; - if (desc->nomask) { - spin_lock(&irq_controller_lock); + if (desc->pending && desc->disable_depth == 0) { + desc->pending = 0; desc->unmask(irq); - spin_unlock(&irq_controller_lock); } - if (!(action->flags & SA_INTERRUPT)) - __sti(); + __do_irq(irq, action, regs); + } while (desc->pending && desc->disable_depth == 0); - do { - status |= action->flags; - action->handler(irq, action->dev_id, regs); - action = action->next; - } while (action); + desc->running = 0; - if (status & SA_SAMPLE_RANDOM) - add_interrupt_randomness(irq); - __cli(); + /* + * If we are disabled or freed, shut down the handler. + */ + if (desc->action && !check_irq_lock(desc, irq, regs)) + desc->unmask(irq); + return; - if (!desc->nomask && desc->enabled) { - spin_lock(&irq_controller_lock); - desc->unmask(irq); - spin_unlock(&irq_controller_lock); + running: + /* + * We got another IRQ while this one was masked or + * currently running. Delay it. + */ + desc->pending = 1; +} + +static void do_pending_irqs(struct pt_regs *regs) +{ + struct list_head head, *l, *n; + + do { + struct irqdesc *desc; + + /* + * First, take the pending interrupts off the list. + * The act of calling the handlers may add some IRQs + * back onto the list. + */ + head = irq_pending; + INIT_LIST_HEAD(&irq_pending); + head.next->prev = &head; + head.prev->next = &head; + + /* + * Now run each entry. We must delete it from our + * list before calling the handler. + */ + list_for_each_safe(l, n, &head) { + desc = list_entry(l, struct irqdesc, pend); + list_del_init(&desc->pend); + do_IRQ(desc - irq_desc, regs); } - } + + /* + * The list must be empty. + */ + BUG_ON(!list_empty(&head)); + } while (!list_empty(&irq_pending)); +} + +/* + * do_IRQ handles all hardware IRQ's. Decoded IRQs should not + * come via this function. Instead, they should provide their + * own 'handler' + */ +asmlinkage void asm_do_IRQ(int irq, struct pt_regs *regs) +{ + irq = fixup_irq(irq); /* - * Debug measure - hopefully we can continue if an - * IRQ lockup problem occurs... + * Some hardware gives randomly wrong interrupts. Rather + * than crashing, do something sensible. */ - check_irq_lock(desc, irq, regs); + if (irq < NR_IRQS) { + int cpu = smp_processor_id(); - irq_exit(cpu, irq); + irq_enter(cpu, irq); + spin_lock(&irq_controller_lock); + do_IRQ(irq, regs); - if (softirq_pending(cpu)) - do_softirq(); - return; + /* + * Now re-run any pending interrupts. + */ + if (!list_empty(&irq_pending)) + do_pending_irqs(regs); + + spin_unlock(&irq_controller_lock); + irq_exit(cpu, irq); + + if (softirq_pending(cpu)) + do_softirq(); + return; + } -bad_irq: irq_err_count += 1; printk(KERN_ERR "IRQ: spurious interrupt %d\n", irq); return; } +static void irqlck_timeout(unsigned long _data) +{ + struct irqdesc *desc = (struct irqdesc *)_data; + + spin_lock(&irq_controller_lock); + + del_timer(&desc->lck_timer); + + desc->lck_cnt = 0; + desc->lck_pc = 0; + desc->lck_jif = 0; + desc->lck_warned = -10; + + if (desc->disable_depth == 0) + desc->unmask(desc - irq_desc); + + spin_unlock(&irq_controller_lock); +} + #ifdef CONFIG_ARCH_ACORN void do_ecard_IRQ(int irq, struct pt_regs *regs) { @@ -302,10 +430,12 @@ *p = new; if (!shared) { - desc->nomask = (new->flags & SA_IRQNOMASK) ? 1 : 0; desc->probing = 0; + desc->running = 0; + desc->pending = 0; + desc->disable_depth = 1; if (!desc->noautoenable) { - desc->enabled = 1; + desc->disable_depth = 0; desc->unmask(irq); } } @@ -383,7 +513,7 @@ * On a shared IRQ the caller must ensure the interrupt is disabled * on the card it drives before calling this function. * - * This function may be called from interrupt context. + * This function must not be called from interrupt context. */ void free_irq(unsigned int irq, void *dev_id) { @@ -416,6 +546,8 @@ spin_unlock_irqrestore(&irq_controller_lock, flags); } +static DECLARE_MUTEX(probe_sem); + /* Start the interrupt probing. Unlike other architectures, * we don't return a mask of interrupts from probe_irq_on, * but return the number of interrupts enabled for the probe. @@ -427,15 +559,15 @@ unsigned int i, irqs = 0; unsigned long delay; + down(&probe_sem); + /* * first snaffle up any unassigned but * probe-able interrupts */ spin_lock_irq(&irq_controller_lock); for (i = 0; i < NR_IRQS; i++) { - if (!irq_desc[i].valid || - !irq_desc[i].probe_ok || - irq_desc[i].action) + if (!irq_desc[i].probe_ok || irq_desc[i].action) continue; irq_desc[i].probing = 1; @@ -456,18 +588,31 @@ */ spin_lock_irq(&irq_controller_lock); for (i = 0; i < NR_IRQS; i++) { - if (irq_desc[i].probing && - irq_desc[i].triggered) { + if (irq_desc[i].probing && irq_desc[i].triggered) { irq_desc[i].probing = 0; irqs -= 1; } } spin_unlock_irq(&irq_controller_lock); - /* now filter out any obviously spurious interrupts */ return irqs; } +unsigned int probe_irq_mask(unsigned long irqs) +{ + unsigned int mask = 0, i; + + spin_lock_irq(&irq_controller_lock); + for (i = 0; i < 16 && i < NR_IRQS; i++) + if (irq_desc[i].probing && irq_desc[i].triggered) + mask |= 1 << i; + spin_unlock_irq(&irq_controller_lock); + + up(&probe_sem); + + return mask; +} + /* * Possible return values: * >= 0 - interrupt number @@ -499,6 +644,8 @@ out: spin_unlock_irq(&irq_controller_lock); + up(&probe_sem); + return irq_found; } @@ -518,6 +665,10 @@ irq_desc[irq].mask_ack = dummy_mask_unmask_irq; irq_desc[irq].mask = dummy_mask_unmask_irq; irq_desc[irq].unmask = dummy_mask_unmask_irq; + INIT_LIST_HEAD(&irq_desc[irq].pend); + init_timer(&irq_desc[irq].lck_timer); + irq_desc[irq].lck_timer.data = (unsigned long)&irq_desc[irq]; + irq_desc[irq].lck_timer.function = irqlck_timeout; } init_arch_irq(); diff -urN orig/arch/arm/kernel/oldlatches.c linux/arch/arm/kernel/oldlatches.c --- orig/arch/arm/kernel/oldlatches.c Mon Sep 3 14:14:40 2001 +++ linux/arch/arm/kernel/oldlatches.c Fri Sep 6 10:56:53 2002 @@ -29,10 +29,10 @@ if (machine_is_archimedes()) { unsigned long flags; - local_save_flags(flags); + local_irq_save(flags); latch_a_copy = (latch_a_copy & ~mask) | newdata; __raw_writeb(latch_a_copy, LATCHA_BASE); - local_restore_flags(flags); + local_irq_restore(flags); printk("Latch: A = 0x%02x\n", latch_a_copy); } else @@ -46,10 +46,10 @@ if (machine_is_archimedes()) { unsigned long flags; - local_save_flags(flags); + local_irq_save(flags); latch_b_copy = (latch_b_copy & ~mask) | newdata; __raw_writeb(latch_b_copy, LATCHB_BASE); - local_restore_flags(flags); + local_irq_restore(flags); printk("Latch: B = 0x%02x\n", latch_b_copy); } else diff -urN orig/arch/arm/kernel/process.c linux/arch/arm/kernel/process.c --- orig/arch/arm/kernel/process.c Mon Aug 5 13:29:44 2002 +++ linux/arch/arm/kernel/process.c Wed Mar 19 15:54:44 2003 @@ -24,18 +24,10 @@ #include #include -#include #include +#include #include -/* - * Values for cpu_do_idle() - */ -#define IDLE_WAIT_SLOW 0 -#define IDLE_WAIT_FAST 1 -#define IDLE_CLOCK_SLOW 2 -#define IDLE_CLOCK_FAST 3 - extern const char *processor_modes[]; extern void setup_mm_for_reboot(char mode); @@ -75,6 +67,18 @@ void (*pm_power_off)(void); /* + * This is our default idle handler. We need to disable + * interrupts here to ensure we don't miss a wakeup call. + */ +void default_idle(void) +{ + local_irq_disable(); + if (!current->need_resched && !hlt_counter) + arch_idle(); + local_irq_enable(); +} + +/* * The idle thread. We try to conserve power, while trying to keep * overall latency low. The architecture specific idle is passed * a value to indicate the level of "idleness" of the system. @@ -89,7 +93,7 @@ while (1) { void (*idle)(void) = pm_idle; if (!idle) - idle = arch_idle; + idle = default_idle; leds_event(led_idle_start); while (!current->need_resched) idle(); @@ -176,15 +180,15 @@ flags & CC_Z_BIT ? 'Z' : 'z', flags & CC_C_BIT ? 'C' : 'c', flags & CC_V_BIT ? 'V' : 'v'); - printk(" IRQs %s FIQs %s Mode %s%s Segment %s\n", - interrupts_enabled(regs) ? "on" : "off", - fast_interrupts_enabled(regs) ? "on" : "off", + printk(" IRQs o%s FIQs o%s Mode %s%s Segment %s\n", + interrupts_enabled(regs) ? "n" : "ff", + fast_interrupts_enabled(regs) ? "n" : "ff", processor_modes[processor_mode(regs)], thumb_mode(regs) ? " (T)" : "", get_fs() == get_ds() ? "kernel" : "user"); #if defined(CONFIG_CPU_32) { - int ctrl, transbase, dac; + unsigned int ctrl, transbase, dac; __asm__ ( " mrc p15, 0, %0, c1, c0\n" " mrc p15, 0, %1, c2, c0\n" @@ -282,12 +286,22 @@ { } +static void default_fp_init(union fp_state *fp) +{ + memset(fp, 0, sizeof(union fp_state)); +} + +void (*fp_init)(union fp_state *) = default_fp_init; + void flush_thread(void) { - memset(¤t->thread.debug, 0, sizeof(struct debug_info)); - memset(¤t->thread.fpstate, 0, sizeof(union fp_state)); - current->used_math = 0; - current->flags &= ~PF_USEDFPU; + struct task_struct *tsk = current; + + tsk->flags &= ~PF_USEDFPU; + tsk->used_math = 0; + + memset(&tsk->thread.debug, 0, sizeof(struct debug_info)); + fp_init(&tsk->thread.fpstate); } void release_thread(struct task_struct *dead_task) @@ -296,16 +310,16 @@ asmlinkage void ret_from_fork(void) __asm__("ret_from_fork"); -int copy_thread(int nr, unsigned long clone_flags, unsigned long esp, - unsigned long unused, - struct task_struct * p, struct pt_regs * regs) +int +copy_thread(int nr, unsigned long clone_flags, unsigned long esp, + unsigned long unused, struct task_struct * p, struct pt_regs * regs) { - struct pt_regs * childregs; + struct pt_regs *childregs; struct context_save_struct * save; atomic_set(&p->thread.refcount, 1); - childregs = ((struct pt_regs *)((unsigned long)p + 8192)) - 1; + childregs = ((struct pt_regs *)((unsigned long)p + 8192 - 8)) - 1; *childregs = *regs; childregs->ARM_r0 = 0; childregs->ARM_sp = esp; @@ -324,10 +338,12 @@ */ int dump_fpu (struct pt_regs *regs, struct user_fp *fp) { - if (current->used_math) + int used_math = current->used_math; + + if (used_math) memcpy(fp, ¤t->thread.fpstate.soft, sizeof (*fp)); - return current->used_math; + return used_math; } /* @@ -347,8 +363,8 @@ dump->u_debugreg[0] = tsk->thread.debug.bp[0].address; dump->u_debugreg[1] = tsk->thread.debug.bp[1].address; - dump->u_debugreg[2] = tsk->thread.debug.bp[0].insn; - dump->u_debugreg[3] = tsk->thread.debug.bp[1].insn; + dump->u_debugreg[2] = tsk->thread.debug.bp[0].insn.arm; + dump->u_debugreg[3] = tsk->thread.debug.bp[1].insn.arm; dump->u_debugreg[4] = tsk->thread.debug.nsaved; if (dump->start_stack < 0x04000000) @@ -366,23 +382,23 @@ * a system call from a "real" process, but the process memory space will * not be free'd until both the parent and the child have exited. */ -pid_t kernel_thread(int (*fn)(void *), void *arg, unsigned long flags) +pid_t arch_kernel_thread(int (*fn)(void *), void *arg, unsigned long flags) { pid_t __ret; __asm__ __volatile__( - "orr r0, %1, %2 @ kernel_thread sys_clone - mov r1, #0 - "__syscall(clone)" - movs %0, r0 @ if we are the child - bne 1f - mov fp, #0 @ ensure that fp is zero - mov r0, %4 - mov lr, pc - mov pc, %3 - b sys_exit + "orr r0, %1, %2 @ kernel_thread sys_clone \n\ + mov r1, #0 \n\ + "__syscall(clone)" \n\ + movs %0, r0 @ if we are the child \n\ + bne 1f \n\ + mov fp, #0 @ ensure that fp is zero \n\ + mov r0, %4 \n\ + mov lr, pc \n\ + mov pc, %3 \n\ + b sys_exit \n\ 1: " - : "=r" (__ret) + : "=&r" (__ret) : "Ir" (flags), "I" (CLONE_VM), "r" (fn), "r" (arg) : "r0", "r1", "lr"); return __ret; @@ -405,7 +421,7 @@ return 0; stack_page = 4096 + (unsigned long)p; - fp = get_css_fp(&p->thread); + fp = thread_saved_fp(&p->thread); do { if (fp < stack_page || fp > 4092+stack_page) return 0; diff -urN orig/arch/arm/kernel/ptrace.c linux/arch/arm/kernel/ptrace.c --- orig/arch/arm/kernel/ptrace.c Mon Oct 1 23:10:02 2001 +++ linux/arch/arm/kernel/ptrace.c Sun Sep 15 11:53:45 2002 @@ -9,6 +9,7 @@ * it under the terms of the GNU General Public License version 2 as * published by the Free Software Foundation. */ +#include #include #include #include @@ -16,6 +17,7 @@ #include #include #include +#include #include #include @@ -33,18 +35,24 @@ /* * Breakpoint SWI instruction: SWI &9F0001 */ -#define BREAKINST 0xef9f0001 +#define BREAKINST_ARM 0xef9f0001 +/* fill this in later */ +#define BREAKINST_THUMB 0xdf00 /* * Get the address of the live pt_regs for the specified task. * These are saved onto the top kernel stack when the process * is not running. + * + * Note: if a user thread is execve'd from kernel space, the + * kernel stack will not be empty on entry to the kernel, so + * ptracing these tasks will fail. */ static inline struct pt_regs * get_user_regs(struct task_struct *task) { return (struct pt_regs *) - ((unsigned long)task + 8192 - sizeof(struct pt_regs)); + ((unsigned long)task + 8192 - 8 - sizeof(struct pt_regs)); } /* @@ -53,7 +61,7 @@ * this routine assumes that all the privileged stacks are in our * data space. */ -static inline long get_stack_long(struct task_struct *task, int offset) +static inline long get_user_reg(struct task_struct *task, int offset) { return get_user_regs(task)->uregs[offset]; } @@ -65,14 +73,14 @@ * data space. */ static inline int -put_stack_long(struct task_struct *task, int offset, long data) +put_user_reg(struct task_struct *task, int offset, long data) { struct pt_regs newregs, *regs = get_user_regs(task); int ret = -EINVAL; newregs = *regs; newregs.uregs[offset] = data; - + if (valid_user_regs(&newregs)) { regs->uregs[offset] = data; ret = 0; @@ -82,23 +90,32 @@ } static inline int -read_tsk_long(struct task_struct *child, unsigned long addr, unsigned long *res) +read_u32(struct task_struct *task, unsigned long addr, u32 *res) { - int copied; + int ret; - copied = access_process_vm(child, addr, res, sizeof(*res), 0); + ret = access_process_vm(task, addr, res, sizeof(*res), 0); - return copied != sizeof(*res) ? -EIO : 0; + return ret == sizeof(*res) ? 0 : -EIO; } static inline int -write_tsk_long(struct task_struct *child, unsigned long addr, unsigned long val) +read_instr(struct task_struct *task, unsigned long addr, u32 *res) { - int copied; - - copied = access_process_vm(child, addr, &val, sizeof(val), 1); + int ret; - return copied != sizeof(val) ? -EIO : 0; + if (addr & 1) { + u16 val; + ret = access_process_vm(task, addr & ~1, &val, sizeof(val), 0); + ret = ret == sizeof(val) ? 0 : -EIO; + *res = val; + } else { + u32 val; + ret = access_process_vm(task, addr & ~3, &val, sizeof(val), 0); + ret = ret == sizeof(val) ? 0 : -EIO; + *res = val; + } + return ret; } /* @@ -110,7 +127,7 @@ unsigned int reg = (insn >> 16) & 15; unsigned long val; - val = get_stack_long(child, reg); + val = get_user_reg(child, reg); if (reg == 15) val = pc_pointer(val + 8); @@ -132,10 +149,10 @@ shift = (insn >> 8) & 15; type = 3; } else { - val = get_stack_long (child, insn & 15); + val = get_user_reg (child, insn & 15); if (insn & (1 << 4)) - shift = (int)get_stack_long (child, (insn >> 8) & 15); + shift = (int)get_user_reg (child, (insn >> 8) & 15); else shift = (insn >> 7) & 31; @@ -165,7 +182,7 @@ int shift; int type; - val = get_stack_long(child, insn & 15); + val = get_user_reg(child, insn & 15); shift = (insn >> 7) & 31; type = (insn >> 5) & 3; @@ -182,10 +199,24 @@ return val; } +#define OP_MASK 0x01e00000 +#define OP_AND 0x00000000 +#define OP_EOR 0x00200000 +#define OP_SUB 0x00400000 +#define OP_RSB 0x00600000 +#define OP_ADD 0x00800000 +#define OP_ADC 0x00a00000 +#define OP_SBC 0x00c00000 +#define OP_RSC 0x00e00000 +#define OP_ORR 0x01800000 +#define OP_MOV 0x01a00000 +#define OP_BIC 0x01c00000 +#define OP_MVN 0x01e00000 + static unsigned long get_branch_address(struct task_struct *child, unsigned long pc, unsigned long insn) { - unsigned long alt = 0; + u32 alt = 0; switch (insn & 0x0e000000) { case 0x00000000: @@ -200,21 +231,21 @@ aluop1 = ptrace_getrn(child, insn); aluop2 = ptrace_getaluop2(child, insn); - ccbit = get_stack_long(child, REG_PSR) & CC_C_BIT ? 1 : 0; + ccbit = get_user_reg(child, REG_PSR) & CC_C_BIT ? 1 : 0; - switch (insn & 0x01e00000) { - case 0x00000000: alt = aluop1 & aluop2; break; - case 0x00200000: alt = aluop1 ^ aluop2; break; - case 0x00400000: alt = aluop1 - aluop2; break; - case 0x00600000: alt = aluop2 - aluop1; break; - case 0x00800000: alt = aluop1 + aluop2; break; - case 0x00a00000: alt = aluop1 + aluop2 + ccbit; break; - case 0x00c00000: alt = aluop1 - aluop2 + ccbit; break; - case 0x00e00000: alt = aluop2 - aluop1 + ccbit; break; - case 0x01800000: alt = aluop1 | aluop2; break; - case 0x01a00000: alt = aluop2; break; - case 0x01c00000: alt = aluop1 & ~aluop2; break; - case 0x01e00000: alt = ~aluop2; break; + switch (insn & OP_MASK) { + case OP_AND: alt = aluop1 & aluop2; break; + case OP_EOR: alt = aluop1 ^ aluop2; break; + case OP_SUB: alt = aluop1 - aluop2; break; + case OP_RSB: alt = aluop2 - aluop1; break; + case OP_ADD: alt = aluop1 + aluop2; break; + case OP_ADC: alt = aluop1 + aluop2 + ccbit; break; + case OP_SBC: alt = aluop1 - aluop2 + ccbit; break; + case OP_RSC: alt = aluop2 - aluop1 + ccbit; break; + case OP_ORR: alt = aluop1 | aluop2; break; + case OP_MOV: alt = aluop2; break; + case OP_BIC: alt = aluop1 & ~aluop2; break; + case OP_MVN: alt = ~aluop2; break; } break; } @@ -241,7 +272,7 @@ else base -= aluop2; } - if (read_tsk_long(child, base, &alt) == 0) + if (read_u32(child, base, &alt) == 0) alt = pc_pointer(alt); } break; @@ -255,13 +286,7 @@ unsigned int nr_regs; if (insn & (1 << 23)) { - nr_regs = insn & 65535; - - nr_regs = (nr_regs & 0x5555) + ((nr_regs & 0xaaaa) >> 1); - nr_regs = (nr_regs & 0x3333) + ((nr_regs & 0xcccc) >> 2); - nr_regs = (nr_regs & 0x0707) + ((nr_regs & 0x7070) >> 4); - nr_regs = (nr_regs & 0x000f) + ((nr_regs & 0x0f00) >> 8); - nr_regs <<= 2; + nr_regs = hweight16(insn & 65535) << 2; if (!(insn & (1 << 24))) nr_regs -= 4; @@ -274,8 +299,8 @@ base = ptrace_getrn(child, insn); - if (read_tsk_long(child, base + nr_regs, &alt) == 0) - alt = pc_pointer (alt); + if (read_u32(child, base + nr_regs, &alt) == 0) + alt = pc_pointer(alt); break; } break; @@ -304,36 +329,82 @@ } static int -add_breakpoint(struct task_struct *child, struct debug_info *dbg, unsigned long addr) +swap_insn(struct task_struct *task, unsigned long addr, + void *old_insn, void *new_insn, int size) +{ + int ret; + + ret = access_process_vm(task, addr, old_insn, size, 0); + if (ret == size) + ret = access_process_vm(task, addr, new_insn, size, 1); + return ret; +} + +static void +add_breakpoint(struct task_struct *task, struct debug_info *dbg, unsigned long addr) { int nr = dbg->nsaved; - int res = -EINVAL; if (nr < 2) { - res = read_tsk_long(child, addr, &dbg->bp[nr].insn); - if (res == 0) - res = write_tsk_long(child, addr, BREAKINST); + u32 new_insn = BREAKINST_ARM; + int res; + + res = swap_insn(task, addr, &dbg->bp[nr].insn, &new_insn, 4); - if (res == 0) { + if (res == 4) { dbg->bp[nr].address = addr; dbg->nsaved += 1; } } else printk(KERN_ERR "ptrace: too many breakpoints\n"); +} + +/* + * Clear one breakpoint in the user program. We copy what the hardware + * does and use bit 0 of the address to indicate whether this is a Thumb + * breakpoint or an ARM breakpoint. + */ +static void clear_breakpoint(struct task_struct *task, struct debug_entry *bp) +{ + unsigned long addr = bp->address; + union debug_insn old_insn; + int ret; - return res; + if (addr & 1) { + ret = swap_insn(task, addr & ~1, &old_insn.thumb, + &bp->insn.thumb, 2); + + if (ret != 2 || old_insn.thumb != BREAKINST_THUMB) + printk(KERN_ERR "%s:%d: corrupted Thumb breakpoint at " + "0x%08lx (0x%04x)\n", task->comm, task->pid, + addr, old_insn.thumb); + } else { + ret = swap_insn(task, addr & ~3, &old_insn.thumb, + &bp->insn.thumb, 4); + + if (ret != 4 || old_insn.arm != BREAKINST_ARM) + printk(KERN_ERR "%s:%d: corrupted ARM breakpoint at " + "0x%08lx (0x%08x)\n", task->comm, task->pid, + addr, old_insn.arm); + } } -int ptrace_set_bpt(struct task_struct *child) +void ptrace_set_bpt(struct task_struct *child) { struct pt_regs *regs; - unsigned long pc, insn; + unsigned long pc; + u32 insn; int res; regs = get_user_regs(child); pc = instruction_pointer(regs); - res = read_tsk_long(child, pc, &insn); + if (thumb_mode(regs)) { + printk(KERN_WARNING "ptrace: can't handle thumb mode\n"); + return; + } + + res = read_instr(child, pc, &insn); if (!res) { struct debug_info *dbg = &child->thread.debug; unsigned long alt; @@ -342,7 +413,7 @@ alt = get_branch_address(child, pc, insn); if (alt) - res = add_breakpoint(child, dbg, alt); + add_breakpoint(child, dbg, alt); /* * Note that we ignore the result of setting the above @@ -354,10 +425,8 @@ * loose control of the thread during single stepping. */ if (!alt || predicate(insn) != PREDICATE_ALWAYS) - res = add_breakpoint(child, dbg, pc + 4); + add_breakpoint(child, dbg, pc + 4); } - - return res; } /* @@ -366,24 +435,17 @@ */ void __ptrace_cancel_bpt(struct task_struct *child) { - struct debug_info *dbg = &child->thread.debug; - int i, nsaved = dbg->nsaved; + int i, nsaved = child->thread.debug.nsaved; - dbg->nsaved = 0; + child->thread.debug.nsaved = 0; if (nsaved > 2) { printk("ptrace_cancel_bpt: bogus nsaved: %d!\n", nsaved); nsaved = 2; } - for (i = 0; i < nsaved; i++) { - unsigned long tmp; - - read_tsk_long(child, dbg->bp[i].address, &tmp); - write_tsk_long(child, dbg->bp[i].address, dbg->bp[i].insn); - if (tmp != BREAKINST) - printk(KERN_ERR "ptrace_cancel_bpt: weirdness\n"); - } + for (i = 0; i < nsaved; i++) + clear_breakpoint(child, &child->thread.debug.bp[i]); } /* @@ -396,6 +458,117 @@ __ptrace_cancel_bpt(child); } +/* + * Handle hitting a breakpoint. regs points at the instruction. + */ +void ptrace_break(struct task_struct *tsk, struct pt_regs *regs) +{ + siginfo_t info; + + /* + * The PC is pointing at the next instruction. Fix this. + */ + regs->ARM_pc -= thumb_mode(regs) ? 2 : 4; + + if (tsk->thread.debug.nsaved == 0) + printk(KERN_ERR "ptrace: bogus breakpoint trap\n"); + + __ptrace_cancel_bpt(tsk); + + info.si_signo = SIGTRAP; + info.si_errno = 0; + info.si_code = TRAP_BRKPT; + info.si_addr = (void *)instruction_pointer(regs); + + force_sig_info(SIGTRAP, &info, tsk); +} + +/* + * Read the word at offset "off" into the "struct user". We + * actually access the pt_regs stored on the kernel stack. + */ +static int ptrace_read_user(struct task_struct *tsk, unsigned long off, + unsigned long *ret) +{ + unsigned long tmp; + + if (off & 3 || off >= sizeof(struct user)) + return -EIO; + + tmp = 0; + if (off < sizeof(struct pt_regs)) + tmp = get_user_reg(tsk, off >> 2); + + return put_user(tmp, ret); +} + +/* + * Write the word at offset "off" into "struct user". We + * actually access the pt_regs stored on the kernel stack. + */ +static int ptrace_write_user(struct task_struct *tsk, unsigned long off, + unsigned long val) +{ + if (off & 3 || off >= sizeof(struct user)) + return -EIO; + + if (off >= sizeof(struct pt_regs)) + return 0; + + return put_user_reg(tsk, off >> 2, val); +} + +/* + * Get all user integer registers. + */ +static int ptrace_getregs(struct task_struct *tsk, void *uregs) +{ + struct pt_regs *regs = get_user_regs(tsk); + + return copy_to_user(uregs, regs, sizeof(struct pt_regs)) ? -EFAULT : 0; +} + +/* + * Set all user integer registers. + */ +static int ptrace_setregs(struct task_struct *tsk, void *uregs) +{ + struct pt_regs newregs; + int ret; + + ret = -EFAULT; + if (copy_from_user(&newregs, uregs, sizeof(struct pt_regs)) == 0) { + struct pt_regs *regs = get_user_regs(tsk); + + ret = -EINVAL; + if (valid_user_regs(&newregs)) { + *regs = newregs; + ret = 0; + } + } + + return ret; +} + +/* + * Get the child FPU state. + */ +static int ptrace_getfpregs(struct task_struct *tsk, void *ufp) +{ + return copy_to_user(ufp, &tsk->thread.fpstate, + sizeof(struct user_fp)) ? -EFAULT : 0; +} + +/* + * Set the child FPU state. + */ +static int ptrace_setfpregs(struct task_struct *tsk, void *ufp) +{ + tsk->used_math = 1; + return copy_from_user(&tsk->thread.fpstate, ufp, + sizeof(struct user_fp)) ? -EFAULT : 0; +} + static int do_ptrace(int request, struct task_struct *child, long addr, long data) { unsigned long tmp; @@ -407,23 +580,16 @@ */ case PTRACE_PEEKTEXT: case PTRACE_PEEKDATA: - ret = read_tsk_long(child, addr, &tmp); - if (!ret) + ret = access_process_vm(child, addr, &tmp, + sizeof(unsigned long), 0); + if (ret == sizeof(unsigned long)) ret = put_user(tmp, (unsigned long *) data); + else + ret = -EIO; break; - /* - * read the word at location "addr" in the user registers. - */ case PTRACE_PEEKUSR: - ret = -EIO; - if ((addr & 3) || addr < 0 || addr >= sizeof(struct user)) - break; - - tmp = 0; /* Default return condition */ - if (addr < sizeof(struct pt_regs)) - tmp = get_stack_long(child, (int)addr >> 2); - ret = put_user(tmp, (unsigned long *)data); + ret = ptrace_read_user(child, addr, (unsigned long *)data); break; /* @@ -431,19 +597,16 @@ */ case PTRACE_POKETEXT: case PTRACE_POKEDATA: - ret = write_tsk_long(child, addr, data); + ret = access_process_vm(child, addr, &data, + sizeof(unsigned long), 1); + if (ret == sizeof(unsigned long)) + ret = 0; + else + ret = -EIO; break; - /* - * write the word at location addr in the user registers. - */ case PTRACE_POKEUSR: - ret = -EIO; - if ((addr & 3) || addr < 0 || addr >= sizeof(struct user)) - break; - - if (addr < sizeof(struct pt_regs)) - ret = put_stack_long(child, (int)addr >> 2, data); + ret = ptrace_write_user(child, addr, data); break; /* @@ -471,14 +634,12 @@ * exit. */ case PTRACE_KILL: - /* already dead */ - ret = 0; - if (child->state == TASK_ZOMBIE) - break; - child->exit_code = SIGKILL; /* make sure single-step breakpoint is gone. */ __ptrace_cancel_bpt(child); - wake_up_process(child); + if (child->state != TASK_ZOMBIE) { + child->exit_code = SIGKILL; + wake_up_process(child); + } ret = 0; break; @@ -497,71 +658,32 @@ ret = 0; break; - /* - * detach a process that was attached. - */ case PTRACE_DETACH: ret = ptrace_detach(child, data); break; - /* - * Get all gp regs from the child. - */ - case PTRACE_GETREGS: { - struct pt_regs *regs = get_user_regs(child); - - ret = 0; - if (copy_to_user((void *)data, regs, - sizeof(struct pt_regs))) - ret = -EFAULT; - + case PTRACE_GETREGS: + ret = ptrace_getregs(child, (void *)data); break; - } - /* - * Set all gp regs in the child. - */ - case PTRACE_SETREGS: { - struct pt_regs newregs; - - ret = -EFAULT; - if (copy_from_user(&newregs, (void *)data, - sizeof(struct pt_regs)) == 0) { - struct pt_regs *regs = get_user_regs(child); - - ret = -EINVAL; - if (valid_user_regs(&newregs)) { - *regs = newregs; - ret = 0; - } - } + case PTRACE_SETREGS: + ret = ptrace_setregs(child, (void *)data); break; - } - /* - * Get the child FPU state. - */ case PTRACE_GETFPREGS: - ret = -EIO; - if (!access_ok(VERIFY_WRITE, (void *)data, sizeof(struct user_fp))) - break; - - /* we should check child->used_math here */ - ret = __copy_to_user((void *)data, &child->thread.fpstate, - sizeof(struct user_fp)) ? -EFAULT : 0; + ret = ptrace_getfpregs(child, (void *)data); break; - /* - * Set the child FPU state. - */ case PTRACE_SETFPREGS: - ret = -EIO; - if (!access_ok(VERIFY_READ, (void *)data, sizeof(struct user_fp))) - break; + ret = ptrace_setfpregs(child, (void *)data); + break; - child->used_math = 1; - ret = __copy_from_user(&child->thread.fpstate, (void *)data, - sizeof(struct user_fp)) ? -EFAULT : 0; + case PTRACE_SETOPTIONS: + if (data & PTRACE_O_TRACESYSGOOD) + child->ptrace |= PT_TRACESYSGOOD; + else + child->ptrace &= ~PT_TRACESYSGOOD; + ret = 0; break; default: diff -urN orig/arch/arm/kernel/ptrace.h linux/arch/arm/kernel/ptrace.h --- orig/arch/arm/kernel/ptrace.h Tue Oct 3 20:08:18 2000 +++ linux/arch/arm/kernel/ptrace.h Fri Sep 6 20:10:01 2002 @@ -1,14 +1,15 @@ /* * linux/arch/arm/kernel/ptrace.h * - * Copyright (C) 2000 Russell King + * Copyright (C) 2000-2002 Russell King * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as * published by the Free Software Foundation. */ extern void __ptrace_cancel_bpt(struct task_struct *); -extern int ptrace_set_bpt(struct task_struct *); +extern void ptrace_set_bpt(struct task_struct *); +extern void ptrace_break(struct task_struct *, struct pt_regs *); /* * Clear a breakpoint, if one exists. diff -urN orig/arch/arm/kernel/semaphore.c linux/arch/arm/kernel/semaphore.c --- orig/arch/arm/kernel/semaphore.c Mon Aug 5 13:29:44 2002 +++ linux/arch/arm/kernel/semaphore.c Sun Sep 15 13:13:12 2002 @@ -177,76 +177,76 @@ * value in some cases.. */ #ifdef CONFIG_CPU_26 -asm(" .align 5 - .globl __down_failed -__down_failed: - stmfd sp!, {r0 - r3, lr} - mov r0, ip - bl __down - ldmfd sp!, {r0 - r3, pc}^ - - .align 5 - .globl __down_interruptible_failed -__down_interruptible_failed: - stmfd sp!, {r0 - r3, lr} - mov r0, ip - bl __down_interruptible - mov ip, r0 - ldmfd sp!, {r0 - r3, pc}^ - - .align 5 - .globl __down_trylock_failed -__down_trylock_failed: - stmfd sp!, {r0 - r3, lr} - mov r0, ip - bl __down_trylock - mov ip, r0 - ldmfd sp!, {r0 - r3, pc}^ - - .align 5 - .globl __up_wakeup -__up_wakeup: - stmfd sp!, {r0 - r3, lr} - mov r0, ip - bl __up - ldmfd sp!, {r0 - r3, pc}^ +asm(" .align 5 \n\ + .globl __down_failed \n\ +__down_failed: \n\ + stmfd sp!, {r0 - r3, lr} \n\ + mov r0, ip \n\ + bl __down \n\ + ldmfd sp!, {r0 - r3, pc}^ \n\ + \n\ + .align 5 \n\ + .globl __down_interruptible_failed \n\ +__down_interruptible_failed: \n\ + stmfd sp!, {r0 - r3, lr} \n\ + mov r0, ip \n\ + bl __down_interruptible \n\ + mov ip, r0 \n\ + ldmfd sp!, {r0 - r3, pc}^ \n\ + + .align 5 \n\ + .globl __down_trylock_failed \n\ +__down_trylock_failed: \n\ + stmfd sp!, {r0 - r3, lr} \n\ + mov r0, ip \n\ + bl __down_trylock \n\ + mov ip, r0 \n\ + ldmfd sp!, {r0 - r3, pc}^ \n\ + \n\ + .align 5 \n\ + .globl __up_wakeup \n\ +__up_wakeup: \n\ + stmfd sp!, {r0 - r3, lr} \n\ + mov r0, ip \n\ + bl __up \n\ + ldmfd sp!, {r0 - r3, pc}^ \n\ "); #else /* 32 bit version */ -asm(" .align 5 - .globl __down_failed -__down_failed: - stmfd sp!, {r0 - r3, lr} - mov r0, ip - bl __down - ldmfd sp!, {r0 - r3, pc} - - .align 5 - .globl __down_interruptible_failed -__down_interruptible_failed: - stmfd sp!, {r0 - r3, lr} - mov r0, ip - bl __down_interruptible - mov ip, r0 - ldmfd sp!, {r0 - r3, pc} - - .align 5 - .globl __down_trylock_failed -__down_trylock_failed: - stmfd sp!, {r0 - r3, lr} - mov r0, ip - bl __down_trylock - mov ip, r0 - ldmfd sp!, {r0 - r3, pc} - - .align 5 - .globl __up_wakeup -__up_wakeup: - stmfd sp!, {r0 - r3, lr} - mov r0, ip - bl __up - ldmfd sp!, {r0 - r3, pc} +asm(" .align 5 \n\ + .globl __down_failed \n\ +__down_failed: \n\ + stmfd sp!, {r0 - r3, lr} \n\ + mov r0, ip \n\ + bl __down \n\ + ldmfd sp!, {r0 - r3, pc} \n\ + \n\ + .align 5 \n\ + .globl __down_interruptible_failed \n\ +__down_interruptible_failed: \n\ + stmfd sp!, {r0 - r3, lr} \n\ + mov r0, ip \n\ + bl __down_interruptible \n\ + mov ip, r0 \n\ + ldmfd sp!, {r0 - r3, pc} \n\ + \n\ + .align 5 \n\ + .globl __down_trylock_failed \n\ +__down_trylock_failed: \n\ + stmfd sp!, {r0 - r3, lr} \n\ + mov r0, ip \n\ + bl __down_trylock \n\ + mov ip, r0 \n\ + ldmfd sp!, {r0 - r3, pc} \n\ + \n\ + .align 5 \n\ + .globl __up_wakeup \n\ +__up_wakeup: \n\ + stmfd sp!, {r0 - r3, lr} \n\ + mov r0, ip \n\ + bl __up \n\ + ldmfd sp!, {r0 - r3, pc} \n\ "); #endif diff -urN orig/arch/arm/kernel/setup.c linux/arch/arm/kernel/setup.c --- orig/arch/arm/kernel/setup.c Fri Nov 23 10:12:06 2001 +++ linux/arch/arm/kernel/setup.c Thu Feb 6 11:31:09 2003 @@ -17,6 +17,8 @@ #include #include #include +#include +#include #include #include @@ -33,10 +35,6 @@ #define MEM_SIZE (16*1024*1024) #endif -#ifndef CONFIG_CMDLINE -#define CONFIG_CMDLINE "" -#endif - #if defined(CONFIG_FPE_NWFPE) || defined(CONFIG_FPE_FASTFPE) char fpe_type[8]; @@ -51,14 +49,14 @@ extern unsigned int mem_fclk_21285; extern void paging_init(struct meminfo *, struct machine_desc *desc); -extern void convert_to_tag_list(struct param_struct *params, int mem_init); +extern void convert_to_tag_list(struct tag *tags); +extern void squash_mem_tags(struct tag *tag); extern void bootmem_init(struct meminfo *); extern void reboot_setup(char *str); extern int root_mountflags; extern int _stext, _text, _etext, _edata, _end; unsigned int processor_id; -unsigned int compat; unsigned int __machine_arch_type; unsigned int system_rev; unsigned int system_serial_low; @@ -72,9 +70,11 @@ unsigned char aux_device_present; char elf_platform[ELF_PLATFORM_SIZE]; char saved_command_line[COMMAND_LINE_SIZE]; +unsigned long phys_initrd_start __initdata = 0; +unsigned long phys_initrd_size __initdata = 0; static struct meminfo meminfo __initdata = { 0, }; -static struct proc_info_item proc_info; +static const char *cpu_name; static const char *machine_name; static char command_line[COMMAND_LINE_SIZE]; @@ -129,15 +129,13 @@ while (1); } - proc_info = *list->info; + cpu_name = list->info->cpu_name; #ifdef MULTI_CPU processor = *list->proc; #endif - printk("Processor: %s %s revision %d\n", - proc_info.manufacturer, proc_info.cpu_name, - (int)processor_id & 15); + printk("CPU: %s revision %d\n", cpu_name, (int)processor_id & 15); sprintf(system_utsname.machine, "%s%c", list->arch_name, ENDIANNESS); sprintf(elf_platform, "%s%c", list->elf_name, ENDIANNESS); @@ -146,7 +144,7 @@ cpu_proc_init(); } -static struct machine_desc * __init setup_architecture(unsigned int nr) +static struct machine_desc * __init setup_machine(unsigned int nr) { extern struct machine_desc __arch_info_begin, __arch_info_end; struct machine_desc *list; @@ -168,12 +166,7 @@ while (1); } - printk("Architecture: %s\n", list->name); - if (compat) - printk(KERN_WARNING "Using compatibility code " - "scheduled for removal in v%d.%d.%d\n", - compat >> 24, (compat >> 12) & 0x3ff, - compat & 0x3ff); + printk("Machine: %s\n", list->name); return list; } @@ -215,6 +208,22 @@ mi->bank[mi->nr_banks].size = size; mi->bank[mi->nr_banks].node = PHYS_TO_NID(start); mi->nr_banks += 1; + } else if (c == ' ' && !memcmp(from, "initrd=", 7)) { + unsigned long start, size; + + /* + * Remove space character + */ + if (to != command_line) + to -= 1; + + start = memparse(from + 7, &from); + if (*from == ',') { + size = memparse(from + 1, &from); + + phys_initrd_start = start; + phys_initrd_size = size; + } } c = *from++; if (!c) @@ -231,7 +240,7 @@ setup_ramdisk(int doload, int prompt, int image_start, unsigned int rd_sz) { #ifdef CONFIG_BLK_DEV_RAM - extern int rd_size; + extern int rd_size, rd_image_start, rd_prompt, rd_doload; rd_image_start = image_start; rd_prompt = prompt; @@ -250,8 +259,8 @@ #ifdef CONFIG_BLK_DEV_INITRD if (start == 0) size = 0; - initrd_start = start; - initrd_end = start + size; + phys_initrd_start = __virt_to_phys(start); + phys_initrd_size = size; #endif } @@ -351,12 +360,12 @@ #if defined(CONFIG_VGA_CONSOLE) || defined(CONFIG_DUMMY_CONSOLE) struct screen_info screen_info = { - orig_video_lines: 30, - orig_video_cols: 80, - orig_video_mode: 0, - orig_video_ega_bx: 0, - orig_video_isVGA: 1, - orig_video_points: 8 + .orig_video_lines = 30, + .orig_video_cols = 80, + .orig_video_mode = 0, + .orig_video_ega_bx = 0, + .orig_video_isVGA = 1, + .orig_video_points = 8 }; static int __init parse_tag_videotext(const struct tag *tag) @@ -388,12 +397,22 @@ static int __init parse_tag_initrd(const struct tag *tag) { - setup_initrd(tag->u.initrd.start, tag->u.initrd.size); + phys_initrd_start = __virt_to_phys(tag->u.initrd.start); + phys_initrd_size = tag->u.initrd.size; return 0; } __tagtable(ATAG_INITRD, parse_tag_initrd); +static int __init parse_tag_initrd2(const struct tag *tag) +{ + phys_initrd_start = tag->u.initrd.start; + phys_initrd_size = tag->u.initrd.size; + return 0; +} + +__tagtable(ATAG_INITRD2, parse_tag_initrd2); + static int __init parse_tag_serialnr(const struct tag *tag) { system_serial_low = tag->u.serialnr.low; @@ -452,16 +471,30 @@ t->hdr.tag); } +static struct init_tags { + struct tag_header hdr1; + struct tag_core core; + struct tag_header hdr2; + struct tag_mem32 mem; + struct tag_header hdr3; +} init_tags __initdata = { + { tag_size(tag_core), ATAG_CORE }, + { 1, PAGE_SIZE, 0xff }, + { tag_size(tag_mem32), ATAG_MEM }, + { MEM_SIZE, PHYS_OFFSET }, + { 0, ATAG_NONE } +}; + void __init setup_arch(char **cmdline_p) { - struct tag *tags = NULL; + struct tag *tags = (struct tag *)&init_tags; struct machine_desc *mdesc; char *from = default_command_line; ROOT_DEV = MKDEV(0, 255); setup_processor(); - mdesc = setup_architecture(machine_arch_type); + mdesc = setup_machine(machine_arch_type); machine_name = mdesc->name; if (mdesc->soft_reboot) @@ -480,14 +513,16 @@ /* * If we have the old style parameters, convert them to - * a tag list before. + * a tag list. */ - if (tags && tags->hdr.tag != ATAG_CORE) - convert_to_tag_list((struct param_struct *)tags, - meminfo.nr_banks == 0); + if (tags->hdr.tag != ATAG_CORE) + convert_to_tag_list(tags); - if (tags && tags->hdr.tag == ATAG_CORE) + if (tags->hdr.tag == ATAG_CORE) { + if (meminfo.nr_banks != 0) + squash_mem_tags(tags); parse_tags(tags); + } if (meminfo.nr_banks == 0) { meminfo.nr_banks = 1; @@ -537,9 +572,8 @@ { int i; - seq_printf(m, "Processor\t: %s %s rev %d (%s)\n", - proc_info.manufacturer, proc_info.cpu_name, - (int)processor_id & 15, elf_platform); + seq_printf(m, "Processor\t: %s rev %d (%s)\n", + cpu_name, (int)processor_id & 15, elf_platform); seq_printf(m, "BogoMIPS\t: %lu.%02lu\n", loops_per_jiffy / (500000/HZ), @@ -578,8 +612,8 @@ } struct seq_operations cpuinfo_op = { - start: c_start, - next: c_next, - stop: c_stop, - show: c_show + .start = c_start, + .next = c_next, + .stop = c_stop, + .show = c_show }; diff -urN orig/arch/arm/kernel/signal.c linux/arch/arm/kernel/signal.c --- orig/arch/arm/kernel/signal.c Mon Aug 5 13:29:44 2002 +++ linux/arch/arm/kernel/signal.c Wed Feb 12 15:45:39 2003 @@ -405,7 +405,7 @@ regs->ARM_r0 = usig; regs->ARM_sp = (unsigned long)frame; regs->ARM_lr = retcode; - regs->ARM_pc = handler & (thumb ? ~1 : ~3); + regs->ARM_pc = handler; #ifdef CONFIG_CPU_32 regs->ARM_cpsr = cpsr; @@ -613,7 +613,7 @@ continue; switch (signr) { - case SIGCONT: case SIGCHLD: case SIGWINCH: + case SIGCONT: case SIGCHLD: case SIGWINCH: case SIGURG: continue; case SIGTSTP: case SIGTTIN: case SIGTTOU: @@ -621,13 +621,17 @@ continue; /* FALLTHRU */ - case SIGSTOP: + case SIGSTOP: { + struct signal_struct *sig; current->state = TASK_STOPPED; current->exit_code = signr; - if (!(current->p_pptr->sig->action[SIGCHLD-1].sa.sa_flags & SA_NOCLDSTOP)) + sig = current->p_pptr->sig; + if (sig && !(sig->action[SIGCHLD-1].sa.sa_flags & SA_NOCLDSTOP)) notify_parent(current, SIGCHLD); schedule(); + single_stepping |= ptrace_cancel_bpt(current); continue; + } case SIGQUIT: case SIGILL: case SIGTRAP: case SIGABRT: case SIGFPE: case SIGSEGV: diff -urN orig/arch/arm/kernel/time.c linux/arch/arm/kernel/time.c --- orig/arch/arm/kernel/time.c Fri Oct 26 16:45:51 2001 +++ linux/arch/arm/kernel/time.c Thu Feb 27 23:22:03 2003 @@ -202,7 +202,8 @@ } static struct irqaction timer_irq = { - name: "timer", + .name = "timer", + .flags = SA_INTERRUPT, }; /* diff -urN orig/arch/arm/kernel/traps.c linux/arch/arm/kernel/traps.c --- orig/arch/arm/kernel/traps.c Mon Aug 5 13:29:44 2002 +++ linux/arch/arm/kernel/traps.c Sun Sep 15 11:55:11 2002 @@ -1,7 +1,7 @@ /* * linux/arch/arm/kernel/traps.c * - * Copyright (C) 1995, 1996 Russell King + * Copyright (C) 1995-2002 Russell King * Fragments that appear the same as linux/arch/i386/kernel/traps.c (C) Linus Torvalds * * This program is free software; you can redistribute it and/or modify @@ -63,8 +63,17 @@ static void dump_mem(const char *str, unsigned long bottom, unsigned long top) { unsigned long p = bottom & ~31; + mm_segment_t fs; int i; + /* + * We need to switch to kernel mode so that we can use __get_user + * to safely read from kernel space. Note that we now dump the + * code first, just in case the backtrace kills us. + */ + fs = get_fs(); + set_fs(KERNEL_DS); + printk("%s(0x%08lx to 0x%08lx)\n", str, bottom, top); for (p = bottom & ~31; p < top;) { @@ -82,6 +91,8 @@ } printk ("\n"); } + + set_fs(fs); } static void dump_instr(struct pt_regs *regs) @@ -89,8 +100,17 @@ unsigned long addr = instruction_pointer(regs); const int thumb = thumb_mode(regs); const int width = thumb ? 4 : 8; + mm_segment_t fs; int i; + /* + * We need to switch to kernel mode so that we can use __get_user + * to safely read from kernel space. Note that we now dump the + * code first, just in case the backtrace kills us. + */ + fs = get_fs(); + set_fs(KERNEL_DS); + printk("Code: "); for (i = -4; i < 1; i++) { unsigned int val, bad; @@ -108,29 +128,36 @@ } } printk("\n"); + + set_fs(fs); } static void dump_stack(struct task_struct *tsk, unsigned long sp) { - dump_mem("Stack: ", sp - 16, 8192+(unsigned long)tsk); + dump_mem("Stack: ", sp, 8192+(unsigned long)tsk); } static void dump_backtrace(struct pt_regs *regs, struct task_struct *tsk) { - unsigned int fp; + unsigned int fp = regs->ARM_fp; + char *msg = ""; int ok = 1; - printk("Backtrace: "); - fp = regs->ARM_fp; +#ifdef CONFIG_FRAME_POINTER if (!fp) { - printk("no frame pointer"); + msg = "no frame pointer"; ok = 0; } else if (verify_stack(fp)) { - printk("invalid frame pointer 0x%08x", fp); + msg = "invalid frame pointer"; ok = 0; } else if (fp < 4096+(unsigned long)tsk) - printk("frame pointer underflow"); - printk("\n"); + msg = "frame pointer underflow"; +#else + msg = "not available"; + ok = 0; +#endif + + printk("Backtrace: %s\n", msg); if (ok) c_backtrace(fp, processor_mode(regs)); @@ -143,7 +170,7 @@ void show_trace_task(struct task_struct *tsk) { if (tsk != current) { - unsigned int fp = tsk->thread.save->fp; + unsigned int fp = thread_saved_fp(&tsk->thread); c_backtrace(fp, 0x10); } } @@ -163,26 +190,13 @@ printk("Internal error: %s: %x\n", str, err); printk("CPU: %d\n", smp_processor_id()); show_regs(regs); - printk("Process %s (pid: %d, stackpage=%08lx)\n", - current->comm, current->pid, 4096+(unsigned long)tsk); + printk("Process %s (pid: %d, stack limit = 0x%p)\n", + current->comm, current->pid, tsk + 1); if (!user_mode(regs) || in_interrupt()) { - mm_segment_t fs; - - /* - * We need to switch to kernel mode so that we can - * use __get_user to safely read from kernel space. - * Note that we now dump the code first, just in case - * the backtrace kills us. - */ - fs = get_fs(); - set_fs(KERNEL_DS); - dump_stack(tsk, (unsigned long)(regs + 1)); dump_backtrace(regs, tsk); dump_instr(regs); - - set_fs(fs); } spin_unlock_irq(&die_lock); @@ -270,7 +284,6 @@ asmlinkage void bad_mode(struct pt_regs *regs, int reason, int proc_mode) { unsigned int vectors = vectors_base(); - mm_segment_t fs; console_verbose(); @@ -278,24 +291,14 @@ handler[reason], processor_modes[proc_mode]); /* - * We need to switch to kernel mode so that we can use __get_user - * to safely read from kernel space. Note that we now dump the - * code first, just in case the backtrace kills us. - */ - fs = get_fs(); - set_fs(KERNEL_DS); - - /* * Dump out the vectors and stub routines. Maybe a better solution * would be to dump them out only if we detect that they are corrupted. */ dump_mem(KERN_CRIT "Vectors: ", vectors, vectors + 0x40); dump_mem(KERN_CRIT "Stubs: ", vectors + 0x200, vectors + 0x4b8); - set_fs(fs); - die("Oops", regs, 0); - cli(); + local_irq_disable(); panic("bad mode"); } @@ -356,20 +359,7 @@ return 0; case NR(breakpoint): /* SWI BREAK_POINT */ - /* - * The PC is always left pointing at the next - * instruction. Fix this. - */ - regs->ARM_pc -= 4; - __ptrace_cancel_bpt(current); - - info.si_signo = SIGTRAP; - info.si_errno = 0; - info.si_code = TRAP_BRKPT; - info.si_addr = (void *)instruction_pointer(regs) - - (thumb_mode(regs) ? 2 : 4); - - force_sig_info(SIGTRAP, &info, current); + ptrace_break(current, regs); return regs->ARM_r0; #ifdef CONFIG_CPU_32 @@ -522,12 +512,13 @@ void __init trap_init(void) { - extern void __trap_init(void *); + extern void __trap_init(unsigned long); + unsigned long base = vectors_base(); - __trap_init((void *)vectors_base()); - if (vectors_base() != 0) - printk(KERN_DEBUG "Relocating machine vectors to 0x%08x\n", - vectors_base()); + __trap_init(base); + if (base != 0) + printk(KERN_DEBUG "Relocating machine vectors to 0x%08lx\n", + base); #ifdef CONFIG_CPU_32 modify_domain(DOMAIN_USER, DOMAIN_CLIENT); #endif diff -urN orig/arch/arm/lib/Makefile linux/arch/arm/lib/Makefile --- orig/arch/arm/lib/Makefile Fri Oct 26 16:45:51 2001 +++ linux/arch/arm/lib/Makefile Thu Oct 24 14:33:56 2002 @@ -27,15 +27,16 @@ obj-l7200 := io-acorn.o obj-shark := io-shark.o obj-edb7211 := io-acorn.o +obj-riscstation := io-acorn.o floppydma.o obj-y += $(obj-$(MACHINE)) -ifeq ($(CONFIG_CPU_32v4),y) - v3 := n - v4 := y -else +ifeq ($(CONFIG_CPU_32v3),y) v3 := y v4 := n +else + v3 := n + v4 := y endif obj-y += io-readsb.o io-writesb.o @@ -51,4 +52,3 @@ csumpartialcopy.o: csumpartialcopygeneric.S csumpartialcopyuser.o: csumpartialcopygeneric.S - diff -urN orig/arch/arm/lib/backtrace.S linux/arch/arm/lib/backtrace.S --- orig/arch/arm/lib/backtrace.S Mon Aug 5 13:29:44 2002 +++ linux/arch/arm/lib/backtrace.S Thu Sep 5 23:15:26 2002 @@ -52,25 +52,32 @@ 3: tst frame, mask @ Check for address exceptions... bne 1b -1001: ldmda frame, {r0, r1, r2, r3} @ fp, sp, lr, pc - mov next, r0 - +1001: ldr next, [frame, #-12] @ get fp +1002: ldr r2, [frame, #-4] @ get lr +1003: ldr r3, [frame, #0] @ get pc sub save, r3, offset @ Correct PC for prefetching bic save, save, mask +1004: ldr r1, [save, #0] @ get instruction at function + mov r1, r1, lsr #10 + ldr r3, .Ldsi+4 + teq r1, r3 + subeq save, save, #4 adr r0, .Lfe mov r1, save bic r2, r2, mask - bl SYMBOL_NAME(printk) @ print pc and link register + bl printk @ print pc and link register - sub r0, frame, #16 -1002: ldr r1, [save, #4] @ get instruction at function+4 + ldr r0, [frame, #-8] @ get sp + sub r0, r0, #4 +1005: ldr r1, [save, #4] @ get instruction at function+4 mov r3, r1, lsr #10 ldr r2, .Ldsi+4 teq r3, r2 @ Check for stmia sp!, {args} addeq save, save, #4 @ next instruction bleq .Ldumpstm -1003: ldr r1, [save, #4] @ Get 'stmia sp!, {rlist, fp, ip, lr, pc}' instruction + sub r0, frame, #16 +1006: ldr r1, [save, #4] @ Get 'stmia sp!, {rlist, fp, ip, lr, pc}' instruction mov r3, r1, lsr #10 ldr r2, .Ldsi teq r3, r2 @@ -87,18 +94,21 @@ */ .section .fixup,"ax" .align 0 -1004: ldr r0, =.Lbad +1007: ldr r0, =.Lbad mov r1, frame - bl SYMBOL_NAME(printk) + bl printk LOADREGS(fd, sp!, {r4 - r8, pc}) .ltorg .previous .section __ex_table,"a" .align 3 - .long 1001b, 1004b - .long 1002b, 1004b - .long 1003b, 1004b + .long 1001b, 1007b + .long 1002b, 1007b + .long 1003b, 1007b + .long 1004b, 1007b + .long 1005b, 1007b + .long 1006b, 1007b .previous #define instr r4 @@ -121,12 +131,12 @@ ldr r2, [stack], #-4 mov r1, reg adr r0, .Lfp - bl SYMBOL_NAME(printk) + bl printk 2: subs reg, reg, #1 bpl 1b teq r7, #0 adrne r0, .Lcr - blne SYMBOL_NAME(printk) + blne printk mov r0, stack LOADREGS(fd, sp!, {instr, reg, stack, r7, pc}) diff -urN orig/arch/arm/lib/csumpartial.S linux/arch/arm/lib/csumpartial.S --- orig/arch/arm/lib/csumpartial.S Fri May 25 20:06:03 2001 +++ linux/arch/arm/lib/csumpartial.S Wed Sep 11 16:20:35 2002 @@ -49,8 +49,9 @@ /* we are now half-word aligned */ .less8_wordlp: -#ifdef __ARM_ARCH_4__ - ldrh td0, [buf], #2 +#if __LINUX_ARM_ARCH__ >= 4 + ldrh td0, [buf] + add buf, buf, #2 sub len, len, #2 #else ldrb td0, [buf], #1 @@ -79,8 +80,9 @@ adcnes sum, sum, td0, lsl #8 @ update checksum tst buf, #2 @ 32-bit aligned? -#ifdef __ARM_ARCH_4__ - ldrneh td0, [buf], #2 @ make 32-bit aligned +#if __LINUX_ARM_ARCH__ >= 4 + ldrneh td0, [buf] @ make 32-bit aligned + add buf, buf, #2 subne len, len, #2 #else ldrneb td0, [buf], #1 diff -urN orig/arch/arm/lib/delay.S linux/arch/arm/lib/delay.S --- orig/arch/arm/lib/delay.S Sat Apr 28 11:24:53 2001 +++ linux/arch/arm/lib/delay.S Fri Jun 8 14:38:48 2001 @@ -29,25 +29,29 @@ RETINSTR(moveq,pc,lr) /* - * loops = (r0 * 0x10c6 * 100 * loops_per_jiffie) / 2^32 + * loops = (r0 * 0x10c6 * 100 * loops_per_jiffy) / 2^32 + * + * Oh, if only we had a cycle counter... */ @ Delay routine ENTRY(__delay) subs r0, r0, #1 - RETINSTR(movcc,pc,lr) +#if 0 + RETINSTR(movls,pc,lr) subs r0, r0, #1 - RETINSTR(movcc,pc,lr) + RETINSTR(movls,pc,lr) subs r0, r0, #1 - RETINSTR(movcc,pc,lr) + RETINSTR(movls,pc,lr) subs r0, r0, #1 - RETINSTR(movcc,pc,lr) + RETINSTR(movls,pc,lr) subs r0, r0, #1 - RETINSTR(movcc,pc,lr) + RETINSTR(movls,pc,lr) subs r0, r0, #1 - RETINSTR(movcc,pc,lr) + RETINSTR(movls,pc,lr) subs r0, r0, #1 - RETINSTR(movcc,pc,lr) + RETINSTR(movls,pc,lr) subs r0, r0, #1 - bcs SYMBOL_NAME(__delay) +#endif + bhi SYMBOL_NAME(__delay) RETINSTR(mov,pc,lr) diff -urN orig/arch/arm/lib/longlong.h linux/arch/arm/lib/longlong.h --- orig/arch/arm/lib/longlong.h Fri Oct 26 16:45:51 2001 +++ linux/arch/arm/lib/longlong.h Thu Sep 5 23:09:22 2002 @@ -75,7 +75,7 @@ #if defined (__arm__) #define add_ssaaaa(sh, sl, ah, al, bh, bl) \ - __asm__ ("adds %1, %4, %5 + __asm__ ("adds %1, %4, %5 \n\ adc %0, %2, %3" \ : "=r" ((USItype) (sh)), \ "=&r" ((USItype) (sl)) \ @@ -84,7 +84,7 @@ "%r" ((USItype) (al)), \ "rI" ((USItype) (bl))) #define sub_ddmmss(sh, sl, ah, al, bh, bl) \ - __asm__ ("subs %1, %4, %5 + __asm__ ("subs %1, %4, %5 \n\ sbc %0, %2, %3" \ : "=r" ((USItype) (sh)), \ "=&r" ((USItype) (sl)) \ @@ -94,18 +94,18 @@ "rI" ((USItype) (bl))) #define umul_ppmm(xh, xl, a, b) \ {register USItype __t0, __t1, __t2; \ - __asm__ ("%@ Inlined umul_ppmm - mov %2, %5, lsr #16 - mov %0, %6, lsr #16 - bic %3, %5, %2, lsl #16 - bic %4, %6, %0, lsl #16 - mul %1, %3, %4 - mul %4, %2, %4 - mul %3, %0, %3 - mul %0, %2, %0 - adds %3, %4, %3 - addcs %0, %0, #65536 - adds %1, %1, %3, lsl #16 + __asm__ ("%@ Inlined umul_ppmm \n\ + mov %2, %5, lsr #16 \n\ + mov %0, %6, lsr #16 \n\ + bic %3, %5, %2, lsl #16 \n\ + bic %4, %6, %0, lsl #16 \n\ + mul %1, %3, %4 \n\ + mul %4, %2, %4 \n\ + mul %3, %0, %3 \n\ + mul %0, %2, %0 \n\ + adds %3, %4, %3 \n\ + addcs %0, %0, #65536 \n\ + adds %1, %1, %3, lsl #16 \n\ adc %0, %0, %3, lsr #16" \ : "=&r" ((USItype) (xh)), \ "=r" ((USItype) (xl)), \ diff -urN orig/arch/arm/lib/memchr.S linux/arch/arm/lib/memchr.S --- orig/arch/arm/lib/memchr.S Tue Oct 3 20:08:19 2000 +++ linux/arch/arm/lib/memchr.S Thu Sep 5 23:10:59 2002 @@ -15,11 +15,11 @@ .text .align 5 ENTRY(memchr) -1: ldrb r3, [r0], #1 +1: subs r2, r2, #1 + bmi 2f + ldrb r3, [r0], #1 teq r3, r1 - beq 2f - subs r2, r2, #1 - bpl 1b + bne 1b + sub r0, r0, #1 2: movne r0, #0 - subeq r0, r0, #1 RETINSTR(mov,pc,lr) diff -urN orig/arch/arm/lib/muldi3.c linux/arch/arm/lib/muldi3.c --- orig/arch/arm/lib/muldi3.c Fri Oct 26 16:45:51 2001 +++ linux/arch/arm/lib/muldi3.c Thu Sep 5 23:11:40 2002 @@ -33,18 +33,18 @@ #define umul_ppmm(xh, xl, a, b) \ {register USItype __t0, __t1, __t2; \ - __asm__ ("%@ Inlined umul_ppmm - mov %2, %5, lsr #16 - mov %0, %6, lsr #16 - bic %3, %5, %2, lsl #16 - bic %4, %6, %0, lsl #16 - mul %1, %3, %4 - mul %4, %2, %4 - mul %3, %0, %3 - mul %0, %2, %0 - adds %3, %4, %3 - addcs %0, %0, #65536 - adds %1, %1, %3, lsl #16 + __asm__ ("%@ Inlined umul_ppmm \n\ + mov %2, %5, lsr #16 \n\ + mov %0, %6, lsr #16 \n\ + bic %3, %5, %2, lsl #16 \n\ + bic %4, %6, %0, lsl #16 \n\ + mul %1, %3, %4 \n\ + mul %4, %2, %4 \n\ + mul %3, %0, %3 \n\ + mul %0, %2, %0 \n\ + adds %3, %4, %3 \n\ + addcs %0, %0, #65536 \n\ + adds %1, %1, %3, lsl #16 \n\ adc %0, %0, %3, lsr #16" \ : "=&r" ((USItype) (xh)), \ "=r" ((USItype) (xl)), \ diff -urN orig/arch/arm/lib/strchr.S linux/arch/arm/lib/strchr.S --- orig/arch/arm/lib/strchr.S Tue Oct 3 20:08:19 2000 +++ linux/arch/arm/lib/strchr.S Sat Sep 21 11:20:06 2002 @@ -15,11 +15,12 @@ .text .align 5 ENTRY(strchr) + and r1, r1, #0xff 1: ldrb r2, [r0], #1 teq r2, r1 teqne r2, #0 bne 1b - teq r2, #0 - moveq r0, #0 - subne r0, r0, #1 + teq r2, r1 + movne r0, #0 + subeq r0, r0, #1 RETINSTR(mov,pc,lr) diff -urN orig/arch/arm/lib/udivdi3.c linux/arch/arm/lib/udivdi3.c --- orig/arch/arm/lib/udivdi3.c Fri Oct 26 16:45:51 2001 +++ linux/arch/arm/lib/udivdi3.c Sat May 18 21:17:48 2002 @@ -229,3 +229,14 @@ { return __udivmoddi4 (n, d, (UDItype *) 0); } + +UDItype +__umoddi3 (UDItype u, UDItype v) +{ + UDItype w; + + (void) __udivmoddi4 (u ,v, &w); + + return w; +} + diff -urN orig/arch/arm/mach-anakin/irq.c linux/arch/arm/mach-anakin/irq.c --- orig/arch/arm/mach-anakin/irq.c Mon Sep 3 14:14:40 2001 +++ linux/arch/arm/mach-anakin/irq.c Tue Feb 4 15:08:01 2003 @@ -15,6 +15,8 @@ #include #include #include +#include +#include #include #include @@ -49,9 +51,9 @@ } static struct irqaction anakin_irq = { - name: "Anakin IRQ", - handler: anakin_interrupt, - flags: SA_INTERRUPT + .name = "Anakin IRQ", + .handler = anakin_interrupt, + .flags = SA_INTERRUPT, }; void __init diff -urN orig/arch/arm/mach-at91rm9200dk/Makefile linux/arch/arm/mach-at91rm9200dk/Makefile --- orig/arch/arm/mach-at91rm9200dk/Makefile Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-at91rm9200dk/Makefile Wed Jan 22 23:46:12 2003 @@ -0,0 +1,21 @@ +# +# Makefile for the linux kernel. +# +# Note! Dependencies are done automagically by 'make dep', which also +# removes any old dependencies. DON'T put your own dependencies here +# unless it's something special (ie not a .c file). + +USE_STANDARD_AS_RULE := true + +O_TARGET := at91rm9200dk.o + +# Object file lists. + +obj-y := core.o +obj-m := +obj-n := +obj- := + +export-objs := + +include $(TOPDIR)/Rules.make diff -urN orig/arch/arm/mach-at91rm9200dk/core.c linux/arch/arm/mach-at91rm9200dk/core.c --- orig/arch/arm/mach-at91rm9200dk/core.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-at91rm9200dk/core.c Thu Feb 27 23:21:11 2003 @@ -0,0 +1,129 @@ +/* + * linux/arch/arm/mach-at91rm9200dk/arch.c + * + * Copyright (C) 2002 ATMEL Rousset + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include + + +static void AT91F_mask_irq(unsigned int irq) +{ + __raw_writel(1 << irq, &(((AT91PS_SYS)AT91C_VA_BASE_SYS)->AIC_IDCR) ); +} + +static void AT91F_unmask_irq(unsigned int irq) +{ + __raw_writel(1 << irq, &(((AT91PS_SYS)AT91C_VA_BASE_SYS)->AIC_IECR) ); +} + +void __init at91rm9200dk_init_irq(void) +{ + unsigned int i; + +#if (NR_IRQS < 32) +#error AT91RM9200 IRQ number < 32 +#endif + + // Init the irq_desc structure + for (i = 0; i < NR_IRQS; i++) { + // Create an IRQ descriptor for each AIC entry + // Init each SMR AIC entry + ((AT91PS_SYS)AT91C_VA_BASE_SYS)->AIC_SVR[i] = i; + ((AT91PS_SYS)AT91C_VA_BASE_SYS)->AIC_SMR[i] = AT91C_AIC_PRIOR_LOWEST | AT91C_AIC_SRCTYPE_INT_LEVEL_SENSITIVE; + irq_desc[i].valid = 1; + irq_desc[i].probe_ok = 1; + irq_desc[i].mask_ack = AT91F_mask_irq; + irq_desc[i].mask = AT91F_mask_irq; + irq_desc[i].unmask = AT91F_unmask_irq; + + // Perform 8 End Of Interrupt Command to make sure AIC will not Lock out nIRQ + if (i < 8) + ((AT91PS_SYS)AT91C_VA_BASE_SYS)->AIC_EOICR = ((AT91PS_SYS)AT91C_VA_BASE_SYS)->AIC_EOICR; + } + + // Disable all interrupts + ((AT91PS_SYS)AT91C_VA_BASE_SYS)->AIC_IDCR = -1; + ((AT91PS_SYS)AT91C_VA_BASE_SYS)->AIC_ICCR = -1; +} + +static struct map_desc at91rm9200dk_io_desc[] __initdata = { + /* virtual physical length domain r w c b */ + { (unsigned long)AT91C_VA_BASE_SYS, (unsigned long)AT91C_BASE_SYS, SZ_4K , DOMAIN_IO, 0,1, 0, 0}, + { (unsigned long)AT91C_VA_BASE_SPI, (unsigned long)AT91C_BASE_SPI, SZ_16K , DOMAIN_IO, 0,1, 0, 0}, + { (unsigned long)AT91C_VA_BASE_SSC2, (unsigned long)AT91C_BASE_SSC2, SZ_16K , DOMAIN_IO, 0,1, 0, 0}, + { (unsigned long)AT91C_VA_BASE_SSC1, (unsigned long)AT91C_BASE_SSC1, SZ_16K , DOMAIN_IO, 0,1, 0, 0}, + { (unsigned long)AT91C_VA_BASE_SSC0, (unsigned long)AT91C_BASE_SSC0, SZ_16K , DOMAIN_IO, 0,1, 0, 0}, + { (unsigned long)AT91C_VA_BASE_US3, (unsigned long)AT91C_BASE_US3, SZ_16K , DOMAIN_IO, 0,1, 0, 0}, + { (unsigned long)AT91C_VA_BASE_US2, (unsigned long)AT91C_BASE_US2, SZ_16K , DOMAIN_IO, 0,1, 0, 0}, + { (unsigned long)AT91C_VA_BASE_US1, (unsigned long)AT91C_BASE_US1, SZ_16K , DOMAIN_IO, 0,1, 0, 0}, + { (unsigned long)AT91C_VA_BASE_US0, (unsigned long)AT91C_BASE_US0, SZ_16K , DOMAIN_IO, 0,1, 0, 0}, + { (unsigned long)AT91C_VA_BASE_EMAC, (unsigned long)AT91C_BASE_EMAC, SZ_16K , DOMAIN_IO, 0,1, 0, 0}, + { (unsigned long)AT91C_VA_BASE_TWI, (unsigned long)AT91C_BASE_TWI, SZ_16K , DOMAIN_IO, 0,1, 0, 0}, + { (unsigned long)AT91C_VA_BASE_MCI, (unsigned long)AT91C_BASE_MCI, SZ_16K , DOMAIN_IO, 0,1, 0, 0}, + { (unsigned long)AT91C_VA_BASE_UDP, (unsigned long)AT91C_BASE_UDP, SZ_16K , DOMAIN_IO, 0,1, 0, 0}, + { (unsigned long)AT91C_VA_BASE_TCB0, (unsigned long)AT91C_BASE_TCB0, SZ_16K , DOMAIN_IO, 0,1, 0, 0}, + { (unsigned long)AT91C_VA_BASE_TCB1, (unsigned long)AT91C_BASE_TCB1, SZ_16K , DOMAIN_IO, 0,1, 0, 0}, + { (unsigned long)AT91C_VA_BASE_DMA_EMAC, (unsigned long)AT91C_BASE_DMA_EMAC, SZ_1M , DOMAIN_IO, 0,1, 0, 0}, + LAST_DESC +}; + +void __init at91rm9200dk_map_io(void) +{ + iotable_init(at91rm9200dk_io_desc); + at91rm9200dk_register_us1(); + at91rm9200dk_register_dbgu(); +} + +#define AT91C_ZIMAGE_SIZE 3*1024*1024 // Size of the compressed ramdisk + +static void __init +at91rm9200dk_fixup(struct machine_desc *desc, struct param_struct *unused, + char **cmdline, struct meminfo *mi) +{ + ROOT_DEV = MKDEV(RAMDISK_MAJOR, 0); + setup_ramdisk( 1, 0, 0, CONFIG_BLK_DEV_RAM_SIZE ); + setup_initrd( 0xc1000000, AT91C_ZIMAGE_SIZE ); +} + +MACHINE_START(AT91RM9200DK, "AT91RM9200DK") + MAINTAINER("ATMEL") + BOOT_MEM(AT91C_BASE_SDRAM, (unsigned long)AT91C_BASE_SYS, (unsigned long)AT91C_VA_BASE_SYS) + BOOT_PARAMS(0x00000000) + FIXUP(at91rm9200dk_fixup) + MAPIO(at91rm9200dk_map_io) + INITIRQ(at91rm9200dk_init_irq) +MACHINE_END diff -urN orig/arch/arm/mach-clps711x/Makefile linux/arch/arm/mach-clps711x/Makefile --- orig/arch/arm/mach-clps711x/Makefile Mon Aug 5 13:29:44 2002 +++ linux/arch/arm/mach-clps711x/Makefile Thu Feb 27 23:20:10 2003 @@ -16,12 +16,13 @@ obj-n := obj- := -export-objs := leds-p720t.o +export-objs := p720t-leds.o obj-$(CONFIG_ARCH_AUTCPU12) += autcpu12.o obj-$(CONFIG_ARCH_CDB89712) += cdb89712.o obj-$(CONFIG_ARCH_CLEP7312) += clep7312.o obj-$(CONFIG_ARCH_EDB7211) += edb7211-arch.o edb7211-mm.o +obj-$(CONFIG_ARCH_GUIDEA07) += guide-a07.o obj-$(CONFIG_ARCH_P720T) += p720t.o obj-$(CONFIG_ARCH_FORTUNET) += fortunet.o leds-$(CONFIG_ARCH_P720T) += p720t-leds.o diff -urN orig/arch/arm/mach-clps711x/fortunet.c linux/arch/arm/mach-clps711x/fortunet.c --- orig/arch/arm/mach-clps711x/fortunet.c Mon Aug 5 13:29:44 2002 +++ linux/arch/arm/mach-clps711x/fortunet.c Sat May 18 21:17:48 2002 @@ -31,6 +31,10 @@ #include #include +#include +#include +#include + #include #include @@ -48,9 +52,11 @@ int extra_param_type; int extra_param_ptr; int command_line; + int extra_ram_start; + int extra_ram_size; } IMAGE_PARAMS; -#define IMAGE_PARAMS_PHYS 0xC01F0000 +#define IMAGE_PARAMS_PHYS 0xC0200000 static void __init fortunet_fixup(struct machine_desc *desc, struct param_struct *params, @@ -67,14 +73,23 @@ } #endif memmap.bank[0].size = ip->ram_size; + memmap.bank[0].node = PHYS_TO_NID(0xC0000000); + if(ip->extra_ram_size) + { + memmap.bank[1].start = ip->extra_ram_start; + memmap.bank[1].size = ip->extra_ram_size; + memmap.bank[1].node = PHYS_TO_NID(ip->extra_ram_start); + mi->nr_banks=2; + } memmap.end = ip->ram_size+0xC0000000; *mi = memmap; } MACHINE_START(FORTUNET, "ARM-FortuNet") MAINTAINER("FortuNet Inc.") - BOOT_MEM(0xc0000000, 0x80000000, 0xf0000000) + BOOT_MEM(0xc0000000, 0x80000000, 0xff000000) BOOT_PARAMS(0x00000000) + VIDEO(0xC0000000,0xC00020000) FIXUP(fortunet_fixup) MAPIO(clps711x_map_io) INITIRQ(clps711x_init_irq) diff -urN orig/arch/arm/mach-clps711x/guide-a07.c linux/arch/arm/mach-clps711x/guide-a07.c --- orig/arch/arm/mach-clps711x/guide-a07.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-clps711x/guide-a07.c Thu Feb 27 23:20:12 2003 @@ -0,0 +1,86 @@ +/* + * linux/arch/arm/mach-clps711x/guide-a07.c + * + * Copyright (C) 2003 Iders Incorporated + * Copyright (C) 2000-2001 Deep Blue Solutions Ltd + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include +#include + +extern void clps711x_init_irq(void); +extern void clps711x_map_io(void); + +#ifndef CONFIG_I2C_GUIDE +#error Config error - The Guide A07 requires I2C Guide GPIO support. Please enable it. +#endif + +/* + * Map the CS89712 Ethernet port. That should be moved to the + * ethernet driver, perhaps. + * also map the Guide FPGA and persistance locations + */ +static struct map_desc guide_a07_io_desc[] __initdata = { + { ETHER_BASE, ETHER_START, ETHER_SIZE, + DOMAIN_IO, 0, 1, 0, 0 }, + { GD_A07_FPGA_BASE, GD_A07_FPGA_START, GD_A07_FPGA_SIZE, + DOMAIN_IO, 0, 1, 0, 0 }, + { GD_A07_PERSISTANCE_BASE, GD_A07_PERSISTANCE_START, GD_A07_PERSISTANCE_SIZE, + DOMAIN_IO, 0, 1, 0, 0 }, + LAST_DESC +}; + +static void __init +fixup_guide_a07(struct machine_desc *desc, struct param_struct *params, + char **cmdline, struct meminfo *mi) +{ +} + +static void __init guide_a07_map_io(void) +{ + clps711x_map_io(); + iotable_init(guide_a07_io_desc); +} + +MACHINE_START(GUIDEA07, "Guide A07 (cs89712 core)") + MAINTAINER("Cam Mayor") + BOOT_MEM(0xc0000000, 0x80000000, 0xff000000) + BOOT_PARAMS(0xc0000100) + FIXUP(fixup_guide_a07) + MAPIO(guide_a07_map_io) + INITIRQ(clps711x_init_irq) +MACHINE_END + +static int guide_a07_hw_init(void) +{ + return 0; +} + +__initcall(guide_a07_hw_init); + diff -urN orig/arch/arm/mach-clps711x/irq.c linux/arch/arm/mach-clps711x/irq.c --- orig/arch/arm/mach-clps711x/irq.c Mon Aug 5 13:29:44 2002 +++ linux/arch/arm/mach-clps711x/irq.c Tue Feb 4 15:07:22 2003 @@ -18,6 +18,8 @@ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include +#include +#include #include #include diff -urN orig/arch/arm/mach-ebsa110/irq.c linux/arch/arm/mach-ebsa110/irq.c --- orig/arch/arm/mach-ebsa110/irq.c Sat Apr 28 11:24:53 2001 +++ linux/arch/arm/mach-ebsa110/irq.c Wed Mar 19 16:52:57 2003 @@ -11,6 +11,9 @@ * 22-08-1998 RMK Restructured IRQ routines */ #include +#include +#include +#include #include #include @@ -35,14 +38,14 @@ unsigned long flags; int irq; - save_flags_cli (flags); + local_irq_save(flags); __raw_writeb(0xff, IRQ_MCLR); __raw_writeb(0x55, IRQ_MSET); __raw_writeb(0x00, IRQ_MSET); if (__raw_readb(IRQ_MASK) != 0x55) while (1); __raw_writeb(0xff, IRQ_MCLR); /* clear all interrupt enables */ - restore_flags (flags); + local_irq_restore(flags); for (irq = 0; irq < NR_IRQS; irq++) { irq_desc[irq].valid = 1; diff -urN orig/arch/arm/mach-epxa/Makefile linux/arch/arm/mach-epxa/Makefile --- orig/arch/arm/mach-epxa/Makefile Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-epxa/Makefile Mon Aug 5 23:25:12 2002 @@ -0,0 +1,24 @@ +# +# Makefile for the linux kernel. +# +# Note! Dependencies are done automagically by 'make dep', which also +# removes any old dependencies. DON'T put your own dependencies here +# unless it's something special (ie not a .c file). + +USE_STANDARD_AS_RULE := true + +O_TARGET := epxa.o + +# Object file lists. + +obj-y := irq.o mm.o time.o +obj-m := +obj-n := +obj- := + +export-objs := + +obj-$(CONFIG_EPXA1DB) += arch-epxa1db.o +obj-$(CONFIG_EPXA10DB) += arch-epxa10db.o + +include $(TOPDIR)/Rules.make diff -urN orig/arch/arm/mach-epxa/arch-epxa10db.c linux/arch/arm/mach-epxa/arch-epxa10db.c --- orig/arch/arm/mach-epxa/arch-epxa10db.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-epxa/arch-epxa10db.c Fri Feb 21 15:17:33 2003 @@ -0,0 +1,60 @@ +/* + * linux/arch/arm/mach-epxa/arch-epxa10db.c + * + * Copyright (C) 2001 Altera Corporation + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ +#include +#include +#include +#include + +#include +#include +#include +#include + +#include + +extern void epxa_map_io(void); +extern void epxa_init_irq(void); + + +static void __init +epxa10db_fixup(struct machine_desc *desc, struct param_struct *params, + char **cmdline, struct meminfo *mi) +{ + + mi->nr_banks = 1; + mi->bank[0].start = 0; + mi->bank[0].size = (128*1024*1024); + mi->bank[0].node = 0; + +/* + ROOT_DEV = MKDEV(RAMDISK_MAJOR,0); + setup_ramdisk( 1, 0, 0, 8192 ); + setup_initrd(0xc0200000, 6*1024*1024); +*/ +} + +MACHINE_START(CAMELOT, "Altera Epxa10db") + MAINTAINER("Altera Corporation") + BOOT_MEM(0x00000000, 0x7fffc000, 0xffffc000) + FIXUP(epxa10db_fixup) + MAPIO(epxa_map_io) + INITIRQ(epxa_init_irq) + BOOT_PARAMS(0x100) +MACHINE_END diff -urN orig/arch/arm/mach-epxa/arch-epxa1db.c linux/arch/arm/mach-epxa/arch-epxa1db.c --- orig/arch/arm/mach-epxa/arch-epxa1db.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-epxa/arch-epxa1db.c Thu Oct 24 13:35:37 2002 @@ -0,0 +1,59 @@ +/* + * linux/arch/arm/mach-epxa/arch-epxa10db.c + * + * Copyright (C) 2001 Altera Corporation + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ +#include +#include +#include +#include + +#include +#include +#include +#include + +#include + +extern void epxa_map_io(void); +extern void epxa_init_irq(void); + +static void __init +epxa1db_fixup(struct machine_desc *desc, struct param_struct *params, + char **cmdline, struct meminfo *mi) +{ + + mi->nr_banks = 1; + mi->bank[0].start = 0; + mi->bank[0].size = (32*1024*1024); + mi->bank[0].node = 0; + +/* + ROOT_DEV = MKDEV(RAMDISK_MAJOR,0); + setup_ramdisk( 1, 0, 0, 8192 ); + setup_initrd(0xc0200000, 6*1024*1024); +*/ +} + +MACHINE_START(CAMELOT, "Altera Epxa1db") + MAINTAINER("Altera Corporation") + BOOT_MEM(0x00000000, 0x7fffc000, 0xffffc000) + FIXUP(epxa1db_fixup) + MAPIO(epxa_map_io) + INITIRQ(epxa_init_irq) + BOOT_PARAMS(0x100) +MACHINE_END diff -urN orig/arch/arm/mach-epxa/dma.c linux/arch/arm/mach-epxa/dma.c --- orig/arch/arm/mach-epxa/dma.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-epxa/dma.c Fri Nov 23 10:20:49 2001 @@ -0,0 +1,36 @@ +/* + * linux/arch/arm/mach-epxa10db/dma.c + * + * Copyright (C) 1999 ARM Limited + * Copyright (C) 2000 Deep Blue Solutions Ltd + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ +#include +#include +#include +#include + +#include +#include +#include +#include +#include + +#include + +void __init arch_dma_init(dma_t *dma) +{ +} diff -urN orig/arch/arm/mach-epxa/irq.c linux/arch/arm/mach-epxa/irq.c --- orig/arch/arm/mach-epxa/irq.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-epxa/irq.c Wed Feb 12 17:14:51 2003 @@ -0,0 +1,74 @@ +/* + * linux/arch/arm/mach-epxa10db/irq.c + * + * Copyright (C) 2001 Altera Corporation + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include + + +static void mask_irq(unsigned int irq) +{ + __raw_writel(1 << irq, INT_MC(IO_ADDRESS(EXC_INT_CTRL00_BASE))); +} + +static void unmask_irq(unsigned int irq) +{ + __raw_writel(1 << irq, INT_MS(IO_ADDRESS(EXC_INT_CTRL00_BASE))); +} + +void __init epxa_init_irq(void) +{ + unsigned int i; + + /* + * This bit sets up the interrupt controller using + * the 6 PLD interrupts mode (the default) each + * irqs is assigned a priority which is the same + * as its interrupt number. This scheme is used because + * its easy, but you may want to change it depending + * on the contents of your PLD + */ + + __raw_writel(3,INT_MODE(IO_ADDRESS(EXC_INT_CTRL00_BASE))); + for (i = 0; i < NR_IRQS; i++){ + __raw_writel(i+1, INT_PRIORITY_P0(IO_ADDRESS(EXC_INT_CTRL00_BASE)) + (4*i)); + } + + + for (i = 0; i < NR_IRQS; i++) { + + irq_desc[i].valid = 1; + irq_desc[i].probe_ok = 1; + irq_desc[i].mask_ack = mask_irq; + irq_desc[i].mask = mask_irq; + irq_desc[i].unmask = unmask_irq; + } + + /* Disable all interrupt */ + __raw_writel(-1,INT_MC(IO_ADDRESS(EXC_INT_CTRL00_BASE))); + +} diff -urN orig/arch/arm/mach-epxa/mm.c linux/arch/arm/mach-epxa/mm.c --- orig/arch/arm/mach-epxa/mm.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-epxa/mm.c Mon Aug 5 23:25:13 2002 @@ -0,0 +1,43 @@ +/* + * linux/arch/arm/mach-epxa10db/mm.c + * + * MM routines for Altera'a Epxa10db board + * + * Copyright (C) 2001 Altera Corporation + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ +#include +#include +#include + +#include +#include +#include +#include +#include + +#include + +/* Page table mapping for I/O region */ + +static struct map_desc epxa_io_desc[] __initdata = { + { IO_ADDRESS(EXC_REGISTERS_BASE), EXC_REGISTERS_BASE, SZ_4K , DOMAIN_IO, 0, 1}, LAST_DESC +}; + +void __init epxa_map_io(void) +{ + iotable_init(epxa_io_desc); +} diff -urN orig/arch/arm/mach-epxa/time.c linux/arch/arm/mach-epxa/time.c --- orig/arch/arm/mach-epxa/time.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-epxa/time.c Fri Oct 12 14:28:54 2001 @@ -0,0 +1,33 @@ +/* + * linux/arch/arm/mach-epxa10db/time.c + * + * Copyright (C) 2000 Deep Blue Solutions + * Copyright (C) 2001 Altera Corporation + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + */ +#include +#include +#include + +#include + + + +extern int (*set_rtc)(void); + +static int epxa10db_set_rtc(void) +{ + return 1; +} + +static int epxa10db_rtc_init(void) +{ + set_rtc = epxa10db_set_rtc; + + return 0; +} + +__initcall(epxa10db_rtc_init); diff -urN orig/arch/arm/mach-epxa10db/Makefile linux/arch/arm/mach-epxa10db/Makefile --- orig/arch/arm/mach-epxa10db/Makefile Fri Nov 16 10:09:48 2001 +++ linux/arch/arm/mach-epxa10db/Makefile Thu Jan 1 01:00:00 1970 @@ -1,22 +0,0 @@ -# -# Makefile for the linux kernel. -# -# Note! Dependencies are done automagically by 'make dep', which also -# removes any old dependencies. DON'T put your own dependencies here -# unless it's something special (ie not a .c file). - -USE_STANDARD_AS_RULE := true - -O_TARGET := epxa10db.o - -# Object file lists. - -obj-y := arch.o irq.o mm.o time.o -obj-m := -obj-n := -obj- := - -export-objs := - - -include $(TOPDIR)/Rules.make diff -urN orig/arch/arm/mach-epxa10db/arch.c linux/arch/arm/mach-epxa10db/arch.c --- orig/arch/arm/mach-epxa10db/arch.c Mon Aug 5 13:29:44 2002 +++ linux/arch/arm/mach-epxa10db/arch.c Thu Jan 1 01:00:00 1970 @@ -1,61 +0,0 @@ -/* - * linux/arch/arm/mach-epxa10db/arch.c - * - * Copyright (C) 2000 Deep Blue Solutions Ltd - * Copyright (C) 2001 Altera Corporation - * - * This program is free software; you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation; either version 2 of the License, or - * (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program; if not, write to the Free Software - * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA - */ -#include -#include -#include -#include - -#include -#include -#include -#include - -#include - -extern void epxa10db_map_io(void); -extern void epxa10db_init_irq(void); - - -static void __init -epxa10db_fixup(struct machine_desc *desc, struct param_struct *params, - char **cmdline, struct meminfo *mi) -{ - - mi->nr_banks = 1; - mi->bank[0].start = 0; - mi->bank[0].size = (32*1024*1024); - mi->bank[0].node = 0; - -/* - ROOT_DEV = MKDEV(RAMDISK_MAJOR,0); - setup_ramdisk( 1, 0, 0, 8192 ); - setup_initrd(0xc0200000, 6*1024*1024); -*/ -} - -MACHINE_START(CAMELOT, "Altera Epxa10db") - MAINTAINER("Altera Corporation") - BOOT_MEM(0x00000000, 0x7fffc000, 0xffffc000) - FIXUP(epxa10db_fixup) - MAPIO(epxa10db_map_io) - INITIRQ(epxa10db_init_irq) - -MACHINE_END diff -urN orig/arch/arm/mach-epxa10db/dma.c linux/arch/arm/mach-epxa10db/dma.c --- orig/arch/arm/mach-epxa10db/dma.c Fri Nov 23 10:12:06 2001 +++ linux/arch/arm/mach-epxa10db/dma.c Thu Jan 1 01:00:00 1970 @@ -1,36 +0,0 @@ -/* - * linux/arch/arm/mach-epxa10db/dma.c - * - * Copyright (C) 1999 ARM Limited - * Copyright (C) 2000 Deep Blue Solutions Ltd - * - * This program is free software; you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation; either version 2 of the License, or - * (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program; if not, write to the Free Software - * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA - */ -#include -#include -#include -#include - -#include -#include -#include -#include -#include - -#include - -void __init arch_dma_init(dma_t *dma) -{ -} diff -urN orig/arch/arm/mach-epxa10db/irq.c linux/arch/arm/mach-epxa10db/irq.c --- orig/arch/arm/mach-epxa10db/irq.c Fri Nov 16 10:09:48 2001 +++ linux/arch/arm/mach-epxa10db/irq.c Thu Jan 1 01:00:00 1970 @@ -1,70 +0,0 @@ -/* - * linux/arch/arm/mach-epxa10db/irq.c - * - * Copyright (C) 2001 Altera Corporation - * - * This program is free software; you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation; either version 2 of the License, or - * (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program; if not, write to the Free Software - * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA - */ -#include -#include -#include -#include -#include -#include -#include - - -static void mask_irq(unsigned int irq) -{ - __raw_writel(1 << irq, INT_MC(IO_ADDRESS(EXC_INT_CTRL00_BASE))); -} - -static void unmask_irq(unsigned int irq) -{ - __raw_writel(1 << irq, INT_MS(IO_ADDRESS(EXC_INT_CTRL00_BASE))); -} - -void __init epxa10db_init_irq(void) -{ - unsigned int i; - - /* - * This bit sets up the interrupt controller using - * the 6 PLD interrupts mode (the default) each - * irqs is assigned a priority which is the same - * as its interrupt number. This scheme is used because - * its easy, but you may want to change it depending - * on the contents of your PLD - */ - - __raw_writel(3,INT_MODE(IO_ADDRESS(EXC_INT_CTRL00_BASE))); - for (i = 0; i < NR_IRQS; i++){ - __raw_writel(i+1, INT_PRIORITY_P0(IO_ADDRESS(EXC_INT_CTRL00_BASE)) + (4*i)); - } - - - for (i = 0; i < NR_IRQS; i++) { - - irq_desc[i].valid = 1; - irq_desc[i].probe_ok = 1; - irq_desc[i].mask_ack = mask_irq; - irq_desc[i].mask = mask_irq; - irq_desc[i].unmask = unmask_irq; - } - - /* Disable all interrupt */ - __raw_writel(-1,INT_MC(IO_ADDRESS(EXC_INT_CTRL00_BASE))); - -} diff -urN orig/arch/arm/mach-epxa10db/mm.c linux/arch/arm/mach-epxa10db/mm.c --- orig/arch/arm/mach-epxa10db/mm.c Mon Aug 5 13:29:44 2002 +++ linux/arch/arm/mach-epxa10db/mm.c Thu Jan 1 01:00:00 1970 @@ -1,49 +0,0 @@ -/* - * linux/arch/arm/mach-epxa10db/mm.c - * - * MM routines for Altera'a Epxa10db board - * - * Copyright (C) 2001 Altera Corporation - * - * This program is free software; you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation; either version 2 of the License, or - * (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program; if not, write to the Free Software - * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA - */ -#include -#include -#include - -#include -#include -#include -#include -#include - -#include - -/* Page table mapping for I/O region */ - -static struct map_desc epxa10db_io_desc[] __initdata = { - { IO_ADDRESS(EXC_REGISTERS_BASE), EXC_REGISTERS_BASE, SZ_16K , DOMAIN_IO, 0, 1}, -{IO_ADDRESS(EXC_PLD_BLOCK0_BASE), EXC_PLD_BLOCK0_BASE, SZ_16K , DOMAIN_IO, 0, 1}, -{IO_ADDRESS(EXC_PLD_BLOCK1_BASE), EXC_PLD_BLOCK1_BASE, SZ_16K , DOMAIN_IO, 0, 1}, -{IO_ADDRESS(EXC_PLD_BLOCK2_BASE), EXC_PLD_BLOCK2_BASE, SZ_16K , DOMAIN_IO, 0, 1}, -{IO_ADDRESS(EXC_PLD_BLOCK3_BASE), EXC_PLD_BLOCK3_BASE, SZ_16K , DOMAIN_IO, 0, 1}, -{ FLASH_VADDR(EXC_EBI_BLOCK0_BASE), EXC_EBI_BLOCK0_BASE, SZ_16M , DOMAIN_IO, 0, 1}, - LAST_DESC -}; - -void __init epxa10db_map_io(void) -{ - iotable_init(epxa10db_io_desc); -} diff -urN orig/arch/arm/mach-epxa10db/time.c linux/arch/arm/mach-epxa10db/time.c --- orig/arch/arm/mach-epxa10db/time.c Fri Nov 16 10:09:48 2001 +++ linux/arch/arm/mach-epxa10db/time.c Thu Jan 1 01:00:00 1970 @@ -1,33 +0,0 @@ -/* - * linux/arch/arm/mach-epxa10db/time.c - * - * Copyright (C) 2000 Deep Blue Solutions - * Copyright (C) 2001 Altera Corporation - * - * This program is free software; you can redistribute it and/or modify - * it under the terms of the GNU General Public License version 2 as - * published by the Free Software Foundation. - */ -#include -#include -#include - -#include - - - -extern int (*set_rtc)(void); - -static int epxa10db_set_rtc(void) -{ - return 1; -} - -static int epxa10db_rtc_init(void) -{ - set_rtc = epxa10db_set_rtc; - - return 0; -} - -__initcall(epxa10db_rtc_init); diff -urN orig/arch/arm/mach-footbridge/arch.c linux/arch/arm/mach-footbridge/arch.c --- orig/arch/arm/mach-footbridge/arch.c Mon Sep 3 14:14:40 2001 +++ linux/arch/arm/mach-footbridge/arch.c Sun Mar 24 21:11:24 2002 @@ -41,10 +41,12 @@ fixup_ebsa285(struct machine_desc *desc, struct param_struct *params, char **cmdline, struct meminfo *mi) { +#if defined(CONFIG_VGA_CONSOLE) || defined(CONFIG_DUMMY_CONSOLE) ORIG_X = params->u1.s.video_x; ORIG_Y = params->u1.s.video_y; ORIG_VIDEO_COLS = params->u1.s.video_num_cols; ORIG_VIDEO_LINES = params->u1.s.video_num_rows; +#endif } MACHINE_START(EBSA285, "EBSA285") diff -urN orig/arch/arm/mach-footbridge/cats-pci.c linux/arch/arm/mach-footbridge/cats-pci.c --- orig/arch/arm/mach-footbridge/cats-pci.c Sat Jul 21 10:46:32 2001 +++ linux/arch/arm/mach-footbridge/cats-pci.c Sun Sep 15 12:57:56 2002 @@ -32,9 +32,9 @@ } struct hw_pci cats_pci __initdata = { - setup_resources: dc21285_setup_resources, - init: dc21285_init, - mem_offset: DC21285_PCI_MEM, - swizzle: no_swizzle, - map_irq: cats_map_irq, + .setup_resources = dc21285_setup_resources, + .init = dc21285_init, + .mem_offset = DC21285_PCI_MEM, + .swizzle = no_swizzle, + .map_irq = cats_map_irq, }; diff -urN orig/arch/arm/mach-footbridge/ebsa285-pci.c linux/arch/arm/mach-footbridge/ebsa285-pci.c --- orig/arch/arm/mach-footbridge/ebsa285-pci.c Sat Jul 21 10:46:32 2001 +++ linux/arch/arm/mach-footbridge/ebsa285-pci.c Sun Sep 15 12:57:56 2002 @@ -34,9 +34,9 @@ } struct hw_pci ebsa285_pci __initdata = { - setup_resources: dc21285_setup_resources, - init: dc21285_init, - mem_offset: DC21285_PCI_MEM, - swizzle: ebsa285_swizzle, - map_irq: ebsa285_map_irq, + .setup_resources = dc21285_setup_resources, + .init = dc21285_init, + .mem_offset = DC21285_PCI_MEM, + .swizzle = ebsa285_swizzle, + .map_irq = ebsa285_map_irq, }; diff -urN orig/arch/arm/mach-footbridge/irq.c linux/arch/arm/mach-footbridge/irq.c --- orig/arch/arm/mach-footbridge/irq.c Sat Apr 28 11:24:53 2001 +++ linux/arch/arm/mach-footbridge/irq.c Tue Feb 4 15:08:01 2003 @@ -18,6 +18,8 @@ #include #include #include +#include +#include #include diff -urN orig/arch/arm/mach-footbridge/netwinder-pci.c linux/arch/arm/mach-footbridge/netwinder-pci.c --- orig/arch/arm/mach-footbridge/netwinder-pci.c Sat Jul 21 10:46:32 2001 +++ linux/arch/arm/mach-footbridge/netwinder-pci.c Sun Sep 15 12:57:56 2002 @@ -49,9 +49,9 @@ } struct hw_pci netwinder_pci __initdata = { - setup_resources: dc21285_setup_resources, - init: dc21285_init, - mem_offset: DC21285_PCI_MEM, - swizzle: no_swizzle, - map_irq: netwinder_map_irq, + .setup_resources = dc21285_setup_resources, + .init = dc21285_init, + .mem_offset = DC21285_PCI_MEM, + .swizzle = no_swizzle, + .map_irq = netwinder_map_irq, }; diff -urN orig/arch/arm/mach-footbridge/personal-pci.c linux/arch/arm/mach-footbridge/personal-pci.c --- orig/arch/arm/mach-footbridge/personal-pci.c Sat Jul 21 10:46:32 2001 +++ linux/arch/arm/mach-footbridge/personal-pci.c Sun Sep 15 12:57:56 2002 @@ -38,9 +38,9 @@ } struct hw_pci personal_server_pci __initdata = { - setup_resources: dc21285_setup_resources, - init: dc21285_init, - mem_offset: DC21285_PCI_MEM, - swizzle: no_swizzle, - map_irq: personal_server_map_irq, + .setup_resources = dc21285_setup_resources, + .init = dc21285_init, + .mem_offset = DC21285_PCI_MEM, + .swizzle = no_swizzle, + .map_irq = personal_server_map_irq, }; diff -urN orig/arch/arm/mach-integrator/arch.c linux/arch/arm/mach-integrator/arch.c --- orig/arch/arm/mach-integrator/arch.c Mon Sep 3 14:14:40 2001 +++ linux/arch/arm/mach-integrator/arch.c Sun Sep 15 12:58:40 2002 @@ -36,17 +36,17 @@ #ifdef CONFIG_KMI_KEYB static struct kmi_info integrator_keyboard __initdata = { - base: IO_ADDRESS(KMI0_BASE), - irq: IRQ_KMIINT0, - divisor: 24 / 8 - 1, - type: KMI_KEYBOARD, + .base = IO_ADDRESS(KMI0_BASE), + .irq = IRQ_KMIINT0, + .divisor = 24 / 8 - 1, + .type = KMI_KEYBOARD, }; static struct kmi_info integrator_mouse __initdata = { - base: IO_ADDRESS(KMI1_BASE), - irq: IRQ_KMIINT1, - divisor: 24 / 8 - 1, - type: KMI_MOUSE, + .base = IO_ADDRESS(KMI1_BASE), + .irq = IRQ_KMIINT1, + .divisor = 24 / 8 - 1, + .type = KMI_MOUSE, }; #endif diff -urN orig/arch/arm/mach-integrator/cpu.c linux/arch/arm/mach-integrator/cpu.c --- orig/arch/arm/mach-integrator/cpu.c Fri Oct 26 16:45:53 2001 +++ linux/arch/arm/mach-integrator/cpu.c Mon Jun 17 11:38:10 2002 @@ -3,7 +3,7 @@ * * Copyright (C) 2001 Deep Blue Solutions Ltd. * - * $Id: cpu.c,v 1.2 2001/09/22 12:11:17 rmk Exp $ + * $Id: cpu.c,v 1.2.2.1 2002/05/30 15:08:03 db Exp $ * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as @@ -121,7 +121,7 @@ cpu_freq_khz = vco_to_freq(vco, 1); #ifdef CONFIG_CPU_FREQ - cpufreq_init(cpu_freq_khz); + cpufreq_init(cpu_freq_khz, 1000, 0); cpufreq_setfunctions(integrator_validatespeed, integrator_setspeed); #endif diff -urN orig/arch/arm/mach-integrator/irq.c linux/arch/arm/mach-integrator/irq.c --- orig/arch/arm/mach-integrator/irq.c Sat Mar 31 23:47:15 2001 +++ linux/arch/arm/mach-integrator/irq.c Tue Feb 4 15:08:01 2003 @@ -18,6 +18,8 @@ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include +#include +#include #include #include diff -urN orig/arch/arm/mach-integrator/pci.c linux/arch/arm/mach-integrator/pci.c --- orig/arch/arm/mach-integrator/pci.c Mon Sep 3 14:14:40 2001 +++ linux/arch/arm/mach-integrator/pci.c Sun Sep 15 12:59:00 2002 @@ -114,9 +114,9 @@ extern void pci_v3_init(void *); struct hw_pci integrator_pci __initdata = { - setup_resources: pci_v3_setup_resources, - init: pci_v3_init, - mem_offset: 0x40000000, - swizzle: integrator_swizzle, - map_irq: integrator_map_irq, + .setup_resources = pci_v3_setup_resources, + .init = pci_v3_init, + .mem_offset = 0x40000000, + .swizzle = integrator_swizzle, + .map_irq = integrator_map_irq, }; diff -urN orig/arch/arm/mach-integrator/pci_v3.c linux/arch/arm/mach-integrator/pci_v3.c --- orig/arch/arm/mach-integrator/pci_v3.c Mon Aug 5 13:29:45 2002 +++ linux/arch/arm/mach-integrator/pci_v3.c Sun Sep 15 12:59:54 2002 @@ -390,26 +390,26 @@ } static struct pci_ops pci_v3_ops = { - read_byte: v3_read_config_byte, - read_word: v3_read_config_word, - read_dword: v3_read_config_dword, - write_byte: v3_write_config_byte, - write_word: v3_write_config_word, - write_dword: v3_write_config_dword, + .read_byte = v3_read_config_byte, + .read_word = v3_read_config_word, + .read_dword = v3_read_config_dword, + .write_byte = v3_write_config_byte, + .write_word = v3_write_config_word, + .write_dword = v3_write_config_dword, }; static struct resource non_mem = { - name: "PCI non-prefetchable", - start: 0x40000000 + PCI_BUS_NONMEM_START, - end: 0x40000000 + PCI_BUS_NONMEM_START + PCI_BUS_NONMEM_SIZE - 1, - flags: IORESOURCE_MEM, + .name = "PCI non-prefetchable", + .start = 0x40000000 + PCI_BUS_NONMEM_START, + .end = 0x40000000 + PCI_BUS_NONMEM_START + PCI_BUS_NONMEM_SIZE - 1, + .flags = IORESOURCE_MEM, }; static struct resource pre_mem = { - name: "PCI prefetchable", - start: 0x40000000 + PCI_BUS_PREMEM_START, - end: 0x40000000 + PCI_BUS_PREMEM_START + PCI_BUS_PREMEM_SIZE - 1, - flags: IORESOURCE_MEM | IORESOURCE_PREFETCH, + .name = "PCI prefetchable", + .start = 0x40000000 + PCI_BUS_PREMEM_START, + .end = 0x40000000 + PCI_BUS_PREMEM_START + PCI_BUS_PREMEM_SIZE - 1, + .flags = IORESOURCE_MEM | IORESOURCE_PREFETCH, }; int __init pci_v3_setup_resources(struct resource **resource) diff -urN orig/arch/arm/mach-mx1ads/irq.c linux/arch/arm/mach-mx1ads/irq.c --- orig/arch/arm/mach-mx1ads/irq.c Mon Aug 5 13:29:45 2002 +++ linux/arch/arm/mach-mx1ads/irq.c Tue Feb 4 15:08:01 2003 @@ -19,6 +19,8 @@ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include +#include +#include #include #include diff -urN orig/arch/arm/mach-omaha/Makefile linux/arch/arm/mach-omaha/Makefile --- orig/arch/arm/mach-omaha/Makefile Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-omaha/Makefile Thu Oct 24 14:58:03 2002 @@ -0,0 +1,21 @@ +# +# Makefile for the linux kernel. +# +# Note! Dependencies are done automagically by 'make dep', which also +# removes any old dependencies. DON'T put your own dependencies here +# unless it's something special (ie not a .c file). + +USE_STANDARD_AS_RULE := true + +O_TARGET := omaha.o + +# Object file lists. + +obj-y := core.o +obj-m := +obj-n := +obj- := + +obj-$(CONFIG_LEDS) += leds.o + +include $(TOPDIR)/Rules.make diff -urN orig/arch/arm/mach-omaha/core.c linux/arch/arm/mach-omaha/core.c --- orig/arch/arm/mach-omaha/core.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-omaha/core.c Thu Feb 6 11:30:13 2003 @@ -0,0 +1,130 @@ +/* + * linux/arch/arm/mach-omaha/core.c + * + * Copyright (C) ARM Limited 2002. + * Copyright (C) 2000 Deep Blue Solutions Ltd + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include + +extern void omaha_map_io(void); + +/* + * All IO addresses are mapped onto VA 0xExxx.xxxx, where x.xxxx + * is the (PA + 0xE0000000). + * + * Setup a VA for the Omaha interrupt controller. + */ + +#define VA_IC_BASE IO_ADDRESS(PLAT_PERIPHERAL_BASE) + +static void sc_mask_and_ack_irq(unsigned int irq) +{ + unsigned int tmp; + + // Mask this interrupt + tmp = __raw_readl(VA_IC_BASE + OMAHA_INTMSK); + tmp = tmp | (1 << irq); + __raw_writel(tmp, VA_IC_BASE + OMAHA_INTMSK); + + // Clear the source pending register + tmp = __raw_readl(VA_IC_BASE + OMAHA_SRCPND); + tmp = tmp | (1 << irq); + __raw_writel(tmp, VA_IC_BASE + OMAHA_SRCPND); + + // Clear the interrupt pending register + tmp = __raw_readl(VA_IC_BASE + OMAHA_INTPND); + tmp = tmp | (1 << irq); + __raw_writel(tmp, VA_IC_BASE + OMAHA_INTPND); + +} + +static void sc_mask_irq(unsigned int irq) +{ + unsigned int tmp; + + // Mask this interrupt + tmp = __raw_readl(VA_IC_BASE + OMAHA_INTMSK); + tmp = tmp | (1 << irq); + __raw_writel(tmp, VA_IC_BASE + OMAHA_INTMSK); +} + +static void sc_unmask_irq(unsigned int irq) +{ + unsigned int tmp; + + tmp = __raw_readl(VA_IC_BASE + OMAHA_INTMSK); + tmp = tmp & ~(1 << irq); + __raw_writel(tmp, VA_IC_BASE + OMAHA_INTMSK); +} + +static void __init omaha_init_irq(void) +{ + unsigned int i; + + /* bootloader disables interrupt hardware, + * so we just set up linux data structures... + */ + + for (i = 0; i < NR_IRQS; i++) { + irq_desc[i].valid = 1; + irq_desc[i].probe_ok = 1; + irq_desc[i].mask_ack = sc_mask_and_ack_irq; + irq_desc[i].mask = sc_mask_irq; + irq_desc[i].unmask = sc_unmask_irq; + } +} + +/* Notes + * + * IO space has been mapped into the top of virtual memory at 0xExxx xxxx + * See IO_ACCESS macro for details. + */ + +/* Logical Physical + * Start 0xE0000000 0x00000000 + * End 0xFAFFFFFC 0x1AFFFFFC + */ + +/* Map the bottom 2Gb of IO space into the top of memory, but leave + * space for the high vector table. + */ +static struct map_desc omaha_io_desc[] __initdata = { + { IO_ADDRESS(0x00000000), 0x00000000, 0x1B000000, DOMAIN_IO, 0, 1 }, + LAST_DESC +}; + +static void __init omaha_map_io(void) +{ diff -urN orig/arch/arm/mach-omaha/leds.c linux/arch/arm/mach-omaha/leds.c --- orig/arch/arm/mach-omaha/leds.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-omaha/leds.c Thu Oct 24 14:58:03 2002 @@ -0,0 +1,89 @@ +/* + * linux/arch/arm/mach-omaha/leds.c + * + * Omaha LED control routines + * + * Copyright (C) 1999-2002 ARM Limited + * Copyright (C) 2000 Deep Blue Solutions Ltd + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ +#include +#include + +#include +#include +#include +#include +#include + +static int saved_leds; + +static void omaha_leds_event(led_event_t ledevt) +{ + unsigned long flags; + unsigned int ctrl = IO_ADDRESS(PLAT_DBG_LEDS); + unsigned int leds; + + // yup, change the LEDs + local_irq_save(flags); + + switch(ledevt) { + case led_idle_start: + leds = __raw_readl(ctrl); + leds |= GREEN_LED; + __raw_writel(leds,ctrl); + break; + + case led_idle_end: + leds = __raw_readl(ctrl); + leds &= ~GREEN_LED; + __raw_writel(leds,ctrl); + break; + + case led_timer: + leds = __raw_readl(ctrl); + leds ^= YELLOW_LED; + __raw_writel(leds,ctrl); + break; + + case led_red_on: + leds = __raw_readl(ctrl); + leds |= RED_LED; + __raw_writel(leds,ctrl); + break; + + case led_red_off: + leds = __raw_readl(ctrl); + leds &= ~RED_LED; + __raw_writel(leds,ctrl); + break; + + default: + break; + } + + local_irq_restore(flags); +} + +static int __init leds_init(void) +{ + if (machine_is_omaha()) + leds_event = omaha_leds_event; + + return 0; +} + +__initcall(leds_init); diff -urN orig/arch/arm/mach-sa1100/Makefile linux/arch/arm/mach-sa1100/Makefile --- orig/arch/arm/mach-sa1100/Makefile Mon Aug 5 13:29:46 2002 +++ linux/arch/arm/mach-sa1100/Makefile Fri Feb 21 15:22:26 2003 @@ -15,10 +15,10 @@ obj-n := obj- := -export-objs := assabet.o dma-sa1100.o dma-sa1111.o \ - flexanet.o freebird.o generic.o h3600.o \ +export-objs := assabet.o consus.o badge4.o dma-sa1100.o dma-sa1111.o \ + flexanet.o freebird.o frodo.o generic.o h3600.o \ huw_webpanel.o irq.o pcipool.o sa1111.o sa1111-pcibuf.o \ - system3.o yopy.o usb_ctl.o usb_recv.o usb_send.o + system3.o yopy.o usb_ctl.o usb_recv.o usb_send.o simputer.o ssp.o # These aren't present yet, and prevents a plain -ac kernel building. # hwtimer.o @@ -29,8 +29,12 @@ # We link the CPU support next, so that RAM timings can be tuned. ifeq ($(CONFIG_CPU_FREQ),y) obj-$(CONFIG_SA1100_ASSABET) += cpu-sa1110.o +obj-$(CONFIG_SA1100_CEP) += cpu-sa1110.o +obj-$(CONFIG_SA1100_CONSUS) += cpu-sa1110.o obj-$(CONFIG_SA1100_CERF) += cpu-sa1110.o +obj-$(CONFIG_SA1100_HACKKIT) += cpu-sa1110.o obj-$(CONFIG_SA1100_PT_SYSTEM3) += cpu-sa1110.o +obj-$(CONFIG_SA1100_SIMPUTER) += cpu-sa1110.o obj-$(CONFIG_SA1100_LART) += cpu-sa1100.o endif @@ -39,12 +43,16 @@ obj-$(CONFIG_USB_OHCI_SA1111) += sa1111-pcibuf.o pcipool.o # Specific board support +obj-$(CONFIG_SA1100_ADSAGC) += adsagc.o obj-$(CONFIG_SA1100_ADSBITSY) += adsbitsy.o +obj-$(CONFIG_SA1100_ADSBITSYPLUS) += adsbitsyplus.o obj-$(CONFIG_SA1100_ASSABET) += assabet.o obj-$(CONFIG_ASSABET_NEPONSET) += neponset.o obj-$(CONFIG_SA1100_BADGE4) += badge4.o obj-$(CONFIG_SA1100_BRUTUS) += brutus.o +obj-$(CONFIG_SA1100_CEP) += cep.o obj-$(CONFIG_SA1100_CERF) += cerf.o +obj-$(CONFIG_SA1100_CONSUS) += consus.o obj-$(CONFIG_SA1100_EMPEG) += empeg.o obj-$(CONFIG_SA1100_FLEXANET) += flexanet.o obj-$(CONFIG_SA1100_FREEBIRD) += freebird.o @@ -52,6 +60,7 @@ obj-$(CONFIG_SA1100_GRAPHICSCLIENT) += graphicsclient.o obj-$(CONFIG_SA1100_GRAPHICSMASTER) += graphicsmaster.o obj-$(CONFIG_SA1100_H3600) += h3600.o +obj-$(CONFIG_SA1100_HACKKIT) += hackkit.o obj-$(CONFIG_SA1100_HUW_WEBPANEL) += huw_webpanel.o obj-$(CONFIG_SA1100_ITSY) += itsy.o obj-$(CONFIG_SA1100_JORNADA720) += jornada720.o @@ -65,20 +74,25 @@ obj-$(CONFIG_SA1100_SHANNON) += shannon.o obj-$(CONFIG_SA1100_SHERMAN) += sherman.o obj-$(CONFIG_SA1100_SIMPAD) += simpad.o +obj-$(CONFIG_SA1100_SIMPUTER) += simputer.o obj-$(CONFIG_SA1100_VICTOR) += victor.o obj-$(CONFIG_SA1100_XP860) += xp860.o obj-$(CONFIG_SA1100_YOPY) += yopy.o # LEDs support leds-y := leds.o +leds-$(CONFIG_SA1100_ADSAGC) += leds-adsagc.o leds-$(CONFIG_SA1100_ADSBITSY) += leds-adsbitsy.o +leds-$(CONFIG_SA1100_ADSBITSYPLUS) += leds-adsbitsyplus.o leds-$(CONFIG_SA1100_ASSABET) += leds-assabet.o leds-$(CONFIG_SA1100_BRUTUS) += leds-brutus.o leds-$(CONFIG_SA1100_CERF) += leds-cerf.o +leds-$(CONFIG_SA1100_CONSUS) += leds-consus.o leds-$(CONFIG_SA1100_FLEXANET) += leds-flexanet.o leds-$(CONFIG_SA1100_FRODO) += leds-frodo.o leds-$(CONFIG_SA1100_GRAPHICSCLIENT) += leds-graphicsclient.o leds-$(CONFIG_SA1100_GRAPHICSMASTER) += leds-graphicsmaster.o +leds-$(CONFIG_SA1100_HACKKIT) += leds-hackkit.o leds-$(CONFIG_SA1100_LART) += leds-lart.o leds-$(CONFIG_SA1100_PFS168) += leds-pfs168.o leds-$(CONFIG_SA1100_SIMPAD) += leds-simpad.o @@ -94,6 +108,8 @@ # Miscelaneous functions obj-$(CONFIG_PM) += pm.o sleep.o + +obj-$(CONFIG_SA1100_SSP) += ssp.o include $(TOPDIR)/Rules.make diff -urN orig/arch/arm/mach-sa1100/adsagc.c linux/arch/arm/mach-sa1100/adsagc.c --- orig/arch/arm/mach-sa1100/adsagc.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-sa1100/adsagc.c Fri Feb 21 15:17:34 2003 @@ -0,0 +1,360 @@ +/* + * linux/arch/arm/mach-sa1100/adsagc.c + * + * Pieces specific to the ADS Advanced Graphics Client board + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + */ + +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include + +#include +#include +#include +#include + +#include + +#include "generic.h" +#include "sa1111.h" + +static int __init adsagc_init(void) +{ + int ret; + + if (!machine_is_adsagc()) + return -ENODEV; + + /* + * Ensure that the memory bus request/grant signals are setup, + * and the grant is held in its inactive state + */ + sa1110_mb_disable(); + + /* Adsagc uses GPIO pins for SPI interface to AVR + * AGC uses the standard pins instead. AGC also has controls + * SA1111 and Smartio reset with GPIO. + */ + + // Set RTS low during sleep + PGSR |= GPIO_GPIO15 | GPIO_GPIO17 | GPIO_GPIO19; + + // Reset SA1111 + GPDR |= GPIO_GPIO12; + GPCR = GPIO_GPIO12; + udelay(1000); + GPSR = GPIO_GPIO12; + + /* use the regular SSP pins */ + PPAR &= ~PPAR_SSPGPIO; + /* + * Probe for SA1111. + */ + ret = sa1111_probe(ADS_SA1111_BASE); + if (ret < 0) + return ret; + + /* + * We found it. Wake the chip up. + */ + sa1111_wake(); + + /* + * The SDRAM configuration of the SA1110 and the SA1111 must + * match. This is very important to ensure that SA1111 accesses + * don't corrupt the SDRAM. Note that this ungates the SA1111's + * MBGNT signal, so we must have called sa1110_mb_disable() + * beforehand. + */ + sa1111_configure_smc(1, + FExtr(MDCNFG, MDCNFG_SA1110_DRAC0), + FExtr(MDCNFG, MDCNFG_SA1110_TDL0)); + + /* + * Enable PWM control for LCD + */ + SKPCR |= SKPCR_PWMCLKEN; + SACR1 &= ~SACR1_L3EN; + ADS_CR3 |= ADS_CR3_BLON; // Enable the backlight + SKPWM1 = 0x7F; // Backlight PWM + SKPEN1 = 1; + SKPWM0 = 0x7F; // VEE + SKPEN0 = 1; + + /* + * We only need to turn on DCLK whenever we want to use the + * DMA. It can otherwise be held firmly in the off position. + */ + SKPCR |= SKPCR_DCLKEN; + + /* + * Enable the SA1110 memory bus request and grant signals. + */ + sa1110_mb_enable(); + + set_GPIO_IRQ_edge(GPIO_GPIO0, GPIO_RISING_EDGE); + sa1111_init_irq(IRQ_GPIO0); + + return 0; +} + +__initcall(adsagc_init); + +/* + * Handlers for Adsagc's external IRQ logic + */ + +#define ADSAGC_N_IRQ (IRQ_ADSAGC_END - IRQ_ADSAGC_START) + +static void ADS_IRQ_demux( int irq, void *dev_id, struct pt_regs *regs ) +{ + int i; + + while( (irq = ADS_INT_ST1 ) ){ + for( i = 0; i < ADSAGC_N_IRQ; i++ ) + if( irq & (1<mapbase == _Ser1UTCR0) { + Ser1SDCR0 |= SDCR0_UART; + } + else if (port->mapbase == _Ser2UTCR0) { + Ser2UTCR4 = Ser2HSCR0 = 0; + } + return ret; +} + +static u_int adsagc_get_mctrl(struct uart_port *port) +{ + u_int result = TIOCM_CD | TIOCM_DSR; + + if (port->mapbase == _Ser1UTCR0) { + if (!(GPLR & GPIO_GPIO14)) + result |= TIOCM_CTS; + } else if (port->mapbase == _Ser2UTCR0) { + if (!(GPLR & GPIO_GPIO16)) + result |= TIOCM_CTS; + } else if (port->mapbase == _Ser3UTCR0) { + if (!(GPLR & GPIO_GPIO18)) + result |= TIOCM_CTS; + } else { + result = TIOCM_CTS; + } + + return result; +} + +static void adsagc_set_mctrl(struct uart_port *port, u_int mctrl) +{ + if (port->mapbase == _Ser1UTCR0) { + if (mctrl & TIOCM_RTS) + GPCR = GPIO_GPIO15; + else + GPSR = GPIO_GPIO15; + } else if (port->mapbase == _Ser2UTCR0) { + if (mctrl & TIOCM_RTS) + GPCR = GPIO_GPIO17; + else + GPSR = GPIO_GPIO17; + } else if (port->mapbase == _Ser3UTCR0) { + if (mctrl & TIOCM_RTS) + GPCR = GPIO_GPIO19; + else + GPSR = GPIO_GPIO19; + } +} + +static void +adsagc_uart_pm(struct uart_port *port, u_int state, u_int oldstate) +{ + // state has ACPI D0-D3 + // ACPI D0 : resume from suspend + // ACPI D1-D3 : enter to a suspend state + if (port->mapbase == _Ser1UTCR0) { + if (state) { + // disable uart + Ser1UTCR3 = 0; + } + } + else if (port->mapbase == _Ser2UTCR0) { + if (state) { + // disable uart + Ser2UTCR3 = 0; + Ser2HSCR0 = 0; + } + } + else if (port->mapbase == _Ser3UTCR0) { + if (state) { + // disable uart + Ser3UTCR3 = 0; + } + } +} + +static struct sa1100_port_fns adsagc_port_fns __initdata = { + .open = adsagc_uart_open, + .get_mctrl = adsagc_get_mctrl, + .set_mctrl = adsagc_set_mctrl, + .pm = adsagc_uart_pm, +}; + +static void __init adsagc_map_io(void) +{ + sa1100_map_io(); + iotable_init(adsagc_io_desc); + + sa1100_register_uart_fns(&adsagc_port_fns); + sa1100_register_uart(0, 3); + sa1100_register_uart(1, 1); + // don't register if you want to use IRDA +#ifndef CONFIG_SA1100_FIR + sa1100_register_uart(2, 2); +#endif + + /* set GPDR now */ + GPDR |= GPIO_GPIO15 | GPIO_GPIO17 | GPIO_GPIO19; + GPDR &= ~(GPIO_GPIO14 | GPIO_GPIO16 | GPIO_GPIO18); +} + +MACHINE_START(ADSAGC, "ADS Advanced GraphicsClient") + BOOT_PARAMS(0xc000003c) + BOOT_MEM(0xc0000000, 0x80000000, 0xf8000000) + MAPIO(adsagc_map_io) + INITIRQ(adsagc_init_irq) +MACHINE_END diff -urN orig/arch/arm/mach-sa1100/adsbitsy.c linux/arch/arm/mach-sa1100/adsbitsy.c --- orig/arch/arm/mach-sa1100/adsbitsy.c Mon Aug 5 13:29:46 2002 +++ linux/arch/arm/mach-sa1100/adsbitsy.c Thu Feb 27 23:20:10 2003 @@ -14,9 +14,13 @@ #include #include #include +#include #include +#include +#include #include +#include #include #include @@ -30,6 +34,7 @@ #include "generic.h" #include "sa1111.h" + static int __init adsbitsy_init(void) { int ret; @@ -43,6 +48,13 @@ */ sa1110_mb_disable(); + /* Bitsy uses GPIO pins for SPI interface to AVR + * Bitsy Plus uses the standard pins instead. + * it also needs to reset the AVR when booting + */ + + PPAR |= PPAR_SSPGPIO; + /* * Reset SA1111 */ @@ -50,6 +62,19 @@ udelay(1000); GPSR |= GPIO_GPIO26; + +#ifndef CONFIG_LEDS_TIMER + // Set Serial port 1 RTS and DTR Low during sleep + PGSR |= GPIO_GPIO15 | GPIO_GPIO20; +#else + // only RTS (because DTR is also the LED + // which should be off during sleep); + PGSR |= GPIO_GPIO15; +#endif + + // Set Serial port 3RTS Low during sleep + PGSR |= GPIO_GPIO19; + /* * Probe for SA1111. */ @@ -74,15 +99,6 @@ FExtr(MDCNFG, MDCNFG_SA1110_TDL0)); /* - * Enable PWM control for LCD - */ - SKPCR |= SKPCR_PWMCLKEN; - SKPWM0 = 0x7F; // VEE - SKPEN0 = 1; - SKPWM1 = 0x01; // Backlight - SKPEN1 = 1; - - /* * We only need to turn on DCLK whenever we want to use the * DMA. It can otherwise be held firmly in the off position. */ @@ -108,48 +124,135 @@ } -/* - * Initialization fixup - */ - -static void __init -fixup_adsbitsy(struct machine_desc *desc, struct param_struct *params, - char **cmdline, struct meminfo *mi) -{ - SET_BANK( 0, 0xc0000000, 32*1024*1024 ); - mi->nr_banks = 1; - - ROOT_DEV = MKDEV(RAMDISK_MAJOR,0); - setup_ramdisk( 1, 0, 0, 8192 ); - setup_initrd( __phys_to_virt(0xc0800000), 4*1024*1024 ); -} - static struct map_desc adsbitsy_io_desc[] __initdata = { /* virtual physical length domain r w c b */ - { 0xe8000000, 0x08000000, 0x01000000, DOMAIN_IO, 0, 1, 0, 0 }, /* Flash bank 1 */ + { 0xe8000000, 0x08000000, 0x02000000, DOMAIN_IO, 0, 1, 0, 0 }, /* Flash bank 1 */ + { 0xf0000000, 0x3C000000, 0x00004000, DOMAIN_IO, 0, 1, 0, 0 }, /* 91C1111 */ { 0xf4000000, 0x18000000, 0x00800000, DOMAIN_IO, 0, 1, 0, 0 }, /* SA1111 */ LAST_DESC }; +/* Use this to see when all uarts are shutdown. Or all are closed. + * We can only turn off RS232 chip if either of these are true. + */ + +static int uart_wake_count[3] = {1, 1, 1}; + +enum {UART_SHUTDOWN, UART_WAKEUP}; + +static void update_uart_counts(int line, int state) +{ + switch (state) { + case UART_WAKEUP: + uart_wake_count[line]++; + break; + case UART_SHUTDOWN: + uart_wake_count[line]--; + break; + } +} + static int adsbitsy_uart_open(struct uart_port *port, struct uart_info *info) { if (port->mapbase == _Ser1UTCR0) { Ser1SDCR0 |= SDCR0_UART; -#error Fixme // Set RTS High (should be done in the set_mctrl fn) - GPCR = GPIO_GPIO15; } else if (port->mapbase == _Ser2UTCR0) { Ser2UTCR4 = Ser2HSCR0 = 0; -#error Fixme // Set RTS High (should be done in the set_mctrl fn) - GPCR = GPIO_GPIO17; - } else if (port->mapbase == _Ser2UTCR0) { -#error Fixme // Set RTS High (should be done in the set_mctrl fn) - GPCR = GPIO_GPIO19; } return 0; } +void adsbitsy_uart_pm(struct uart_port *port, u_int state, u_int oldstate) +{ + // state has ACPI D0-D3 + // ACPI D0 : resume from suspend + // ACPI D1-D3 : enter to a suspend state + if (port->mapbase == _Ser1UTCR0) { + if (state) { + update_uart_counts(1, UART_SHUTDOWN); + // disable uart + Ser1UTCR3 = 0; + } + else { + update_uart_counts(1, UART_WAKEUP); + } + } + else if (port->mapbase == _Ser2UTCR0) { + if (state) { + update_uart_counts(2, UART_SHUTDOWN); + // disable uart + Ser2UTCR3 = 0; + Ser2HSCR0 = 0; + } + else { + update_uart_counts(2, UART_WAKEUP); + } + } + else if (port->mapbase == _Ser3UTCR0) { + if (state) { + update_uart_counts(0, UART_SHUTDOWN); + // disable uart + Ser3UTCR3 = 0; + } + else { + update_uart_counts(0, UART_WAKEUP); + } + } +} + +static void adsbitsy_set_mctrl(struct uart_port *port, u_int mctrl) +{ + // note: only ports 1 and 3 have modem control + if (port->mapbase == _Ser1UTCR0) { + if (mctrl & TIOCM_RTS) + // Set RTS High + GPCR = GPIO_GPIO15; + else + // Set RTS LOW + GPSR = GPIO_GPIO15; + if (mctrl & TIOCM_DTR) + // Set DTR High + GPCR = GPIO_GPIO20; + else + // Set DTR Low + GPSR = GPIO_GPIO20; + } else if (port->mapbase == _Ser3UTCR0) { + if (mctrl & TIOCM_RTS) + // Set RTS High + GPCR = GPIO_GPIO19; + else + // Set RTS LOW + GPSR = GPIO_GPIO19; + } +} + +static u_int adsbitsy_get_mctrl(struct uart_port *port) +{ + u_int ret = 0; + + // note: only ports 1 and 3 have modem control + if (port->mapbase == _Ser1UTCR0) { + if (!(GPLR & GPIO_GPIO14)) + ret |= TIOCM_CTS; + if (!(GPLR & GPIO_GPIO24)) + ret |= TIOCM_DSR; + if (!(GPLR & GPIO_GPIO16)) + ret |= TIOCM_RI; + if (!(GPLR & GPIO_GPIO17)) + ret |= TIOCM_CD; + } else if (port->mapbase == _Ser3UTCR0) { + if (!(GPLR & GPIO_GPIO18)) + ret |= TIOCM_CTS; + } + + return ret; +} + static struct sa1100_port_fns adsbitsy_port_fns __initdata = { - open: adsbitsy_uart_open, + .set_mctrl = adsbitsy_set_mctrl, + .get_mctrl = adsbitsy_get_mctrl, + .open = adsbitsy_uart_open, + .pm = adsbitsy_uart_pm, }; static void __init adsbitsy_map_io(void) @@ -160,14 +263,27 @@ sa1100_register_uart_fns(&adsbitsy_port_fns); sa1100_register_uart(0, 3); sa1100_register_uart(1, 1); + + // don't register if you want to use IRDA +#ifndef CONFIG_SA1100_FIR sa1100_register_uart(2, 2); - GPDR |= GPIO_GPIO15 | GPIO_GPIO17 | GPIO_GPIO19; - GPDR &= ~(GPIO_GPIO14 | GPIO_GPIO16 | GPIO_GPIO18); +#endif + + // COM1 Set RTS and DTR Output + GPDR |= GPIO_GPIO15 | GPIO_GPIO20; + // Set CTS, DSR, RI and CD Input + GPDR &= ~(GPIO_GPIO14 | GPIO_GPIO24 | GPIO_GPIO16 | GPIO_GPIO17); + + // COM3 Set RTS Output + GPDR |= GPIO_GPIO19; + // Set CTS Input + GPDR &= ~GPIO_GPIO18; } + MACHINE_START(ADSBITSY, "ADS Bitsy") + BOOT_PARAMS(0xc000003c) BOOT_MEM(0xc0000000, 0x80000000, 0xf8000000) - FIXUP(fixup_adsbitsy) MAPIO(adsbitsy_map_io) INITIRQ(adsbitsy_init_irq) MACHINE_END diff -urN orig/arch/arm/mach-sa1100/adsbitsyplus.c linux/arch/arm/mach-sa1100/adsbitsyplus.c --- orig/arch/arm/mach-sa1100/adsbitsyplus.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-sa1100/adsbitsyplus.c Fri Feb 21 15:17:33 2003 @@ -0,0 +1,395 @@ +/* + * linux/arch/arm/mach-sa1100/adsbitsyplus.c + * + * Author: Robert Whaley + * + * This file comes from adsbitsy.c of Woojung Huh + * + * Pieces specific to the ADS Bitsy Plus + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + */ + +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include + +#include +#include +#include +#include + +#include + +#include "generic.h" +#include "sa1111.h" + +/* unfortunately, we can't detect the difference between REV 2 and REV A + connector boards. So by convention, the registers.txt file is used + to set a byte to either 0x02 or 0x0a to make this distinction. The + bootloader must detect this by default we assume rev A since most boards + will be rev A */ + +static int adsbitsyplus_connector_board_rev_number = 0xA; + +static int __init adsbitsyplus_connector_board_rev_setup(char *str) +{ + adsbitsyplus_connector_board_rev_number = simple_strtol(str,NULL,0); + return 1; +} + +int adsbitsyplus_connector_board_rev(void) +{ + static int only_once = 1; + if (only_once) { + printk(KERN_INFO "Bitsy Connector Board REV: %#x\n", adsbitsyplus_connector_board_rev_number); + only_once = 0; + } + return adsbitsyplus_connector_board_rev_number; +} + +static int __init adsbitsyplus_init(void) +{ + int ret; + + if (!machine_is_adsbitsyplus()) + return -ENODEV; + + /* + * Ensure that the memory bus request/grant signals are setup, + * and the grant is held in its inactive state + */ + sa1110_mb_disable(); + + /* Bitsy uses GPIO pins for SPI interface to AVR + * Bitsy Plus uses the standard pins instead. + * it also needs to reset the AVR when booting + */ + + PPAR &= ~PPAR_SSPGPIO; + ADS_CPLD_SUPPC |= ADS_SUPPC_AVR_WKP; + mdelay(100); + ADS_CPLD_SUPPC &= ~ADS_SUPPC_AVR_WKP; + + /* + * Reset SA1111 + */ + GPCR |= GPIO_GPIO26; + udelay(1000); + GPSR |= GPIO_GPIO26; + +#ifndef CONFIG_LEDS_TIMER + // Set Serial port 1 RTS and DTR Low during sleep + PGSR |= GPIO_GPIO15 | GPIO_GPIO20; +#else + // only RTS (because DTR is also the LED + // which should be off during sleep); + PGSR |= GPIO_GPIO15; +#endif + + // Set Serial port 3RTS Low during sleep + PGSR |= GPIO_GPIO19; + + /* + * Probe for SA1111. + */ + ret = sa1111_probe(ADSBITSYPLUS_SA1111_BASE); + if (ret < 0) + return ret; + + /* + * We found it. Wake the chip up. + */ + sa1111_wake(); + + /* + * The SDRAM configuration of the SA1110 and the SA1111 must + * match. This is very important to ensure that SA1111 accesses + * don't corrupt the SDRAM. Note that this ungates the SA1111's + * MBGNT signal, so we must have called sa1110_mb_disable() + * beforehand. + */ + sa1111_configure_smc(1, + FExtr(MDCNFG, MDCNFG_SA1110_DRAC0), + FExtr(MDCNFG, MDCNFG_SA1110_TDL0)); + + /* + * We only need to turn on DCLK whenever we want to use the + * DMA. It can otherwise be held firmly in the off position. + */ + SKPCR |= SKPCR_DCLKEN; + + /* + * Enable the SA1110 memory bus request and grant signals. + */ + sa1110_mb_enable(); + + set_GPIO_IRQ_edge(GPIO_GPIO0, GPIO_RISING_EDGE); + sa1111_init_irq(IRQ_GPIO0); + + return 0; +} + +__initcall(adsbitsyplus_init); + +static void __init adsbitsyplus_init_irq(void) +{ + /* First the standard SA1100 IRQs */ + sa1100_init_irq(); +} + +/* + * Resume SA1111 when system wakes up + */ +void adsbitsyplus_sa1111_wake(unsigned long pa_dwr) +{ + // Turn ON SA1111 + GPCR |= GPIO_GPIO26; + mdelay(1); + GPSR |= GPIO_GPIO26; + + GAFR |= GPIO_32_768kHz; + GPDR |= GPIO_32_768kHz; + TUCR = TUCR_3_6864MHz; + + SBI_SKCR = SKCR_PLL_BYPASS | SKCR_RDYEN | SKCR_OE_EN; + udelay(100); + SBI_SKCR = SKCR_PLL_BYPASS | SKCR_RCLKEN | SKCR_RDYEN | SKCR_OE_EN; + + GAFR |= (GPIO_MBGNT | GPIO_MBREQ); + GPDR |= GPIO_MBGNT; + GPDR &= ~GPIO_MBREQ; + TUCR |= TUCR_MR; + + sa1111_configure_smc(1, + FExtr(MDCNFG, MDCNFG_SA1110_DRAC0), + FExtr(MDCNFG, MDCNFG_SA1110_TDL0)); + SKPCR |= SKPCR_DCLKEN; + + // Reset PCMCIA + PCCR = 0xFF; + mdelay(100); + PA_DDR = 0x00; + // PA_DWR = GPIO_GPIO0 | GPIO_GPIO1 | GPIO_GPIO2 | GPIO_GPIO3; + // PA_DWR = GPIO_GPIO0 | GPIO_GPIO2 | GPIO_GPIO3; + PA_DWR = pa_dwr; + PCCR = ~(PCCR_S0_RST | PCCR_S1_RST); + +#ifdef CONFIG_USB_OHCI_SA1111 + // Turn ON clock + SKPCR |= SKPCR_UCLKEN; + udelay(100); + + // force a RESET + USB_RESET = 0x01; + USB_RESET |= 0x02; + udelay(100); + + // Set Power Sense and Control Line + USB_RESET = 0; + USB_RESET = USB_RESET_PWRSENSELOW; + USB_STATUS = 0; + udelay(10); +#endif +} + + +static struct map_desc adsbitsyplus_io_desc[] __initdata = { + /* virtual physical length domain r w c b */ + { 0xe8000000, 0x08000000, 0x02000000, DOMAIN_IO, 0, 1, 0, 0 }, /* Flash bank 1 */ + { 0xf0000000, 0x3C000000, 0x00004000, DOMAIN_IO, 0, 1, 0, 0 }, /* 91C1111 */ + { 0xf4000000, 0x18000000, 0x00800000, DOMAIN_IO, 0, 1, 0, 0 }, /* SA1111 */ + { 0xf1000000, 0x10000000, 0x00001000, DOMAIN_IO, 0, 1, 0, 0 }, /* CPLD Controller */ + LAST_DESC +}; + +/* Use this to see when all uarts are shutdown. Or all are closed. + * We can only turn off RS232 chip if either of these are true. + */ + +static int uart_wake_count[3] = {1, 1, 1}; + +enum {UART_SHUTDOWN, UART_WAKEUP}; + +static void update_uart_counts(int line, int state) +{ + switch (state) { + case UART_WAKEUP: + uart_wake_count[line]++; + break; + case UART_SHUTDOWN: + uart_wake_count[line]--; + break; + } +} + +static int adsbitsyplus_uart_open(struct uart_port *port, struct uart_info *info) +{ + if (port->mapbase == _Ser1UTCR0) { + Ser1SDCR0 |= SDCR0_UART; + } else if (port->mapbase == _Ser2UTCR0) { + Ser2UTCR4 = Ser2HSCR0 = 0; + } + return 0; +} + +void adsbitsyplus_uart_pm(struct uart_port *port, u_int state, u_int oldstate) +{ + // state has ACPI D0-D3 + // ACPI D0 : resume from suspend + // ACPI D1-D3 : enter to a suspend state + if (port->mapbase == _Ser1UTCR0) { + if (state) { + update_uart_counts(1, UART_SHUTDOWN); + // disable uart + Ser1UTCR3 = 0; + } + else { + update_uart_counts(1, UART_WAKEUP); + } + } + else if (port->mapbase == _Ser2UTCR0) { + if (state) { + update_uart_counts(2, UART_SHUTDOWN); + // disable uart + Ser2UTCR3 = 0; + Ser2HSCR0 = 0; + } + else { + update_uart_counts(2, UART_WAKEUP); + } + } + else if (port->mapbase == _Ser3UTCR0) { + if (state) { + update_uart_counts(0, UART_SHUTDOWN); + // disable uart + Ser3UTCR3 = 0; + } + else { + update_uart_counts(0, UART_WAKEUP); + } + } + if (state == 0) { + // Turn power on if uarts are awake + if (uart_wake_count[0] + uart_wake_count[1] != 0) { + // make sure RS-232 is turned on if 1 or 3 are open + ADS_CPLD_PCON |= ADS_PCON_COM1_3_ON; + } + if (uart_wake_count[2] != 0) { + if (adsbitsyplus_connector_board_rev() >= 0x0a) + ADS_CPLD_PCON |= ADS_PCON_CONN_B_PE2; + } + } + else { + // Turn power off if uarts are asleep + if (uart_wake_count[0] + uart_wake_count[1] == 0) { + // save power if we are sleeping + ADS_CPLD_PCON &= ~ADS_PCON_COM1_3_ON; + GAFR &= ~(GPIO_GPIO15 | GPIO_GPIO19 | GPIO_GPIO20); + GPDR |= GPIO_GPIO15 | GPIO_GPIO19 | GPIO_GPIO20; + } + if (uart_wake_count[2] == 0) { + if (adsbitsyplus_connector_board_rev() >= 0x0a) + ADS_CPLD_PCON &= ~ADS_PCON_CONN_B_PE2; + } + } +} + +static void adsbitsyplus_set_mctrl(struct uart_port *port, u_int mctrl) +{ + // note: only ports 1 and 3 have modem control + if (port->mapbase == _Ser1UTCR0) { + if (mctrl & TIOCM_RTS) + // Set RTS High + GPCR = GPIO_GPIO15; + else + // Set RTS LOW + GPSR = GPIO_GPIO15; + if (mctrl & TIOCM_DTR) + // Set DTR High + GPCR = GPIO_GPIO20; + else + // Set DTR Low + GPSR = GPIO_GPIO20; + } else if (port->mapbase == _Ser3UTCR0) { + if (mctrl & TIOCM_RTS) + // Set RTS High + GPCR = GPIO_GPIO19; + else + // Set RTS LOW + GPSR = GPIO_GPIO19; + } +} + +static u_int adsbitsyplus_get_mctrl(struct uart_port *port) +{ + u_int ret = 0; + + // note: only ports 1 and 3 have modem control + if (port->mapbase == _Ser1UTCR0) { + if (!(GPLR & GPIO_GPIO14)) + ret |= TIOCM_CTS; + if (!(GPLR & GPIO_GPIO24)) + ret |= TIOCM_DSR; + if (!(GPLR & GPIO_GPIO16)) + ret |= TIOCM_RI; + if (!(GPLR & GPIO_GPIO17)) + ret |= TIOCM_CD; + } else if (port->mapbase == _Ser3UTCR0) { + if (!(GPLR & GPIO_GPIO18)) + ret |= TIOCM_CTS; + } + + return ret; +} + +static struct sa1100_port_fns adsbitsyplus_port_fns __initdata = { + .set_mctrl = adsbitsyplus_set_mctrl, + .get_mctrl = adsbitsyplus_get_mctrl, + .open = adsbitsyplus_uart_open, + .pm = adsbitsyplus_uart_pm, +}; + +static void __init adsbitsyplus_map_io(void) +{ + sa1100_map_io(); + iotable_init(adsbitsyplus_io_desc); + + sa1100_register_uart_fns(&adsbitsyplus_port_fns); + sa1100_register_uart(0, 3); + sa1100_register_uart(1, 1); + + // don't register if you want to use IRDA +#ifndef CONFIG_SA1100_FIR + sa1100_register_uart(2, 2); +#endif + + // COM1 Set RTS and DTR Output + GPDR |= GPIO_GPIO15 | GPIO_GPIO20; + // Set CTS, DSR, RI and CD Input + GPDR &= ~(GPIO_GPIO14 | GPIO_GPIO24 | GPIO_GPIO16 | GPIO_GPIO17); + + // COM3 Set RTS Output + GPDR |= GPIO_GPIO19; + // Set CTS Input + GPDR &= ~GPIO_GPIO18; +} + +__setup("adsbitsyplus_conn_board_rev=", adsbitsyplus_connector_board_rev_setup); + +MACHINE_START(ADSBITSYPLUS, "ADS Bitsy Plus") + BOOT_PARAMS(0xc000003c) + BOOT_MEM(0xc0000000, 0x80000000, 0xf8000000) + MAPIO(adsbitsyplus_map_io) + INITIRQ(adsbitsyplus_init_irq) +MACHINE_END diff -urN orig/arch/arm/mach-sa1100/assabet.c linux/arch/arm/mach-sa1100/assabet.c --- orig/arch/arm/mach-sa1100/assabet.c Mon Aug 5 13:29:46 2002 +++ linux/arch/arm/mach-sa1100/assabet.c Sun Sep 15 13:00:50 2002 @@ -42,7 +42,8 @@ ASSABET_BCR_LED_GREEN | ASSABET_BCR_LED_RED | \ ASSABET_BCR_RS232EN | ASSABET_BCR_LCD_12RGB | \ ASSABET_BCR_CF_BUS_OFF | ASSABET_BCR_STEREO_LB | \ - ASSABET_BCR_IRDA_MD0 | ASSABET_BCR_CF_RST) + ASSABET_BCR_IRDA_MD0 | ASSABET_BCR_CF_RST | \ + ASSABET_BCR_CODEC_RST ) unsigned long SCR_value = ASSABET_SCR_INIT; EXPORT_SYMBOL(SCR_value); @@ -285,9 +286,9 @@ } static struct sa1100_port_fns assabet_port_fns __initdata = { - set_mctrl: assabet_set_mctrl, - get_mctrl: assabet_get_mctrl, - pm: assabet_uart_pm, + .set_mctrl = assabet_set_mctrl, + .get_mctrl = assabet_get_mctrl, + .pm = assabet_uart_pm, }; static void __init assabet_map_io(void) diff -urN orig/arch/arm/mach-sa1100/badge4.c linux/arch/arm/mach-sa1100/badge4.c --- orig/arch/arm/mach-sa1100/badge4.c Mon Aug 5 13:29:46 2002 +++ linux/arch/arm/mach-sa1100/badge4.c Sat May 18 21:17:48 2002 @@ -14,6 +14,7 @@ * */ +#include #include #include #include diff -urN orig/arch/arm/mach-sa1100/cep.c linux/arch/arm/mach-sa1100/cep.c --- orig/arch/arm/mach-sa1100/cep.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-sa1100/cep.c Sun Jun 16 12:54:24 2002 @@ -0,0 +1,123 @@ +/* + * linux/arch/arm/mach-sa1100/cep.c + * + * Author: Matthias Gorjup + * + * This file contains all Cep - specific tweaks. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + */ + +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include + +#include +#include +#include + +#include "generic.h" + + +extern void convert_to_tag_list(struct param_struct *params, int mem_init); + +static void __init +fixup_cep(struct machine_desc *desc, struct param_struct *params, + char **cmdline, struct meminfo *mi) +{ + struct tag *t = (struct tag *)params; + + + /* + * Apparantly bootldr uses a param_struct. Groan. + */ + if (t->hdr.tag != ATAG_CORE) + convert_to_tag_list(params, 1); + + if (t->hdr.tag != ATAG_CORE) { + t->hdr.tag = ATAG_CORE; + t->hdr.size = tag_size(tag_core); + t->u.core.flags = 0; + t->u.core.pagesize = PAGE_SIZE; + t->u.core.rootdev = RAMDISK_MAJOR << 8 | 0; + t = tag_next(t); + + t->hdr.tag = ATAG_MEM; + t->hdr.size = tag_size(tag_mem32); + t->u.mem.start = 0xc0000000; + t->u.mem.size = 32 * 1024 * 1024; + t = tag_next(t); + + + t->hdr.tag = ATAG_RAMDISK; + t->hdr.size = tag_size(tag_ramdisk); + t->u.ramdisk.flags = 1; + t->u.ramdisk.size = 8192; + t->u.ramdisk.start = 0; + t = tag_next(t); + + t->hdr.tag = ATAG_INITRD; + t->hdr.size = tag_size(tag_initrd); + t->u.initrd.start = 0xc0800000; + t->u.initrd.size = 3 * 1024 * 1024; + t = tag_next(t); + + t->hdr.tag = ATAG_NONE; + t->hdr.size = 0; + } +} + + +static struct map_desc cep_io_desc[] __initdata = { + /* virtual physical length domain r w c b */ + { 0xe8000000, 0x00000000, 0x02000000, DOMAIN_IO, 0, 1, 0, 0 }, /* Flash bank 0 */ + LAST_DESC +}; + +static void __init cep_map_io(void) +{ + + sa1100_map_io(); + iotable_init(cep_io_desc); + + sa1100_register_uart(0, 1); /* com port */ + sa1100_register_uart(2, 3); /* radio module */ + + /* + * Ensure that these pins are set as outputs and are driving + * logic 0. This ensures that we won't inadvertently toggle + * the WS latch in the CPLD, and we don't float causing + * excessive power drain. --rmk + */ + GPDR |= GPIO_SSP_TXD | GPIO_SSP_SCLK | GPIO_SSP_SFRM; + GPCR = GPIO_SSP_TXD | GPIO_SSP_SCLK | GPIO_SSP_SFRM; + + /* + * Set up registers for sleep mode. + */ + PWER = PWER_GPIO0; + PGSR = 0; + PCFR = 0; + PSDR = 0; +} + + +MACHINE_START(CEP, "Iskratel Cep") + BOOT_MEM(0xc0000000, 0x80000000, 0xf8000000) + BOOT_PARAMS(0xc0000100) + FIXUP(fixup_cep) + MAPIO(cep_map_io) + INITIRQ(sa1100_init_irq) +MACHINE_END diff -urN orig/arch/arm/mach-sa1100/cpu-sa1100.c linux/arch/arm/mach-sa1100/cpu-sa1100.c --- orig/arch/arm/mach-sa1100/cpu-sa1100.c Mon Aug 5 13:29:46 2002 +++ linux/arch/arm/mach-sa1100/cpu-sa1100.c Sun Sep 15 13:06:39 2002 @@ -216,7 +216,7 @@ static struct notifier_block sa1100_dram_block = { - notifier_call: sa1100_dram_notifier, + .notifier_call = sa1100_dram_notifier, }; diff -urN orig/arch/arm/mach-sa1100/cpu-sa1110.c linux/arch/arm/mach-sa1100/cpu-sa1110.c --- orig/arch/arm/mach-sa1100/cpu-sa1110.c Mon Aug 5 13:29:46 2002 +++ linux/arch/arm/mach-sa1100/cpu-sa1110.c Sun Sep 15 23:42:59 2002 @@ -49,33 +49,33 @@ }; static struct sdram_params tc59sm716_cl2_params __initdata = { - rows: 12, - tck: 10, - trcd: 20, - trp: 20, - twr: 10, - refresh: 64000, - cas_latency: 2, + .rows = 12, + .tck = 10, + .trcd = 20, + .trp = 20, + .twr = 10, + .refresh = 64000, + .cas_latency = 2, }; static struct sdram_params tc59sm716_cl3_params __initdata = { - rows: 12, - tck: 8, - trcd: 20, - trp: 20, - twr: 8, - refresh: 64000, - cas_latency: 3, + .rows = 12, + .tck = 8, + .trcd = 20, + .trp = 20, + .twr = 8, + .refresh = 64000, + .cas_latency = 3, }; static struct sdram_params samsung_k4s641632d_tc75 __initdata = { - rows: 14, - tck: 9, - trcd: 27, - trp: 20, - twr: 9, - refresh: 64000, - cas_latency: 3, + .rows = 14, + .tck = 9, + .trcd = 27, + .trp = 20, + .twr = 9, + .refresh = 64000, + .cas_latency = 3, }; static struct sdram_params sdram_params; @@ -235,21 +235,21 @@ * the programming. */ local_irq_save(flags); - asm("mcr p15, 0, %0, c10, c4" : : "r" (0)); + asm("mcr p15, 0, %0, c7, c10, 4" : : "r" (0)); udelay(10); - __asm__ __volatile__(" - b 2f - .align 5 -1: str %3, [%1, #0] @ MDCNFG - str %4, [%1, #28] @ MDREFR - str %5, [%1, #4] @ MDCAS0 - str %6, [%1, #8] @ MDCAS1 - str %7, [%1, #12] @ MDCAS2 - str %8, [%2, #0] @ PPCR - ldr %0, [%1, #0] - b 3f -2: b 1b -3: nop + __asm__ __volatile__(" \n\ + b 2f \n\ + .align 5 \n\ +1: str %3, [%1, #0] @ MDCNFG \n\ + str %4, [%1, #28] @ MDREFR \n\ + str %5, [%1, #4] @ MDCAS0 \n\ + str %6, [%1, #8] @ MDCAS1 \n\ + str %7, [%1, #12] @ MDCAS2 \n\ + str %8, [%2, #0] @ PPCR \n\ + ldr %0, [%1, #0] \n\ + b 3f \n\ +2: b 1b \n\ +3: nop \n\ nop" : "=&r" (unused) : "r" (&MDCNFG), "r" (&PPCR), "0" (sd.mdcnfg), diff -urN orig/arch/arm/mach-sa1100/dma-sa1100.c linux/arch/arm/mach-sa1100/dma-sa1100.c --- orig/arch/arm/mach-sa1100/dma-sa1100.c Mon Aug 5 13:29:46 2002 +++ linux/arch/arm/mach-sa1100/dma-sa1100.c Fri Sep 6 10:52:52 2002 @@ -477,11 +477,11 @@ return sa1111_dma_resume(channel); if (dma->stopped) { - int flags; - save_flags_cli(flags); + unsigned long flags; + local_irq_save(flags); dma->stopped = 0; process_dma(dma); - restore_flags(flags); + local_irq_restore(flags); } return 0; } diff -urN orig/arch/arm/mach-sa1100/dma-sa1111.c linux/arch/arm/mach-sa1100/dma-sa1111.c --- orig/arch/arm/mach-sa1100/dma-sa1111.c Mon Aug 5 13:29:46 2002 +++ linux/arch/arm/mach-sa1100/dma-sa1111.c Fri Feb 21 15:17:34 2003 @@ -305,7 +305,8 @@ * SDRAM bank 1 on Neponset). The default configuration selects * Assabet, so any address in bank 1 is necessarily invalid. */ - if((machine_is_assabet() || machine_is_pfs168()) && addr >= 0xc8000000) + if((machine_is_assabet() || machine_is_pfs168() || + machine_is_graphicsmaster() || machine_is_adsagc()) && addr >= 0xc8000000) return -1; /* The bug only applies to buffers located more than one megabyte diff -urN orig/arch/arm/mach-sa1100/flexanet.c linux/arch/arm/mach-sa1100/flexanet.c --- orig/arch/arm/mach-sa1100/flexanet.c Mon Aug 5 13:29:46 2002 +++ linux/arch/arm/mach-sa1100/flexanet.c Mon Aug 5 13:41:22 2002 @@ -26,6 +26,7 @@ #include #include #include +#include #include "generic.h" @@ -145,6 +146,11 @@ static int __init flexanet_init(void) { + /* Set IRQ edges */ + set_GPIO_IRQ_edge(GPIO_GUI_IRQ, GPIO_RISING_EDGE); + + /* deassert the GUI reset */ + FLEXANET_BCR_set(FHH_BCR_GUI_NRST); return 0; } @@ -155,10 +161,6 @@ fixup_flexanet(struct machine_desc *desc, struct param_struct *params, char **cmdline, struct meminfo *mi) { - int status; - unsigned long now; - - /* fixed RAM size, by now (64MB) */ SET_BANK( 0, 0xc0000000, 64*1024*1024 ); mi->nr_banks = 1; @@ -175,7 +177,7 @@ { 0xe8000000, 0x00000000, 0x02000000, DOMAIN_IO, 0, 1, 0, 0 }, /* Flash bank 0 */ { 0xf0000000, 0x10000000, 0x00001000, DOMAIN_IO, 0, 1, 0, 0 }, /* Board Control Register */ { 0xf1000000, 0x18000000, 0x01000000, DOMAIN_IO, 0, 1, 0, 0 }, /* Ethernet controller */ - { 0xD0000000, 0x40000000, 0x01000000, DOMAIN_IO, 0, 1, 0, 0 }, /* Instrument boards */ + { 0xD0000000, 0x40000000, 0x04000000, DOMAIN_IO, 0, 1, 0, 0 }, /* Instrument boards */ { 0xD8000000, 0x48000000, 0x01000000, DOMAIN_IO, 0, 1, 0, 0 }, /* External peripherals */ LAST_DESC }; @@ -198,13 +200,6 @@ */ PCFR = PCFR_OPDE | PCFR_FP | PCFR_FS; - /* deassert the GUI reset */ - FLEXANET_BCR_set(FHH_BCR_GUI_NRST); - - /* - * Set IRQ edges - */ - set_GPIO_IRQ_edge(GPIO_GUI_IRQ, GPIO_RISING_EDGE); } diff -urN orig/arch/arm/mach-sa1100/frodo.c linux/arch/arm/mach-sa1100/frodo.c --- orig/arch/arm/mach-sa1100/frodo.c Mon Aug 5 13:29:46 2002 +++ linux/arch/arm/mach-sa1100/frodo.c Thu Oct 24 13:01:38 2002 @@ -12,15 +12,22 @@ * * History: * + * 2002/05/27 Setup GPIOs for all the onboard peripherals so + * that we don't have to do it in each driver + * * 2002/01/31 Initial version */ #include #include +#include #include +#include -#include #include +#include +#include +#include #include #include @@ -28,29 +35,107 @@ #include "generic.h" -static struct map_desc frodo_io_desc[] __initdata = -{ - /* virtual physical length domain r w c b */ - { 0xe8000000, 0x00000000, 0x04000000, DOMAIN_IO, 1, 1, 0, 0 }, /* flash memory */ - { 0xf0000000, 0x40000000, 0x00100000, DOMAIN_IO, 1, 1, 0, 0 }, /* 16-bit on-board devices (including CPLDs) */ - { 0xf1000000, 0x18000000, 0x04000000, DOMAIN_IO, 1, 1, 0, 0 }, /* 32-bit daughter card */ - LAST_DESC +static struct map_desc frodo_io_desc[] __initdata = { + /* virtual physical length domain r w c b */ + { 0xe8000000, 0x00000000, 0x04000000, DOMAIN_IO, 0, 1, 0, 0 }, /* flash memory */ + { 0xf0000000, 0x40000000, 0x00100000, DOMAIN_IO, 0, 1, 0, 0 }, /* 16-bit on-board devices (including CPLDs) */ + { 0xf1000000, 0x18000000, 0x04000000, DOMAIN_IO, 0, 1, 0, 0 }, /* 32-bit daughter card */ + LAST_DESC }; +static spinlock_t frodo_cpld_lock = SPIN_LOCK_UNLOCKED; +static volatile u16 *frodo_cpld_memory = (u16 *) 0xf0000000; + static void __init frodo_map_io (void) { - sa1100_map_io (); - iotable_init (frodo_io_desc); + sa1100_map_io (); + iotable_init (frodo_io_desc); + + sa1100_register_uart (0,2); /* UART2 (serial console) */ + sa1100_register_uart (1,1); /* UART1 (big kahuna flow control serial port) */ + + /* + * Set SUS bit in SDCR0 so serial port 1 acts as a UART. + * See Intel SA-1110 Developers Manual Section 11.9.2.1 (GPCLK/UART Select) + */ + Ser1SDCR0 |= SDCR0_SUS; +} + +static int __init frodo_init_irq(void) +{ + int i,gpio[] = { + FRODO_IDE_GPIO, + FRODO_ETH_GPIO, + FRODO_USB_DC_GPIO, + FRODO_USB_HC_GPIO, + FRODO_RTC_GPIO, + FRODO_UART1_GPIO, + FRODO_UART2_GPIO, + FRODO_PCMCIA_STATUS_GPIO, + FRODO_PCMCIA_RDYBSY_GPIO + }; + + for (i = 0; i < sizeof (gpio) / sizeof (gpio[0]); i++) + set_GPIO_IRQ_edge (gpio[i],GPIO_RISING_EDGE); + + return (0); +} - sa1100_register_uart (0,2); /* UART2 (serial console) */ - sa1100_register_uart (1,1); /* UART1 (big kahuna flow control serial port) */ +__initcall(frodo_init_irq); +#if 0 +static int __init frodo_init_cpld(void) +{ + if ((frodo_cpld_memory = ioremap (FRODO_CPLD_BASE,FRODO_CPLD_LENGTH)) == NULL) + panic ("Couldn't map CPLD memory to a virtual address. We're screwed!\n"); - /* - * Set SUS bit in SDCR0 so serial port 1 acts as a UART. - * See Intel SA-1110 Developers Manual Section 11.9.2.1 (GPCLK/UART Select) - */ - Ser1SDCR0 |= SDCR0_SUS; + return (0); } + +__initcall(frodo_init_cpld); +#endif +u16 frodo_cpld_read (u32 reg) +{ + unsigned long flags; + u16 value; + + spin_lock_irqsave (&frodo_cpld_lock,flags); + value = frodo_cpld_memory[reg / 2]; + spin_unlock_irqrestore (&frodo_cpld_lock,flags); + + return (value); +} + +void frodo_cpld_write (u32 reg,u16 value) +{ + unsigned long flags; + + spin_lock_irqsave (&frodo_cpld_lock,flags); + frodo_cpld_memory[reg / 2] = value; + spin_unlock_irqrestore (&frodo_cpld_lock,flags); +} + +void frodo_cpld_set (u32 reg,u16 mask) +{ + unsigned long flags; + + spin_lock_irqsave (&frodo_cpld_lock,flags); + frodo_cpld_memory[reg / 2] |= mask; + spin_unlock_irqrestore (&frodo_cpld_lock,flags); +} + +void frodo_cpld_clear (u32 reg,u16 mask) +{ + unsigned long flags; + + spin_lock_irqsave (&frodo_cpld_lock,flags); + frodo_cpld_memory[reg / 2] &= ~mask; + spin_unlock_irqrestore (&frodo_cpld_lock,flags); +} + +EXPORT_SYMBOL (frodo_cpld_read); +EXPORT_SYMBOL (frodo_cpld_write); +EXPORT_SYMBOL (frodo_cpld_set); +EXPORT_SYMBOL (frodo_cpld_clear); MACHINE_START (FRODO,"2d3D, Inc. SA-1110 Development Board") BOOT_MEM (0xc0000000,0x80000000,0xf8000000) diff -urN orig/arch/arm/mach-sa1100/generic.c linux/arch/arm/mach-sa1100/generic.c --- orig/arch/arm/mach-sa1100/generic.c Mon Aug 5 13:29:46 2002 +++ linux/arch/arm/mach-sa1100/generic.c Sat May 18 17:03:16 2002 @@ -86,7 +86,7 @@ static int __init sa11x0_init_clock(void) { - cpufreq_init(cclk_frequency_100khz[PPCR & 0xf] * 100); + cpufreq_init(cclk_frequency_100khz[PPCR & 0xf] * 100, 59000, 287000); return 0; } diff -urN orig/arch/arm/mach-sa1100/graphicsclient.c linux/arch/arm/mach-sa1100/graphicsclient.c --- orig/arch/arm/mach-sa1100/graphicsclient.c Mon Aug 5 13:29:46 2002 +++ linux/arch/arm/mach-sa1100/graphicsclient.c Fri Feb 21 15:17:35 2003 @@ -14,6 +14,8 @@ #include #include #include +#include +#include #include #include @@ -34,55 +36,58 @@ * Handlers for GraphicsClient's external IRQ logic */ +#define GRAPHICSCLIENT_N_IRQ (IRQ_GRAPHICSCLIENT_END - IRQ_GRAPHICSCLIENT_START) +#define GRAPHICSCLIENT_FIRST_IRQ (IRQ_GRAPHICSCLIENT_CAN - IRQ_GRAPHICSCLIENT_START) + static void ADS_IRQ_demux( int irq, void *dev_id, struct pt_regs *regs ) { int i; while( (irq = ADS_INT_ST1 | (ADS_INT_ST2 << 8)) ){ - for( i = 0; i < 16; i++ ) + for( i = GRAPHICSCLIENT_FIRST_IRQ; i < GRAPHICSCLIENT_N_IRQ; i++ ) if( irq & (1<nr_banks = 2; - - ROOT_DEV = MKDEV(RAMDISK_MAJOR,0); - setup_ramdisk( 1, 0, 0, 8192 ); - setup_initrd( __phys_to_virt(0xc0800000), 4*1024*1024 ); -} - static struct map_desc graphicsclient_io_desc[] __initdata = { /* virtual physical length domain r w c b */ { 0xe8000000, 0x08000000, 0x02000000, DOMAIN_IO, 0, 1, 0, 0 }, /* Flash bank 1 */ @@ -143,120 +133,31 @@ LAST_DESC }; -static struct gc_uart_ctrl_data_t gc_uart_ctrl_data[] = { - { GPIO_GC_UART0_CTS, 0, NULL,NULL }, - { GPIO_GC_UART1_CTS, 0, NULL,NULL }, - { GPIO_GC_UART2_CTS, 0, NULL,NULL } -}; - -static void -graphicsclient_cts_intr(int irq, void *dev_id, struct pt_regs *regs) -{ - struct gc_uart_ctrl_data_t * uart_data = (struct gc_uart_ctrl_data_t *)dev_id; - int cts = !(GPLR & uart_data->cts_gpio); - - /* NOTE: I supose that we will no get any interrupt - if the GPIO is not changed, so maybe - the cts_prev_state can be removed ... */ - if (cts != uart_data->cts_prev_state) { - uart_data->cts_prev_state = cts; - - uart_handle_cts_change(uart_data->info, cts); - } -} - -static int -graphicsclient_register_cts_intr(int gpio, int irq, - struct gc_uart_ctrl_data_t *uart_data) -{ - int ret = 0; - - set_GPIO_IRQ_edge(gpio, GPIO_BOTH_EDGES); - - ret = request_irq(irq, graphicsclient_cts_intr, - 0, "GC RS232 CTS", uart_data); - if (ret) { - printk(KERN_ERR "uart_open: failed to register CTS irq (%d)\n", - ret); - free_irq(irq, uart_data); - } - - return ret; -} - -static int -graphicsclient_uart_open(struct uart_port *port, struct uart_info *info) +static int graphicsclient_uart_open(struct uart_port *port, struct uart_info *info) { - int ret = 0; + int ret = 0; if (port->mapbase == _Ser1UTCR0) { Ser1SDCR0 |= SDCR0_UART; - /* Set RTS Output */ - GPSR = GPIO_GC_UART0_RTS; - - gc_uart_ctrl_data[0].cts_prev_state = 0; - gc_uart_ctrl_data[0].info = info; - gc_uart_ctrl_data[0].port = port; - - /* register uart0 CTS irq */ - ret = graphicsclient_register_cts_intr(GPIO_GC_UART0_CTS, - IRQ_GC_UART0_CTS, - &gc_uart_ctrl_data[0]); - } else if (port->mapbase == _Ser2UTCR0) { + } + else if (port->mapbase == _Ser2UTCR0) { Ser2UTCR4 = Ser2HSCR0 = 0; - /* Set RTS Output */ - GPSR = GPIO_GC_UART1_RTS; - - gc_uart_ctrl_data[1].cts_prev_state = 0; - gc_uart_ctrl_data[1].info = info; - gc_uart_ctrl_data[1].port = port; - - /* register uart1 CTS irq */ - ret = graphicsclient_register_cts_intr(GPIO_GC_UART1_CTS, - IRQ_GC_UART1_CTS, - &gc_uart_ctrl_data[1]); - } else if (port->mapbase == _Ser3UTCR0) { - /* Set RTS Output */ - GPSR = GPIO_GC_UART2_RTS; - - gc_uart_ctrl_data[2].cts_prev_state = 0; - gc_uart_ctrl_data[2].info = info; - gc_uart_ctrl_data[2].port = port; - - /* register uart2 CTS irq */ - ret = graphicsclient_register_cts_intr(GPIO_GC_UART2_CTS, - IRQ_GC_UART2_CTS, - &gc_uart_ctrl_data[2]); } return ret; } -static int -graphicsclient_uart_close(struct uart_port *port, struct uart_info *info) -{ - if (port->mapbase == _Ser1UTCR0) { - free_irq(IRQ_GC_UART0_CTS, &gc_uart_ctrl_data[0]); - } else if (port->mapbase == _Ser2UTCR0) { - free_irq(IRQ_GC_UART1_CTS, &gc_uart_ctrl_data[1]); - } else if (port->mapbase == _Ser3UTCR0) { - free_irq(IRQ_GC_UART2_CTS, &gc_uart_ctrl_data[2]); - } - - return 0; -} - static u_int graphicsclient_get_mctrl(struct uart_port *port) { u_int result = TIOCM_CD | TIOCM_DSR; if (port->mapbase == _Ser1UTCR0) { - if (!(GPLR & GPIO_GC_UART0_CTS)) + if (!(GPLR & GPIO_GPIO14)) result |= TIOCM_CTS; } else if (port->mapbase == _Ser2UTCR0) { - if (!(GPLR & GPIO_GC_UART1_CTS)) + if (!(GPLR & GPIO_GPIO16)) result |= TIOCM_CTS; } else if (port->mapbase == _Ser3UTCR0) { - if (!(GPLR & GPIO_GC_UART2_CTS)) + if (!(GPLR & GPIO_GPIO18)) result |= TIOCM_CTS; } else { result = TIOCM_CTS; @@ -269,39 +170,54 @@ { if (port->mapbase == _Ser1UTCR0) { if (mctrl & TIOCM_RTS) - GPCR = GPIO_GC_UART0_RTS; + GPCR = GPIO_GPIO15; else - GPSR = GPIO_GC_UART0_RTS; + GPSR = GPIO_GPIO15; } else if (port->mapbase == _Ser2UTCR0) { if (mctrl & TIOCM_RTS) - GPCR = GPIO_GC_UART1_RTS; + GPCR = GPIO_GPIO17; else - GPSR = GPIO_GC_UART1_RTS; + GPSR = GPIO_GPIO17; } else if (port->mapbase == _Ser3UTCR0) { if (mctrl & TIOCM_RTS) - GPCR = GPIO_GC_UART2_RTS; + GPCR = GPIO_GPIO19; else - GPSR = GPIO_GC_UART2_RTS; + GPSR = GPIO_GPIO19; } } static void graphicsclient_uart_pm(struct uart_port *port, u_int state, u_int oldstate) { - if (!state) { - /* make serial ports work ... */ - Ser2UTCR4 = 0; - Ser2HSCR0 = 0; - Ser1SDCR0 |= SDCR0_UART; + // state has ACPI D0-D3 + // ACPI D0 : resume from suspend + // ACPI D1-D3 : enter to a suspend state + if (port->mapbase == _Ser1UTCR0) { + if (state) { + // disable uart + Ser1UTCR3 = 0; + } + } + else if (port->mapbase == _Ser2UTCR0) { + if (state) { + // disable uart + Ser2UTCR3 = 0; + Ser2HSCR0 = 0; + } + } + else if (port->mapbase == _Ser3UTCR0) { + if (state) { + // disable uart + Ser3UTCR3 = 0; + } } } static struct sa1100_port_fns graphicsclient_port_fns __initdata = { - open: graphicsclient_uart_open, - close: graphicsclient_uart_close, - get_mctrl: graphicsclient_get_mctrl, - set_mctrl: graphicsclient_set_mctrl, - pm: graphicsclient_uart_pm, + .open = graphicsclient_uart_open, + .get_mctrl = graphicsclient_get_mctrl, + .set_mctrl = graphicsclient_set_mctrl, + .pm = graphicsclient_uart_pm, }; static void __init graphicsclient_map_io(void) @@ -312,14 +228,20 @@ sa1100_register_uart_fns(&graphicsclient_port_fns); sa1100_register_uart(0, 3); sa1100_register_uart(1, 1); + + // don't register if you want to use IRDA +#ifndef CONFIG_SA1100_FIR sa1100_register_uart(2, 2); - GPDR |= GPIO_GC_UART0_RTS | GPIO_GC_UART1_RTS | GPIO_GC_UART2_RTS; - GPDR &= ~(GPIO_GC_UART0_CTS | GPIO_GC_UART1_RTS | GPIO_GC_UART2_RTS); +#endif + + /* set GPDR now */ + GPDR |= GPIO_GPIO15 | GPIO_GPIO17 | GPIO_GPIO19; + GPDR &= ~(GPIO_GPIO14 | GPIO_GPIO16 | GPIO_GPIO18); } MACHINE_START(GRAPHICSCLIENT, "ADS GraphicsClient") + BOOT_PARAMS(0xc000003c) BOOT_MEM(0xc0000000, 0x80000000, 0xf8000000) - FIXUP(fixup_graphicsclient) MAPIO(graphicsclient_map_io) INITIRQ(graphicsclient_init_irq) MACHINE_END diff -urN orig/arch/arm/mach-sa1100/graphicsmaster.c linux/arch/arm/mach-sa1100/graphicsmaster.c --- orig/arch/arm/mach-sa1100/graphicsmaster.c Mon Aug 5 13:29:46 2002 +++ linux/arch/arm/mach-sa1100/graphicsmaster.c Thu Feb 27 23:20:10 2003 @@ -12,8 +12,13 @@ #include #include #include +#include +#include +#include +#include #include +#include #include #include @@ -40,6 +45,15 @@ */ sa1110_mb_disable(); + /* GraphicsMaster uses GPIO pins for SPI interface to AVR + */ + + /* use the alternate SSP pins */ + PPAR |= PPAR_SSPGPIO; + + // Set RTS low during sleep + PGSR |= GPIO_GPIO15 | GPIO_GPIO17 | GPIO_GPIO19; + /* * Probe for SA1111. */ @@ -67,9 +81,11 @@ * Enable PWM control for LCD */ SKPCR |= SKPCR_PWMCLKEN; - SKPWM0 = 0x7F; // VEE + SACR1 &= ~SACR1_L3EN; + ADS_DCR |= DCR_BACKLITE_ON; + SKPWM0 = 0x01; // Backlight SKPEN0 = 1; - SKPWM1 = 0x01; // Backlight + SKPWM1 = 0x7F; // VEE SKPEN1 = 1; /* @@ -83,7 +99,7 @@ */ sa1110_mb_enable(); - sa1111_init_irq(ADS_EXT_IRQ(0)); + sa1111_init_irq(IRQ_GRAPHICSMASTER_SA1111); return 0; } @@ -94,56 +110,58 @@ * Handlers for GraphicsMaster's external IRQ logic */ +#define GRAPHICSMASTER_N_IRQ (IRQ_GRAPHICSMASTER_END - IRQ_GRAPHICSMASTER_START) + static void ADS_IRQ_demux( int irq, void *dev_id, struct pt_regs *regs ) { int i; while( (irq = ADS_INT_ST1 | (ADS_INT_ST2 << 8)) ){ - for( i = 0; i < 16; i++ ) + for( i = 0; i < GRAPHICSMASTER_N_IRQ; i++ ) if( irq & (1<nr_banks = 1; - SET_BANK( 1, 0xc8000000, 16*1024*1024 ); - mi->nr_banks = 2; - - ROOT_DEV = MKDEV(RAMDISK_MAJOR,0); - setup_ramdisk( 1, 0, 0, 8192 ); - setup_initrd( __phys_to_virt(0xc0800000), 4*1024*1024 ); -} - static struct map_desc graphicsmaster_io_desc[] __initdata = { /* virtual physical length domain r w c b */ { 0xe8000000, 0x08000000, 0x02000000, DOMAIN_IO, 0, 1, 0, 0 }, /* Flash bank 1 */ @@ -212,17 +214,9 @@ if (port->mapbase == _Ser1UTCR0) { Ser1SDCR0 |= SDCR0_UART; - /* Set RTS Output */ - GPSR = GPIO_GPIO15; } else if (port->mapbase == _Ser2UTCR0) { Ser2UTCR4 = Ser2HSCR0 = 0; - /* Set RTS Output */ - GPSR = GPIO_GPIO17; - } - else if (port->mapbase == _Ser3UTCR0) { - /* Set RTS Output */ - GPSR = GPIO_GPIO19; } return ret; } @@ -238,7 +232,7 @@ if (!(GPLR & GPIO_GPIO16)) result |= TIOCM_CTS; } else if (port->mapbase == _Ser3UTCR0) { - if (!(GPLR & GPIO_GPIO17)) + if (!(GPLR & GPIO_GPIO18)) result |= TIOCM_CTS; } else { result = TIOCM_CTS; @@ -270,19 +264,35 @@ static void graphicsmaster_uart_pm(struct uart_port *port, u_int state, u_int oldstate) { - if (!state) { - /* make serial ports work ... */ - Ser2UTCR4 = 0; - Ser2HSCR0 = 0; - Ser1SDCR0 |= SDCR0_UART; + // state has ACPI D0-D3 + // ACPI D0 : resume from suspend + // ACPI D1-D3 : enter to a suspend state + if (port->mapbase == _Ser1UTCR0) { + if (state) { + // disable uart + Ser1UTCR3 = 0; + } + } + else if (port->mapbase == _Ser2UTCR0) { + if (state) { + // disable uart + Ser2UTCR3 = 0; + Ser2HSCR0 = 0; + } + } + else if (port->mapbase == _Ser3UTCR0) { + if (state) { + // disable uart + Ser3UTCR3 = 0; + } } } static struct sa1100_port_fns graphicsmaster_port_fns __initdata = { - open: graphicsmaster_uart_open, - get_mctrl: graphicsmaster_get_mctrl, - set_mctrl: graphicsmaster_set_mctrl, - pm: graphicsmaster_uart_pm, + .open = graphicsmaster_uart_open, + .get_mctrl = graphicsmaster_get_mctrl, + .set_mctrl = graphicsmaster_set_mctrl, + .pm = graphicsmaster_uart_pm, }; static void __init graphicsmaster_map_io(void) @@ -293,7 +303,10 @@ sa1100_register_uart_fns(&graphicsmaster_port_fns); sa1100_register_uart(0, 3); sa1100_register_uart(1, 1); + // don't register if you want to use IRDA +#ifndef CONFIG_SA1100_FIR sa1100_register_uart(2, 2); +#endif /* set GPDR now */ GPDR |= GPIO_GPIO15 | GPIO_GPIO17 | GPIO_GPIO19; @@ -301,8 +314,8 @@ } MACHINE_START(GRAPHICSMASTER, "ADS GraphicsMaster") + BOOT_PARAMS(0xc000003c) BOOT_MEM(0xc0000000, 0x80000000, 0xf8000000) - FIXUP(fixup_graphicsmaster) MAPIO(graphicsmaster_map_io) INITIRQ(graphicsmaster_init_irq) MACHINE_END diff -urN orig/arch/arm/mach-sa1100/h3600.c linux/arch/arm/mach-sa1100/h3600.c --- orig/arch/arm/mach-sa1100/h3600.c Mon Aug 5 13:29:46 2002 +++ linux/arch/arm/mach-sa1100/h3600.c Sun Sep 15 13:02:38 2002 @@ -144,11 +144,11 @@ } static struct ipaq_model_ops h3100_model_ops __initdata = { - model : IPAQ_H3100, - generic_name : "3100", - initialize : h3100_init_egpio, - control : h3100_control_egpio, - read : h3100_read_egpio + .model = IPAQ_H3100, + .generic_name = "3100", + .initialize = h3100_init_egpio, + .control = h3100_control_egpio, + .read = h3100_read_egpio }; @@ -223,11 +223,11 @@ } static struct ipaq_model_ops h3600_model_ops __initdata = { - model : IPAQ_H3600, - generic_name : "3600", - initialize : h3600_init_egpio, - control : h3600_control_egpio, - read : h3600_read_egpio + .model = IPAQ_H3600, + .generic_name = "3600", + .initialize = h3600_init_egpio, + .control = h3600_control_egpio, + .read = h3600_read_egpio }; /************************* H3800 *************************/ @@ -331,11 +331,11 @@ } static struct ipaq_model_ops h3800_model_ops __initdata = { - model : IPAQ_H3800, - generic_name : "3800", - initialize : h3800_init_egpio, - control : h3800_control_egpio, - read : h3800_read_egpio + .model = IPAQ_H3800, + .generic_name = "3800", + .initialize = h3800_init_egpio, + .control = h3800_control_egpio, + .read = h3800_read_egpio }; @@ -468,12 +468,12 @@ } static struct sa1100_port_fns h3600_port_fns __initdata = { - set_mctrl: h3600_uart_set_mctrl, - get_mctrl: h3600_uart_get_mctrl, - pm: h3600_uart_pm, - set_wake: h3600_uart_set_wake, - open: h3600_uart_open, - close: h3600_uart_close, + .set_mctrl = h3600_uart_set_mctrl, + .get_mctrl = h3600_uart_get_mctrl, + .pm = h3600_uart_pm, + .set_wake = h3600_uart_set_wake, + .open = h3600_uart_open, + .close = h3600_uart_close, }; static struct map_desc h3600_io_desc[] __initdata = { diff -urN orig/arch/arm/mach-sa1100/hackkit.c linux/arch/arm/mach-sa1100/hackkit.c --- orig/arch/arm/mach-sa1100/hackkit.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-sa1100/hackkit.c Thu Feb 6 11:33:12 2003 @@ -0,0 +1,221 @@ +/* + * linux/arch/arm/mach-sa1100/hackkit.c + * + * Copyright (C) 2002 Stefan Eletzhofer + * + * This file contains all HackKit tweaks. Based on original work from + * Nicolas Pitre's assabet fixes + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + * + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include + +#include +#include +#include +#include + +#include + +#include + +#include "generic.h" + +#define DEBUG 1 + +#ifdef DEBUG +# define DPRINTK( x, args... ) printk( "%s: line %d: "x, __FUNCTION__, __LINE__, ## args ); +#else +# define DPRINTK( x, args... ) /* nix */ +#endif + +/********************************************************************** + * prototypes + */ + +/* init funcs */ +static void __init fixup_hackkit(struct machine_desc *desc, + struct param_struct *params, char **cmdline, struct meminfo *mi); +static void __init get_hackkit_scr(void); +static int __init hackkit_init(void); +static void __init hackkit_init_irq(void); +static void __init hackkit_map_io(void); + +static int hackkit_get_mctrl(struct uart_port *port); +static void hackkit_set_mctrl(struct uart_port *port, u_int mctrl); +static void hackkit_uart_pm(struct uart_port *port, u_int state, u_int oldstate); + +extern void convert_to_tag_list(struct param_struct *params, int mem_init); + + +/********************************************************************** + * global data + */ + +/********************************************************************** + * static data + */ + +static struct map_desc hackkit_io_desc[] __initdata = { + /* virtual physical length domain r w c b */ + { 0xe8000000, 0x00000000, 0x01000000, DOMAIN_IO, 0, 1, 0, 0 }, /* Flash bank 0 */ + LAST_DESC +}; + +static struct sa1100_port_fns hackkit_port_fns __initdata = { + .set_mctrl = hackkit_set_mctrl, + .get_mctrl = hackkit_get_mctrl, + .pm = hackkit_uart_pm, +}; + +/********************************************************************** + * Static functions + */ + +static void __init hackkit_map_io(void) +{ + DPRINTK( "%s\n", "START" ); + sa1100_map_io(); + iotable_init(hackkit_io_desc); + + sa1100_register_uart_fns(&hackkit_port_fns); + sa1100_register_uart(0, 1); /* com port */ + sa1100_register_uart(1, 2); + sa1100_register_uart(2, 3); /* radio module */ + + Ser1SDCR0 |= SDCR0_SUS; +} + +static void __init hackkit_init_irq(void) +{ + /* none used yet */ +} + +/** + * fixup_hackkit - fixup function for system 3 board + * @desc: machine description + * @param: kernel params + * @cmdline: kernel cmdline + * @mi: memory info struct + * + */ +static void __init fixup_hackkit(struct machine_desc *desc, + struct param_struct *params, char **cmdline, struct meminfo *mi) +{ + DPRINTK( "%s\n", "START" ); + + ROOT_DEV = MKDEV(RAMDISK_MAJOR,0); + setup_ramdisk( 1, 0, 0, 8192 ); + setup_initrd( 0xc0800000, 8*1024*1024 ); +} + + +/** + * hackkit_uart_pm - powermgmt callback function for system 3 UART + * @port: uart port structure + * @state: pm state + * @oldstate: old pm state + * + */ +static void hackkit_uart_pm(struct uart_port *port, u_int state, u_int oldstate) +{ + /* TODO: switch on/off uart in powersave mode */ +} + +/* + * Note! this can be called from IRQ context. + * FIXME: No modem ctrl lines yet. + */ +static void hackkit_set_mctrl(struct uart_port *port, u_int mctrl) +{ +#if 0 + if (port->mapbase == _Ser1UTCR0) { + u_int set = 0, clear = 0; + + if (mctrl & TIOCM_RTS) + set |= PT_CTRL2_RS1_RTS; + else + clear |= PT_CTRL2_RS1_RTS; + + if (mctrl & TIOCM_DTR) + set |= PT_CTRL2_RS1_DTR; + else + clear |= PT_CTRL2_RS1_DTR; + + PTCTRL2_clear(clear); + PTCTRL2_set(set); + } +#endif +} + +static int hackkit_get_mctrl(struct uart_port *port) +{ + u_int ret = 0; +#if 0 + u_int irqsr = PT_IRQSR; + + /* need 2 reads to read current value */ + irqsr = PT_IRQSR; + + /* TODO: check IRQ source register for modem/com + status lines and set them correctly. */ +#endif + + ret = TIOCM_CD | TIOCM_CTS | TIOCM_DSR; + + return ret; +} + +static int __init hackkit_init(void) +{ + int ret = 0; + DPRINTK( "%s\n", "START" ); + + if ( !machine_is_hackkit() ) { + ret = -EINVAL; + goto DONE; + } + + hackkit_init_irq(); + + ret = 0; +DONE: + DPRINTK( "ret=%d\n", ret ); + return ret; +} + +/********************************************************************** + * Exported Functions + */ + +/********************************************************************** + * kernel magic macros + */ +__initcall(hackkit_init); + +MACHINE_START(HACKKIT, "HackKit Cpu Board") + BOOT_MEM(0xc0000000, 0x80000000, 0xf8000000) + BOOT_PARAMS(0xc0000100) + FIXUP(fixup_hackkit) + MAPIO(hackkit_map_io) + INITIRQ(sa1100_init_irq) +MACHINE_END diff -urN orig/arch/arm/mach-sa1100/irq.c linux/arch/arm/mach-sa1100/irq.c --- orig/arch/arm/mach-sa1100/irq.c Mon Aug 5 13:29:46 2002 +++ linux/arch/arm/mach-sa1100/irq.c Tue Feb 4 15:08:01 2003 @@ -15,6 +15,8 @@ #include #include #include +#include +#include #include #include @@ -30,9 +32,9 @@ * This must be called *before* the appropriate IRQ is registered. * Use this instead of directly setting GRER/GFER. */ - static int GPIO_IRQ_rising_edge; static int GPIO_IRQ_falling_edge; +static int GPIO_IRQ_mask = (1 << 11) - 1; void set_GPIO_IRQ_edge(int gpio_mask, int edge) { @@ -51,6 +53,8 @@ GPIO_IRQ_rising_edge &= ~gpio_mask; GPDR &= ~gpio_mask; GAFR &= ~gpio_mask; + GRER = GPIO_IRQ_rising_edge & GPIO_IRQ_mask; + GFER = GPIO_IRQ_falling_edge & GPIO_IRQ_mask; while (gpio_mask) { if (irq == 11) irq = IRQ_GPIO11; @@ -86,8 +90,10 @@ static void sa1100_mask_and_ack_GPIO0_10_irq(unsigned int irq) { - ICMR &= ~(1 << irq); - GEDR = (1 << irq); + unsigned int mask = 1 << irq; + + ICMR &= ~mask; + GEDR = mask; } static void sa1100_mask_GPIO0_10_irq(unsigned int irq) @@ -97,16 +103,13 @@ static void sa1100_unmask_GPIO0_10_irq(unsigned int irq) { - GRER = (GRER & ~(1 << irq)) | (GPIO_IRQ_rising_edge & (1 << irq)); - GFER = (GFER & ~(1 << irq)) | (GPIO_IRQ_falling_edge & (1 << irq)); - ICMR |= (1 << irq); + ICMR |= 1 << irq; } /* * Install handler for GPIO 11-27 edge detect interrupts */ -static int GPIO_11_27_enabled; /* enabled i.e. unmasked GPIO IRQs */ static int GPIO_11_27_spurious; /* GPIOs that triggered when masked */ static void sa1100_GPIO11_27_demux(int irq, void *dev_id, @@ -124,7 +127,7 @@ * enabled at this point are considered spurious. Those * are cleared but only de-activated if they happen twice. */ - spurious = irq & ~GPIO_11_27_enabled; + spurious = irq & ~GPIO_IRQ_mask; if (spurious) { GEDR = spurious; GRER &= ~(spurious & GPIO_11_27_spurious); @@ -143,29 +146,29 @@ } static struct irqaction GPIO11_27_irq = { - name: "GPIO 11-27", - handler: sa1100_GPIO11_27_demux, - flags: SA_INTERRUPT + .name = "GPIO 11-27", + .handler = sa1100_GPIO11_27_demux, + .flags = SA_INTERRUPT }; static void sa1100_mask_and_ack_GPIO11_27_irq(unsigned int irq) { - int mask = (1 << GPIO_11_27_IRQ(irq)); + unsigned int mask = (1 << GPIO_11_27_IRQ(irq)); GPIO_11_27_spurious &= ~mask; - GPIO_11_27_enabled &= ~mask; + GPIO_IRQ_mask &= ~mask; GEDR = mask; } static void sa1100_mask_GPIO11_27_irq(unsigned int irq) { - int mask = (1 << GPIO_11_27_IRQ(irq)); + unsigned int mask = (1 << GPIO_11_27_IRQ(irq)); GPIO_11_27_spurious &= ~mask; - GPIO_11_27_enabled &= ~mask; + GPIO_IRQ_mask &= ~mask; } static void sa1100_unmask_GPIO11_27_irq(unsigned int irq) { - int mask = (1 << GPIO_11_27_IRQ(irq)); + unsigned int mask = (1 << GPIO_11_27_IRQ(irq)); if (GPIO_11_27_spurious & mask) { /* * We don't want to miss an interrupt that would have occurred @@ -185,15 +188,17 @@ return; } } - GPIO_11_27_enabled |= mask; - GRER = (GRER & ~mask) | (GPIO_IRQ_rising_edge & mask); - GFER = (GFER & ~mask) | (GPIO_IRQ_falling_edge & mask); + + GPIO_IRQ_mask |= mask; + + GRER = GPIO_IRQ_rising_edge & GPIO_IRQ_mask; + GFER = GPIO_IRQ_falling_edge & GPIO_IRQ_mask; } static struct resource irq_resource = { - name: "irqs", - start: 0x90050000, - end: 0x9005ffff, + .name = "irqs", + .start = 0x90050000, + .end = 0x9005ffff, }; void __init sa1100_init_irq(void) @@ -248,10 +253,4 @@ irq_desc[irq].unmask = sa1100_unmask_GPIO11_27_irq; } setup_arm_irq( IRQ_GPIO11_27, &GPIO11_27_irq ); - - /* - * We generally don't want the LCD IRQ being - * enabled as soon as we request it. - */ - irq_desc[IRQ_LCD].noautoenable = 1; } diff -urN orig/arch/arm/mach-sa1100/jornada720.c linux/arch/arm/mach-sa1100/jornada720.c --- orig/arch/arm/mach-sa1100/jornada720.c Mon Aug 5 13:29:46 2002 +++ linux/arch/arm/mach-sa1100/jornada720.c Sun Dec 29 12:01:34 2002 @@ -8,6 +8,8 @@ #include #include +#include +#include #include #include @@ -34,7 +36,7 @@ udelay(1); GPSR = GPIO_GPIO20; udelay(20); - SKCR = JORSKCR_INIT; /* Turn on the PLL, enable Ready and enable nOE assertion from DC */ + SBI_SKCR = JORSKCR_INIT;/* Turn on the PLL, enable Ready and enable nOE assertion from DC */ mdelay(100); SBI_SKCR = JORSKCR_RCLK;/* turn on the RCLOCK */ diff -urN orig/arch/arm/mach-sa1100/leds-adsagc.c linux/arch/arm/mach-sa1100/leds-adsagc.c --- orig/arch/arm/mach-sa1100/leds-adsagc.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-sa1100/leds-adsagc.c Fri Feb 21 15:17:34 2003 @@ -0,0 +1,107 @@ +/* + * linux/arch/arm/mach-sa1100/leds-adsagc.c + * + * ADS Advanced Graphics Client LEDs support + * Robert Whaley, Feb 7, 2003 + */ +#include +#include + +#include +#include +#include + +#include "leds.h" + + +#define LED_STATE_ENABLED 1 +#define LED_STATE_CLAIMED 2 + +static unsigned int led_state; +static unsigned int hw_led_state; + +#define LED_TIMER ADS_LED0 /* green heartbeat */ +#define LED_USER ADS_LED1 /* amber, boots to on */ +#define LED_IDLE ADS_LED2 /* red has the idle led, if any */ + +#define LED_MASK (ADS_LED0|ADS_LED1|ADS_LED2) + +void adsagc_leds_event(led_event_t evt) +{ + unsigned long flags; + + local_irq_save(flags); + + switch (evt) { + case led_start: + hw_led_state = 0; /* gc leds are positive logic */ + led_state = LED_STATE_ENABLED; + break; + + case led_stop: + led_state &= ~LED_STATE_ENABLED; + break; + + case led_claim: + led_state |= LED_STATE_CLAIMED; + hw_led_state = LED_MASK; + break; + + case led_release: + led_state &= ~LED_STATE_CLAIMED; + hw_led_state = LED_MASK; + break; + +#ifdef CONFIG_LEDS_TIMER + case led_timer: + if (!(led_state & LED_STATE_CLAIMED)) + hw_led_state ^= LED_TIMER; + break; +#endif + +#ifdef CONFIG_LEDS_CPU + case led_idle_start: + if (!(led_state & LED_STATE_CLAIMED)) + hw_led_state &= ~LED_IDLE; + break; + + case led_idle_end: + if (!(led_state & LED_STATE_CLAIMED)) + hw_led_state |= LED_IDLE; + break; +#endif + + case led_green_on: + break; + + case led_green_off: + break; + + case led_amber_on: + hw_led_state |= LED_USER; + break; + + case led_amber_off: + hw_led_state &= ~LED_USER; + break; + + case led_red_on: + break; + + case led_red_off: + break; + + default: + break; + } + + if (led_state & LED_STATE_ENABLED) { + + unsigned char ads_cr3 = ADS_CR3; + ads_cr3 &= ~LED_MASK; + ads_cr3 |= hw_led_state; + ADS_CR3 = ads_cr3; + } + + local_irq_restore(flags); +} diff -urN orig/arch/arm/mach-sa1100/leds-adsbitsy.c linux/arch/arm/mach-sa1100/leds-adsbitsy.c --- orig/arch/arm/mach-sa1100/leds-adsbitsy.c Fri Nov 16 10:09:48 2001 +++ linux/arch/arm/mach-sa1100/leds-adsbitsy.c Fri Sep 6 10:53:47 2002 @@ -28,7 +28,7 @@ { unsigned long flags; - save_flags_cli(flags); + local_irq_save(flags); switch (evt) { case led_start: @@ -92,5 +92,5 @@ GPCR = hw_led_state ^ LED_MASK; } - restore_flags(flags); + local_irq_restore(flags); } diff -urN orig/arch/arm/mach-sa1100/leds-adsbitsyplus.c linux/arch/arm/mach-sa1100/leds-adsbitsyplus.c --- orig/arch/arm/mach-sa1100/leds-adsbitsyplus.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-sa1100/leds-adsbitsyplus.c Fri Feb 21 15:17:33 2003 @@ -0,0 +1,97 @@ +/* + * linux/arch/arm/mach-sa1100/leds-adsbitsy.c + * + * ADS Bitsy LED + * 2/7/2003 Robert Whaley + * This file comes from leds-adsbitsy.c of Woojung Huh + */ +#include +#include + +#include +#include +#include + +#include "leds.h" + + +#define LED_STATE_ENABLED 1 +#define LED_STATE_CLAIMED 2 + +static unsigned int led_state; +static unsigned int hw_led_state; + +#define LED_TIMER GPIO_GPIO20 /* green heartbeat */ + +#define LED_MASK (LED_TIMER) + +void adsbitsyplus_leds_event(led_event_t evt) +{ + unsigned long flags; + + local_irq_save(flags); + + switch (evt) { + case led_start: + hw_led_state = 0; /* gc leds are positive logic */ + led_state = LED_STATE_ENABLED; + break; + + case led_stop: + led_state &= ~LED_STATE_ENABLED; + break; + + case led_claim: + led_state |= LED_STATE_CLAIMED; + hw_led_state = LED_MASK; + break; + + case led_release: + led_state &= ~LED_STATE_CLAIMED; + hw_led_state = LED_MASK; + break; + +#ifdef CONFIG_LEDS_TIMER + case led_timer: + if (!(led_state & LED_STATE_CLAIMED)) + hw_led_state ^= LED_TIMER; + break; +#endif + +#ifdef CONFIG_LEDS_CPU + case led_idle_start: + break; + + case led_idle_end: + break; +#endif + + case led_green_on: + break; + + case led_green_off: + break; + + case led_amber_on: + break; + + case led_amber_off: + break; + + case led_red_on: + break; + + case led_red_off: + break; + + default: + break; + } + + if (led_state & LED_STATE_ENABLED) { + GPSR = hw_led_state; + GPCR = hw_led_state ^ LED_MASK; + } + + local_irq_restore(flags); +} diff -urN orig/arch/arm/mach-sa1100/leds-frodo.c linux/arch/arm/mach-sa1100/leds-frodo.c --- orig/arch/arm/mach-sa1100/leds-frodo.c Mon Aug 5 13:29:46 2002 +++ linux/arch/arm/mach-sa1100/leds-frodo.c Thu Oct 24 13:01:38 2002 @@ -22,22 +22,22 @@ #include "leds.h" -#define led6_on() FRODO_CPLD_GENERAL |= FRODO_LED2 -#define led6_off() FRODO_CPLD_GENERAL &= ~FRODO_LED2 -#define led6_invert() do { \ - if ((FRODO_CPLD_GENERAL & FRODO_LED2)) \ - led6_off (); \ - else \ - led6_on (); \ +#define led6_on() frodo_cpld_set (FRODO_CPLD_GENERAL,FRODO_LED2) +#define led6_off() frodo_cpld_clear (FRODO_CPLD_GENERAL,FRODO_LED2) +#define led6_invert() do { \ + if ((frodo_cpld_read (FRODO_CPLD_GENERAL) & FRODO_LED2)) \ + led6_off (); \ + else \ + led6_on (); \ } while (0) -#define led7_on() FRODO_CPLD_GENERAL |= FRODO_LED1 -#define led7_off() FRODO_CPLD_GENERAL &= ~FRODO_LED1 +#define led7_on() frodo_cpld_set (FRODO_CPLD_GENERAL,FRODO_LED1) +#define led7_off() frodo_cpld_clear (FRODO_CPLD_GENERAL,FRODO_LED1) #define led7_invert() do { \ - if ((FRODO_CPLD_GENERAL & FRODO_LED1)) \ - led7_off (); \ - else \ - led7_on (); \ + if ((frodo_cpld_read (FRODO_CPLD_GENERAL) & FRODO_LED1)) \ + led7_off (); \ + else \ + led7_on (); \ } while (0) static int claimed; diff -urN orig/arch/arm/mach-sa1100/leds-graphicsclient.c linux/arch/arm/mach-sa1100/leds-graphicsclient.c --- orig/arch/arm/mach-sa1100/leds-graphicsclient.c Mon Sep 3 14:14:40 2001 +++ linux/arch/arm/mach-sa1100/leds-graphicsclient.c Fri Sep 6 10:54:23 2002 @@ -30,7 +30,7 @@ { unsigned long flags; - save_flags_cli(flags); + local_irq_save(flags); switch (evt) { case led_start: @@ -100,5 +100,5 @@ GPCR = hw_led_state ^ LED_MASK; } - restore_flags(flags); + local_irq_restore(flags); } diff -urN orig/arch/arm/mach-sa1100/leds-graphicsmaster.c linux/arch/arm/mach-sa1100/leds-graphicsmaster.c --- orig/arch/arm/mach-sa1100/leds-graphicsmaster.c Fri Oct 26 16:45:53 2001 +++ linux/arch/arm/mach-sa1100/leds-graphicsmaster.c Fri Sep 6 10:54:23 2002 @@ -30,7 +30,7 @@ { unsigned long flags; - save_flags_cli(flags); + local_irq_save(flags); switch (evt) { case led_start: @@ -100,5 +100,5 @@ GPCR = hw_led_state ^ LED_MASK; } - restore_flags(flags); + local_irq_restore(flags); } diff -urN orig/arch/arm/mach-sa1100/leds-hackkit.c linux/arch/arm/mach-sa1100/leds-hackkit.c --- orig/arch/arm/mach-sa1100/leds-hackkit.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-sa1100/leds-hackkit.c Thu Dec 12 22:37:25 2002 @@ -0,0 +1,113 @@ +/* + * linux/arch/arm/mach-sa1100/leds-hackkit.c + * + * based on leds-lart.c + * + * (C) Erik Mouw (J.A.K.Mouw@its.tudelft.nl), April 21, 2000 + * (C) Stefan Eletzhofer , 2002 + * + * The HackKit has two leds (GPIO 22/23). The red led (gpio 22) is used + * as cpu led, the green one is used as timer led. + */ +#include +#include + +#include +#include +#include + +#include "leds.h" + + +#define LED_STATE_ENABLED 1 +#define LED_STATE_CLAIMED 2 + +static unsigned int led_state; +static unsigned int hw_led_state; + +#define LED_GREEN GPIO_GPIO23 +#define LED_RED GPIO_GPIO22 +#define LED_MASK (LED_RED | LED_GREEN) + +void hackkit_leds_event(led_event_t evt) +{ + unsigned long flags; + + local_irq_save(flags); + + switch(evt) { + case led_start: + /* pin 22/23 are outputs */ + GPDR |= LED_MASK; + hw_led_state = LED_MASK; + led_state = LED_STATE_ENABLED; + break; + + case led_stop: + led_state &= ~LED_STATE_ENABLED; + break; + + case led_claim: + led_state |= LED_STATE_CLAIMED; + hw_led_state = LED_MASK; + break; + + case led_release: + led_state &= ~LED_STATE_CLAIMED; + hw_led_state = LED_MASK; + break; + +#ifdef CONFIG_LEDS_TIMER + case led_timer: + if (!(led_state & LED_STATE_CLAIMED)) + hw_led_state ^= LED_GREEN; + break; +#endif + +#ifdef CONFIG_LEDS_CPU + case led_idle_start: + /* The LART people like the LED to be off when the + system is idle... */ + if (!(led_state & LED_STATE_CLAIMED)) + hw_led_state &= ~LED_RED; + break; + + case led_idle_end: + /* ... and on if the system is not idle */ + if (!(led_state & LED_STATE_CLAIMED)) + hw_led_state |= LED_RED; + break; +#endif + + case led_red_on: + if (led_state & LED_STATE_CLAIMED) + hw_led_state &= ~LED_RED; + break; + + case led_red_off: + if (led_state & LED_STATE_CLAIMED) + hw_led_state |= LED_RED; + break; + + case led_green_on: + if (led_state & LED_STATE_CLAIMED) + hw_led_state &= ~LED_GREEN; + break; + + case led_green_off: + if (led_state & LED_STATE_CLAIMED) + hw_led_state |= LED_GREEN; + break; + + default: + break; + } + + /* Now set the GPIO state, or nothing will happen at all */ + if (led_state & LED_STATE_ENABLED) { + GPSR = hw_led_state; + GPCR = hw_led_state ^ LED_MASK; + } + + local_irq_restore(flags); +} diff -urN orig/arch/arm/mach-sa1100/leds.c linux/arch/arm/mach-sa1100/leds.c --- orig/arch/arm/mach-sa1100/leds.c Mon Aug 5 13:29:46 2002 +++ linux/arch/arm/mach-sa1100/leds.c Fri Feb 21 15:22:51 2003 @@ -15,8 +15,12 @@ static int __init sa1100_leds_init(void) { + if (machine_is_adsagc()) + leds_event = adsagc_leds_event; if (machine_is_adsbitsy()) leds_event = adsbitsy_leds_event; + if (machine_is_adsbitsyplus()) + leds_event = adsbitsyplus_leds_event; if (machine_is_assabet()) leds_event = assabet_leds_event; if (machine_is_consus()) @@ -33,6 +37,8 @@ leds_event = graphicsclient_leds_event; if (machine_is_graphicsmaster()) leds_event = graphicsmaster_leds_event; + if (machine_is_hackkit()) + leds_event = hackkit_leds_event; if (machine_is_lart()) leds_event = lart_leds_event; if (machine_is_pfs168()) diff -urN orig/arch/arm/mach-sa1100/leds.h linux/arch/arm/mach-sa1100/leds.h --- orig/arch/arm/mach-sa1100/leds.h Mon Aug 5 13:29:46 2002 +++ linux/arch/arm/mach-sa1100/leds.h Fri Feb 21 15:17:34 2003 @@ -1,12 +1,15 @@ +extern void adsagc_leds_event(led_event_t evt); +extern void adsbitsy_leds_event(led_event_t evt); +extern void adsbitsyplus_leds_event(led_event_t evt); extern void assabet_leds_event(led_event_t evt); extern void consus_leds_event(led_event_t evt); extern void brutus_leds_event(led_event_t evt); extern void cerf_leds_event(led_event_t evt); extern void flexanet_leds_event(led_event_t evt); +extern void frodo_leds_event(led_event_t evt); extern void graphicsclient_leds_event(led_event_t evt); +extern void graphicsmaster_leds_event(led_event_t evt); +extern void hackkit_leds_event(led_event_t evt); extern void lart_leds_event(led_event_t evt); extern void pfs168_leds_event(led_event_t evt); -extern void graphicsmaster_leds_event(led_event_t evt); -extern void adsbitsy_leds_event(led_event_t evt); extern void system3_leds_event(led_event_t evt); -extern void frodo_leds_event(led_event_t evt); diff -urN orig/arch/arm/mach-sa1100/nanoengine.c linux/arch/arm/mach-sa1100/nanoengine.c --- orig/arch/arm/mach-sa1100/nanoengine.c Mon Aug 5 13:29:46 2002 +++ linux/arch/arm/mach-sa1100/nanoengine.c Fri Jun 21 14:13:59 2002 @@ -16,22 +16,6 @@ #include "generic.h" -static void __init -fixup_nanoengine(struct machine_desc *desc, struct param_struct *params, - char **cmdline, struct meminfo *mi) -{ - SET_BANK( 0, 0xc0000000, 32*1024*1024 ); - mi->nr_banks = 1; - - ROOT_DEV = MKDEV(RAMDISK_MAJOR,0); - setup_ramdisk( 1, 0, 0, 8192 ); - setup_initrd( __phys_to_virt(0xc0800000), 4*1024*1024 ); - - /* Get command line parameters passed from the loader (if any) */ - if( *((char*)0xc0000100) ) - *cmdline = ((char *)0xc0000100); -} - static struct map_desc nanoengine_io_desc[] __initdata = { /* virtual physical length domain r w c b */ { 0xe8000000, 0x00000000, 0x02000000, DOMAIN_IO, 0, 1, 0, 0 }, /* Flash bank 0 */ @@ -56,7 +40,7 @@ MACHINE_START(NANOENGINE, "BSE nanoEngine") BOOT_MEM(0xc0000000, 0x80000000, 0xf8000000) - FIXUP(fixup_nanoengine) + BOOT_PARAMS(0xc0000100) MAPIO(nanoengine_map_io) INITIRQ(sa1100_init_irq) MACHINE_END diff -urN orig/arch/arm/mach-sa1100/neponset.c linux/arch/arm/mach-sa1100/neponset.c --- orig/arch/arm/mach-sa1100/neponset.c Mon Aug 5 13:29:46 2002 +++ linux/arch/arm/mach-sa1100/neponset.c Thu Feb 6 11:33:12 2003 @@ -10,6 +10,8 @@ #include #include #include +#include +#include #include #include @@ -54,9 +56,9 @@ } static struct irqaction neponset_irq = { - name: "Neponset", - handler: neponset_IRQ_demux, - flags: SA_INTERRUPT + .name = "Neponset", + .handler = neponset_IRQ_demux, + .flags = SA_INTERRUPT }; static void __init neponset_init_irq(void) @@ -214,8 +216,8 @@ } static struct sa1100_port_fns neponset_port_fns __initdata = { - set_mctrl: neponset_set_mctrl, - get_mctrl: neponset_get_mctrl, + .set_mctrl = neponset_set_mctrl, + .get_mctrl = neponset_get_mctrl, }; void __init neponset_map_io(void) diff -urN orig/arch/arm/mach-sa1100/pcipool.c linux/arch/arm/mach-sa1100/pcipool.c --- orig/arch/arm/mach-sa1100/pcipool.c Mon Sep 3 14:14:40 2001 +++ linux/arch/arm/mach-sa1100/pcipool.c Thu Dec 12 22:33:33 2002 @@ -20,8 +20,6 @@ #include -#include "pcipool.h" - /* * Pool allocator ... wraps the pci_alloc_consistent page allocator, so * small blocks are easily used by drivers for bus mastering controllers. @@ -33,7 +31,6 @@ spinlock_t lock; size_t blocks_per_page; size_t size; - int flags; struct pci_dev *dev; size_t allocation; char name [32]; @@ -52,6 +49,19 @@ // #define CONFIG_PCIPOOL_DEBUG +static inline const char *slot_name(const struct pci_pool *pool) +{ + const struct pci_dev *pdev = pool->dev; + + if (pdev == 0) + return "[0]"; + + else if (dev_is_sa1111(pdev)) + return "[SA-1111]"; + else + return pdev->slot_name; +} + /** * pci_pool_create - Creates a pool of pci consistent memory blocks, for dma. @@ -60,7 +70,7 @@ * @size: size of the blocks in this pool. * @align: alignment requirement for blocks; must be a power of two * @allocation: returned blocks won't cross this boundary (or zero) - * @flags: SLAB_* flags (not all are supported). + * @mem_flags: SLAB_* flags. * * Returns a pci allocation pool with the requested characteristics, or * null if one can't be created. Given one of these pools, pci_pool_alloc() @@ -76,7 +86,7 @@ */ struct pci_pool * pci_pool_create (const char *name, struct pci_dev *pdev, - size_t size, size_t align, size_t allocation, int flags) + size_t size, size_t align, size_t allocation, int mem_flags) { struct pci_pool *retval; @@ -100,13 +110,9 @@ } else if (allocation < size) return 0; - if (!(retval = kmalloc (sizeof *retval, flags))) + if (!(retval = kmalloc (sizeof *retval, mem_flags))) return retval; -#ifdef CONFIG_PCIPOOL_DEBUG - flags |= SLAB_POISON; -#endif - strncpy (retval->name, name, sizeof retval->name); retval->name [sizeof retval->name - 1] = 0; @@ -114,14 +120,13 @@ INIT_LIST_HEAD (&retval->page_list); spin_lock_init (&retval->lock); retval->size = size; - retval->flags = flags; retval->allocation = allocation; retval->blocks_per_page = allocation / size; init_waitqueue_head (&retval->waitq); #ifdef CONFIG_PCIPOOL_DEBUG printk (KERN_DEBUG "pcipool create %s/%s size %d, %d/page (%d alloc)\n", - pdev ? pdev->slot_name : NULL, retval->name, size, + slot_name(retval), retval->name, size, retval->blocks_per_page, allocation); #endif @@ -143,11 +148,13 @@ if (!page) return 0; page->vaddr = pci_alloc_consistent (pool->dev, - pool->allocation, &page->dma); + pool->allocation, + &page->dma); if (page->vaddr) { memset (page->bitmap, 0xff, mapsize); // bit set == free - if (pool->flags & SLAB_POISON) - memset (page->vaddr, POOL_POISON_BYTE, pool->allocation); +#ifdef CONFIG_DEBUG_SLAB + memset (page->vaddr, POOL_POISON_BYTE, pool->allocation); +#endif list_add (&page->page_list, &pool->page_list); } else { kfree (page); @@ -173,8 +180,9 @@ { dma_addr_t dma = page->dma; - if (pool->flags & SLAB_POISON) - memset (page->vaddr, POOL_POISON_BYTE, pool->allocation); +#ifdef CONFIG_DEBUG_SLAB + memset (page->vaddr, POOL_POISON_BYTE, pool->allocation); +#endif pci_free_consistent (pool->dev, pool->allocation, page->vaddr, dma); list_del (&page->page_list); kfree (page); @@ -195,8 +203,7 @@ #ifdef CONFIG_PCIPOOL_DEBUG printk (KERN_DEBUG "pcipool destroy %s/%s\n", - pool->dev ? pool->dev->slot_name : NULL, - pool->name); + slot_name(pool), pool->name); #endif spin_lock_irqsave (&pool->lock, flags); @@ -206,8 +213,7 @@ struct pci_page, page_list); if (is_page_busy (pool->blocks_per_page, page->bitmap)) { printk (KERN_ERR "pci_pool_destroy %s/%s, %p busy\n", - pool->dev ? pool->dev->slot_name : NULL, - pool->name, page->vaddr); + slot_name(pool), pool->name, page->vaddr); /* leak the still-in-use consistent memory */ list_del (&page->page_list); kfree (page); @@ -327,35 +333,32 @@ int map, block; if ((page = pool_find_page (pool, dma)) == 0) { - printk (KERN_ERR "pci_pool_free %s/%s, %p/%x (bad dma)\n", + printk (KERN_ERR "pci_pool_free %s/%s, %p/%lx (bad dma)\n", pool->dev ? pool->dev->slot_name : NULL, - pool->name, vaddr, dma); + pool->name, vaddr, (unsigned long) dma); return; } -#ifdef CONFIG_PCIPOOL_DEBUG - if (((dma - page->dma) + (void *)page->vaddr) != vaddr) { - printk (KERN_ERR "pci_pool_free %s/%s, %p (bad vaddr)/%x\n", - pool->dev ? pool->dev->slot_name : NULL, - pool->name, vaddr, dma); - return; - } -#endif block = dma - page->dma; block /= pool->size; map = block / BITS_PER_LONG; block %= BITS_PER_LONG; -#ifdef CONFIG_PCIPOOL_DEBUG +#ifdef CONFIG_DEBUG_SLAB + if (((dma - page->dma) + (void *)page->vaddr) != vaddr) { + printk (KERN_ERR "pci_pool_free %s/%s, %p (bad vaddr)/%lx\n", + pool->dev ? pool->dev->slot_name : NULL, + pool->name, vaddr, (unsigned long) dma); + return; + } if (page->bitmap [map] & (1UL << block)) { printk (KERN_ERR "pci_pool_free %s/%s, dma %x already free\n", pool->dev ? pool->dev->slot_name : NULL, pool->name, dma); return; } + memset (vaddr, POOL_POISON_BYTE, pool->size); #endif - if (pool->flags & SLAB_POISON) - memset (vaddr, POOL_POISON_BYTE, pool->size); spin_lock_irqsave (&pool->lock, flags); set_bit (block, &page->bitmap [map]); @@ -375,3 +378,14 @@ EXPORT_SYMBOL (pci_pool_alloc); EXPORT_SYMBOL (pci_pool_free); +/* **************************************** */ + +static int __init pcipool_init(void) +{ + MOD_INC_USE_COUNT; /* never unload */ + + return 0; +} +module_init(pcipool_init); + +MODULE_LICENSE("GPL"); diff -urN orig/arch/arm/mach-sa1100/pcipool.h linux/arch/arm/mach-sa1100/pcipool.h --- orig/arch/arm/mach-sa1100/pcipool.h Mon Sep 3 14:14:40 2001 +++ linux/arch/arm/mach-sa1100/pcipool.h Thu Jan 1 01:00:00 1970 @@ -1,8 +0,0 @@ - -struct pci_pool *pci_pool_create (const char *name, struct pci_dev *dev, - size_t size, size_t align, size_t allocation, int flags); -void pci_pool_destroy (struct pci_pool *pool); - -void *pci_pool_alloc (struct pci_pool *pool, int flags, dma_addr_t *handle); -void pci_pool_free (struct pci_pool *pool, void *vaddr, dma_addr_t addr); - diff -urN orig/arch/arm/mach-sa1100/pm.c linux/arch/arm/mach-sa1100/pm.c --- orig/arch/arm/mach-sa1100/pm.c Mon Aug 5 13:29:46 2002 +++ linux/arch/arm/mach-sa1100/pm.c Mon Aug 5 13:41:24 2002 @@ -18,11 +18,13 @@ * 2001-08-29: Nicolas Pitre * Cleaned up, pushed platform dependent stuff * in the platform specific files. + * + * 2002-05-27: Nicolas Pitre Killed sleep.h and the kmalloced save array. + * Storage is local on the stack now. */ #include #include #include -#include #include #include #include @@ -33,7 +35,6 @@ #include #include -#include "sleep.h" /* * Debug macros @@ -43,20 +44,32 @@ extern void sa1100_cpu_suspend(void); extern void sa1100_cpu_resume(void); -extern unsigned long *sleep_save; /* virtual address */ -extern unsigned long sleep_save_p; /* physical address */ - #define SAVE(x) sleep_save[SLEEP_SAVE_##x] = x #define RESTORE(x) x = sleep_save[SLEEP_SAVE_##x] +/* + * List of global SA11x0 peripheral registers to preserve. + * More ones like CP and general purpose register values are preserved + * with the stack location in sleep.S. + */ +enum { SLEEP_SAVE_START = 0, + + SLEEP_SAVE_OSCR, SLEEP_SAVE_OIER, + SLEEP_SAVE_OSMR0, SLEEP_SAVE_OSMR1, SLEEP_SAVE_OSMR2, SLEEP_SAVE_OSMR3, + + SLEEP_SAVE_GPDR, SLEEP_SAVE_GRER, SLEEP_SAVE_GFER, SLEEP_SAVE_GAFR, + SLEEP_SAVE_PPDR, SLEEP_SAVE_PPSR, SLEEP_SAVE_PPAR, SLEEP_SAVE_PSDR, + + SLEEP_SAVE_ICMR, + SLEEP_SAVE_Ser1SDCR0, + + SLEEP_SAVE_SIZE +}; + + int pm_do_suspend(void) { - /* set up pointer to sleep parameters */ - sleep_save = kmalloc(SLEEP_SAVE_SIZE*sizeof(long), GFP_ATOMIC); - if (!sleep_save) - return -ENOMEM; - - sleep_save_p = virt_to_phys(sleep_save); + unsigned long sleep_save[SLEEP_SAVE_SIZE]; cli(); @@ -144,8 +157,6 @@ sti(); - kfree (sleep_save); - /* * Restore the CPU frequency settings. */ @@ -154,6 +165,11 @@ #endif return 0; +} + +unsigned long sleep_phys_sp(void *sp) +{ + return virt_to_phys(sp); } #ifdef CONFIG_SYSCTL diff -urN orig/arch/arm/mach-sa1100/sa1100_usb.h linux/arch/arm/mach-sa1100/sa1100_usb.h --- orig/arch/arm/mach-sa1100/sa1100_usb.h Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-sa1100/sa1100_usb.h Tue Oct 16 20:02:23 2001 @@ -0,0 +1,193 @@ +/* + * sa1100_usb.h + * + * Public interface to the sa1100 USB core. For use by client modules + * like usb-eth and usb-char. + * + */ + +#ifndef _SA1100_USB_H +#define _SA1100_USB_H +#include + +typedef void (*usb_callback_t)(int flag, int size); + +/* in usb_ctl.c (see also descriptor methods at bottom of file) */ + +// Open the USB client for client and initialize data structures +// to default values, but _do not_ start UDC. +int sa1100_usb_open( const char * client_name ); + +// Start UDC running +int sa1100_usb_start( void ); + +// Immediately stop udc, fire off completion routines w/-EINTR +int sa1100_usb_stop( void ) ; + +// Disconnect client from usb core +int sa1100_usb_close( void ) ; + +// set notify callback for when core reaches configured state +// return previous pointer (if any) +typedef void (*usb_notify_t)(void); +usb_notify_t sa1100_set_configured_callback( usb_notify_t callback ); + +/* in usb_send.c */ +int sa1100_usb_xmitter_avail( void ); +int sa1100_usb_send(char *buf, int len, usb_callback_t callback); +void sa1100_usb_send_reset(void); + +/* in usb_recev.c */ +int sa1100_usb_recv(char *buf, int len, usb_callback_t callback); +void sa1100_usb_recv_reset(void); + +////////////////////////////////////////////////////////////////////////////// +// Descriptor Management +////////////////////////////////////////////////////////////////////////////// + +#define DescriptorHeader \ + __u8 bLength; \ + __u8 bDescriptorType + + +// --- Device Descriptor ------------------- + +typedef struct { + DescriptorHeader; + __u16 bcdUSB; /* USB specification revision number in BCD */ + __u8 bDeviceClass; /* USB class for entire device */ + __u8 bDeviceSubClass; /* USB subclass information for entire device */ + __u8 bDeviceProtocol; /* USB protocol information for entire device */ + __u8 bMaxPacketSize0; /* Max packet size for endpoint zero */ + __u16 idVendor; /* USB vendor ID */ + __u16 idProduct; /* USB product ID */ + __u16 bcdDevice; /* vendor assigned device release number */ + __u8 iManufacturer; /* index of manufacturer string */ + __u8 iProduct; /* index of string that describes product */ + __u8 iSerialNumber; /* index of string containing device serial number */ + __u8 bNumConfigurations; /* number fo configurations */ +} __attribute__ ((packed)) device_desc_t; + +// --- Configuration Descriptor ------------ + +typedef struct { + DescriptorHeader; + __u16 wTotalLength; /* total # of bytes returned in the cfg buf 4 this cfg */ + __u8 bNumInterfaces; /* number of interfaces in this cfg */ + __u8 bConfigurationValue; /* used to uniquely ID this cfg */ + __u8 iConfiguration; /* index of string describing configuration */ + __u8 bmAttributes; /* bitmap of attributes for ths cfg */ + __u8 MaxPower; /* power draw in 2ma units */ +} __attribute__ ((packed)) config_desc_t; + +// bmAttributes: +enum { USB_CONFIG_REMOTEWAKE=0x20, USB_CONFIG_SELFPOWERED=0x40, + USB_CONFIG_BUSPOWERED=0x80 }; +// MaxPower: +#define USB_POWER( x) ((x)>>1) /* convert mA to descriptor units of A for MaxPower */ + +// --- Interface Descriptor --------------- + +typedef struct { + DescriptorHeader; + __u8 bInterfaceNumber; /* Index uniquely identfying this interface */ + __u8 bAlternateSetting; /* ids an alternate setting for this interface */ + __u8 bNumEndpoints; /* number of endpoints in this interface */ + __u8 bInterfaceClass; /* USB class info applying to this interface */ + __u8 bInterfaceSubClass; /* USB subclass info applying to this interface */ + __u8 bInterfaceProtocol; /* USB protocol info applying to this interface */ + __u8 iInterface; /* index of string describing interface */ +} __attribute__ ((packed)) intf_desc_t; + +// --- Endpoint Descriptor --------------- + +typedef struct { + DescriptorHeader; + __u8 bEndpointAddress; /* 0..3 ep num, bit 7: 0 = 0ut 1= in */ + __u8 bmAttributes; /* 0..1 = 0: ctrl, 1: isoc, 2: bulk 3: intr */ + __u16 wMaxPacketSize; /* data payload size for this ep in this cfg */ + __u8 bInterval; /* polling interval for this ep in this cfg */ +} __attribute__ ((packed)) ep_desc_t; + +// bEndpointAddress: +enum { USB_OUT= 0, USB_IN=1 }; +#define USB_EP_ADDRESS(a,d) (((a)&0xf) | ((d) << 7)) +// bmAttributes: +enum { USB_EP_CNTRL=0, USB_EP_BULK=2, USB_EP_INT=3 }; + +// --- String Descriptor ------------------- + +typedef struct { + DescriptorHeader; + __u16 bString[1]; /* unicode string .. actaully 'n' __u16s */ +} __attribute__ ((packed)) string_desc_t; + +/*======================================================= + * Handy helpers when working with above + * + */ +// these are x86-style 16 bit "words" ... +#define make_word_c( w ) __constant_cpu_to_le16(w) +#define make_word( w ) __cpu_to_le16(w) + +// descriptor types +enum { USB_DESC_DEVICE=1, USB_DESC_CONFIG=2, USB_DESC_STRING=3, + USB_DESC_INTERFACE=4, USB_DESC_ENDPOINT=5 }; + + +/*======================================================= + * Default descriptor layout for SA-1100 and SA-1110 UDC + */ + +/* "config descriptor buffer" - that is, one config, + ..one interface and 2 endpoints */ +struct cdb { + config_desc_t cfg; + intf_desc_t intf; + ep_desc_t ep1; + ep_desc_t ep2; +} __attribute__ ((packed)); + + +/* all SA device descriptors */ +typedef struct { + device_desc_t dev; /* device descriptor */ + struct cdb b; /* bundle of descriptors for this cfg */ +} __attribute__ ((packed)) desc_t; + + +/*======================================================= + * Descriptor API + */ + +/* Get the address of the statically allocated desc_t structure + in the usb core driver. Clients can modify this between + the time they call sa1100_usb_open() and sa1100_usb_start() +*/ +desc_t * +sa1100_usb_get_descriptor_ptr( void ); + + +/* Set a pointer to the string descriptor at "index". The driver + ..has room for 8 string indicies internally. Index zero holds + ..a LANGID code and is set to US English by default. Inidices + ..1-7 are available for use in the config descriptors as client's + ..see fit. This pointer is assumed to be good as long as the + ..SA usb core is open (so statically allocate them). Returnes -EINVAL + ..if index out of range */ +int sa1100_usb_set_string_descriptor( int index, string_desc_t * p ); + +/* reverse of above */ +string_desc_t * +sa1100_usb_get_string_descriptor( int index ); + +/* kmalloc() a string descriptor and convert "p" to unicode in it */ +string_desc_t * +sa1100_usb_kmalloc_string_descriptor( const char * p ); + + + + + + +#endif /* _SA1100_USB_H */ diff -urN orig/arch/arm/mach-sa1100/sa1111-ohci.c linux/arch/arm/mach-sa1100/sa1111-ohci.c --- orig/arch/arm/mach-sa1100/sa1111-ohci.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-sa1100/sa1111-ohci.c Wed Feb 27 11:37:49 2002 @@ -0,0 +1,140 @@ +#include +#include +#include +#include + +#ifdef CONFIG_USB_OHCI + +/* + * The SA-1111 errata says that the DMA hardware needs to be exercised + * before the clocks are turned on to work properly. This code does + * a tiny dma transfer to prime to hardware. + * + * What DMA errata? I've checked October 1999 and February 2001, both + * of which do not mention such a bug, let alone any details of this + * work-around. + */ +static void __init sa1111_dma_setup(void) +{ + dma_addr_t dma_buf; + void * vbuf; + + /* DMA init & setup */ + + /* WARNING: The SA-1111 L3 function is used as part of this + * SA-1111 DMA errata workaround. + * + * N.B., When the L3 function is enabled, it uses GPIO_B<4:5> + * and takes precedence over the PS/2 mouse and GPIO_B + * functions. Refer to "Intel StrongARM SA-1111 Microprocessor + * Companion Chip, Sect 10.2" for details. So this "fix" may + * "break" support of either PS/2 mouse or GPIO_B if + * precautions are not taken to avoid collisions in + * configuration and use of these pins. AFAIK, no precautions + * are taken at this time. So it is likely that the action + * taken here may cause problems in PS/2 mouse and/or GPIO_B + * pin use elsewhere. + * + * But wait, there's more... What we're doing here is + * obviously altogether a bad idea. We're indiscrimanately bit + * flipping config for a few different functions here which + * are "owned" by other drivers. This needs to be handled + * better than it is being done here at this time. */ + + /* prime the dma engine with a tiny dma */ + SKPCR |= SKPCR_I2SCLKEN; + SKAUD |= SKPCR_L3CLKEN | SKPCR_SCLKEN; + + SACR0 |= 0x00003305; + SACR1 = 0x00000000; + + /* + * We need memory below 1MB. + * NOTE: consistent_alloc gives you some random virtual + * address as its return value, and the DMA address via + * the dma_addr_t pointer. + */ + vbuf = consistent_alloc(GFP_KERNEL | GFP_DMA, 4, &dma_buf); + + SADTSA = (unsigned long)dma_buf; + SADTCA = 4; + + SADTCS |= 0x00000011; + SKPCR |= SKPCR_DCLKEN; + + /* wait */ + udelay(100); + + /* clear reserved but, then disable SAC */ + SACR0 &= ~(0x00000002); + SACR0 &= ~(0x00000001); + + /* toggle bit clock direction */ + SACR0 |= 0x00000004; + SACR0 &= ~(0x00000004); + + SKAUD &= ~(SKPCR_L3CLKEN | SKPCR_SCLKEN); + + SKPCR &= ~SKPCR_I2SCLKEN; + + consistent_free(vbuf, 4, dma_buf); +} + +/* + * reset the SA-1111 usb controller and turn on it's clocks + */ +int __init sa1111_ohci_hcd_init(void) +{ + unsigned int usb_reset = 0; + + if (machine_is_xp860() || + machine_has_neponset() || + machine_is_pfs168() || + machine_is_badge4()) + usb_reset = USB_RESET_PWRSENSELOW | USB_RESET_PWRCTRLLOW; + + /* + * turn on USB clocks + */ + SKPCR |= SKPCR_UCLKEN; + udelay(100); + + /* + * Force USB reset + */ + USB_RESET = USB_RESET_FORCEIFRESET; + USB_RESET |= USB_RESET_FORCEHCRESET; + udelay(100); + + /* + * Take out of reset + */ + USB_RESET = 0; + + /* + * set power sense and control lines (this from the diags code) + */ + USB_RESET = usb_reset; + + /* + * Huh? This is a _read only_ register --rmk + */ + USB_STATUS = 0; + + udelay(10); + + /* + * compensate for dma bug + */ + sa1111_dma_setup(); + + return 0; +} + +void sa1111_ohci_hcd_cleanup(void) +{ + /* turn the USB clock off */ + SKPCR &= ~SKPCR_UCLKEN; +} + +#endif /* CONFIG_USB_OHCI */ diff -urN orig/arch/arm/mach-sa1100/sa1111-pcibuf.c linux/arch/arm/mach-sa1100/sa1111-pcibuf.c --- orig/arch/arm/mach-sa1100/sa1111-pcibuf.c Mon Aug 5 13:29:46 2002 +++ linux/arch/arm/mach-sa1100/sa1111-pcibuf.c Fri Jan 24 16:20:58 2003 @@ -1,272 +1,400 @@ /* * linux/arch/arm/mach-sa1100/pci-sa1111.c * - * Special pci_map/unmap_single routines for SA-1111. These functions - * compensate for a bug in the SA-1111 hardware which don't allow DMA - * to/from addresses above 1MB. + * Special pci_{map/unmap/dma_sync}_* routines for SA-1111. * - * Brad Parker (brad@heeltoe.com) + * These functions utilize bouncer buffers to compensate for a bug in + * the SA-1111 hardware which don't allow DMA to/from addresses + * certain addresses above 1MB. * - * This program is free software; you can redistribute it and/or modify - * it under the terms of the GNU General Public License version 2 as - * published by the Free Software Foundation. + * Re-written by Christopher Hoover + * Original version by Brad Parker (brad@heeltoe.com) * - * 06/13/2001 - created. - */ + * Copyright (C) 2002 Hewlett Packard Company. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * version 2 as published by the Free Software Foundation. + * */ + #include #include #include #include +#include +#include +//#define DEBUG +#ifdef DEBUG +#define DPRINTK(...) do { printk(KERN_DEBUG __VA_ARGS__); } while (0) +#else +#define DPRINTK(...) do { } while (0) +#endif + +struct safe_buffer { + struct list_head node; + + /* original request */ + void *ptr; + size_t size; + int direction; -#include "pcipool.h" + /* safe buffer info */ + struct pci_pool *pool; + void *safe; + dma_addr_t safe_dma_addr; +}; -/* - * simple buffer allocator for copying of unsafe to safe buffers - * uses __alloc/__free for actual buffers - * keeps track of safe buffers we've allocated so we can recover the - * unsafe buffers. - */ +LIST_HEAD(safe_buffers); -#define MAX_SAFE 32 #define SIZE_SMALL 1024 #define SIZE_LARGE (16*1024) -static long mapped_alloc_size; -static char *safe_buffers[MAX_SAFE][2]; - - -static struct pci_pool *small_buffer_cache, *large_buffer_cache; +static struct pci_pool *small_buffer_pool, *large_buffer_pool; -static int -init_safe_buffers(struct pci_dev *dev) +static int __init +create_safe_buffer_pools(void) { - small_buffer_cache = pci_pool_create("pci_small_buffer", - dev, + small_buffer_pool = pci_pool_create("sa1111_small_dma_buffer", + SA1111_FAKE_PCIDEV, SIZE_SMALL, 0 /* byte alignment */, 0 /* no page-crossing issues */, - GFP_KERNEL | GFP_DMA); - - if (small_buffer_cache == 0) + SLAB_KERNEL); + if (0 == small_buffer_pool) { + printk(KERN_ERR + "sa1111_pcibuf: could not allocate small pci pool\n"); return -1; + } - large_buffer_cache = pci_pool_create("pci_large_buffer", - dev, + large_buffer_pool = pci_pool_create("sa1111_large_dma_buffer", + SA1111_FAKE_PCIDEV, SIZE_LARGE, 0 /* byte alignment */, 0 /* no page-crossing issues */, - GFP_KERNEL | GFP_DMA); - if (large_buffer_cache == 0) + SLAB_KERNEL); + if (0 == large_buffer_pool) { + printk(KERN_ERR + "sa1111_pcibuf: could not allocate large pci pool\n"); + pci_pool_destroy(small_buffer_pool); + small_buffer_pool = 0; return -1; + } return 0; } +static void __exit +destroy_safe_buffer_pools(void) +{ + if (small_buffer_pool) + pci_pool_destroy(small_buffer_pool); + if (large_buffer_pool) + pci_pool_destroy(large_buffer_pool); + + small_buffer_pool = large_buffer_pool = 0; +} + + /* allocate a 'safe' buffer and keep track of it */ -static char * -alloc_safe_buffer(char *unsafe, int size, dma_addr_t *pbus) +static struct safe_buffer * +alloc_safe_buffer(void *ptr, size_t size, int direction) { - char *safe; - dma_addr_t busptr; + struct safe_buffer *buf; struct pci_pool *pool; - int i; - - if (0) printk("alloc_safe_buffer(size=%d)\n", size); + void *safe; + dma_addr_t safe_dma_addr; - if (size <= SIZE_SMALL) - pool = small_buffer_cache; - else - if (size < SIZE_LARGE) - pool = large_buffer_cache; - else - return 0; + DPRINTK("%s(ptr=%p, size=%d, direction=%d)\n", + __func__, ptr, size, direction); - safe = pci_pool_alloc(pool, SLAB_ATOMIC, &busptr); - if (safe == 0) + buf = kmalloc(sizeof(struct safe_buffer), GFP_ATOMIC); + if (buf == 0) { + printk(KERN_WARNING "%s: kmalloc failed\n", __func__); return 0; + } - for (i = 0; i < MAX_SAFE; i++) - if (safe_buffers[i][0] == 0) { - break; - } - - if (i == MAX_SAFE) { - panic(__FILE__ ": exceeded MAX_SAFE buffers"); + if (size <= SIZE_SMALL) { + pool = small_buffer_pool; + safe = pci_pool_alloc(pool, GFP_ATOMIC, &safe_dma_addr); + } else if (size <= SIZE_LARGE) { + pool = large_buffer_pool; + safe = pci_pool_alloc(pool, GFP_ATOMIC, &safe_dma_addr); + } else { + printk(KERN_DEBUG + "sa111_pcibuf: resorting to pci_alloc_consistent\n"); + pool = 0; + safe = pci_alloc_consistent(SA1111_FAKE_PCIDEV, size, + &safe_dma_addr); } - /* place the size index and the old buffer ptr in the first 8 bytes - * and return a ptr + 12 to caller - */ - ((int *)safe)[0] = i; - ((char **)safe)[1] = (char *)pool; - ((char **)safe)[2] = unsafe; + if (safe == 0) { + printk(KERN_WARNING + "%s: could not alloc dma memory (size=%d)\n", + __func__, size); + kfree(buf); + return 0; + } - busptr += sizeof(int) + sizeof(char *) + sizeof(char *); + BUG_ON(sa1111_check_dma_bug(safe_dma_addr)); // paranoia - safe_buffers[i][0] = (void *)busptr; - safe_buffers[i][1] = (void *)safe; + buf->ptr = ptr; + buf->size = size; + buf->direction = direction; + buf->pool = pool; + buf->safe = safe; + buf->safe_dma_addr = safe_dma_addr; - safe += sizeof(int) + sizeof(char *) + sizeof(char *); + MOD_INC_USE_COUNT; + list_add(&buf->node, &safe_buffers); - *pbus = busptr; - return safe; + return buf; } /* determine if a buffer is from our "safe" pool */ -static char * -find_safe_buffer(char *busptr, char **unsafe) +static struct safe_buffer * +find_safe_buffer(dma_addr_t safe_dma_addr) { - int i; - char *buf; + struct list_head *entry; - for (i = 0; i < MAX_SAFE; i++) { - if (safe_buffers[i][0] == busptr) { - if (0) printk("find_safe_buffer(%p) found @ %d\n", busptr, i); - buf = safe_buffers[i][1]; - *unsafe = ((char **)buf)[2]; - return buf + sizeof(int) + sizeof(char *) + sizeof(char *); + list_for_each(entry, &safe_buffers) { + struct safe_buffer *b = + list_entry(entry, struct safe_buffer, node); + + if (b->safe_dma_addr == safe_dma_addr) { + return b; } } - return (char *)0; + return 0; } static void -free_safe_buffer(char *buf) +free_safe_buffer(struct safe_buffer *buf) { - int index; - struct pci_pool *pool; - char *dma; + DPRINTK("%s(buf=%p)\n", __func__, buf); - if (0) printk("free_safe_buffer(buf=%p)\n", buf); + list_del(&buf->node); - /* retrieve the buffer size index */ - buf -= sizeof(int) + sizeof(char*) + sizeof(char*); - index = ((int *)buf)[0]; - pool = (struct pci_pool *)((char **)buf)[1]; + if (buf->pool) + pci_pool_free(buf->pool, buf->safe, buf->safe_dma_addr); + else + pci_free_consistent(SA1111_FAKE_PCIDEV, buf->size, buf->safe, + buf->safe_dma_addr); + kfree(buf); - if (0) printk("free_safe_buffer(%p) index %d\n", - buf, index); + MOD_DEC_USE_COUNT; +} - if (index < 0 || index >= MAX_SAFE) { - printk(__FILE__ ": free_safe_buffer() corrupt buffer\n"); - return; - } +static inline int +dma_range_is_safe(dma_addr_t addr, size_t size) +{ + unsigned int physaddr = SA1111_DMA_ADDR((unsigned int) addr); - dma = safe_buffers[index][0]; - safe_buffers[index][0] = 0; + /* Any address within one megabyte of the start of the target + * bank will be OK. This is an overly conservative test: + * other addresses can be OK depending on the dram + * configuration. (See sa1111.c:sa1111_check_dma_bug() * for + * details.) + * + * We take care to ensure the entire dma region is within + * the safe range. + */ - pci_pool_free(pool, buf, (u32)dma); + return ((physaddr + size - 1) < (1<<20)); } /* - NOTE: - replace pci_map/unmap_single with local routines which will - do buffer copies if buffer is above 1mb... -*/ - -/* * see if a buffer address is in an 'unsafe' range. if it is * allocate a 'safe' buffer and copy the unsafe buffer into it. * substitute the safe buffer for the unsafe one. * (basically move the buffer from an unsafe area to a safe one) - * - * we assume calls to map_single are symmetric with calls to unmap_single... */ dma_addr_t -sa1111_map_single(struct pci_dev *hwdev, void *virtptr, - size_t size, int direction) +sa1111_map_single(void *ptr, size_t size, int direction) { - dma_addr_t busptr; + unsigned long flags; + dma_addr_t dma_addr; - mapped_alloc_size += size; + DPRINTK("%s(ptr=%p,size=%d,dir=%x)\n", + __func__, ptr, size, direction); - if (0) printk("pci_map_single(hwdev=%p,ptr=%p,size=%d,dir=%x) " - "alloced=%ld\n", - hwdev, virtptr, size, direction, mapped_alloc_size); + BUG_ON(direction == PCI_DMA_NONE); - busptr = virt_to_bus(virtptr); + local_irq_save(flags); - /* we assume here that a buffer will never be >=64k */ - if ( (((unsigned long)busptr) & 0x100000) || - ((((unsigned long)busptr)+size) & 0x100000) ) - { - char *safe; + dma_addr = virt_to_bus(ptr); - safe = alloc_safe_buffer(virtptr, size, &busptr); - if (safe == 0) { - printk("unable to map unsafe buffer %p!\n", virtptr); + if (!dma_range_is_safe(dma_addr, size)) { + struct safe_buffer *buf; + + buf = alloc_safe_buffer(ptr, size, direction); + if (buf == 0) { + printk(KERN_ERR + "%s: unable to map unsafe buffer %p!\n", + __func__, ptr); + local_irq_restore(flags); return 0; } - if (0) printk("unsafe buffer %p (phy=%p) mapped to %p (phy=%p)\n", - virtptr, (void *)virt_to_bus(virtptr), - safe, (void *)busptr); - - memcpy(safe, virtptr, size); - consistent_sync(safe, size, direction); + DPRINTK("%s: unsafe buffer %p (phy=%p) mapped to %p (phy=%p)\n", + __func__, + buf->ptr, (void *) virt_to_bus(buf->ptr), + buf->safe, (void *) buf->safe_dma_addr); + + if ((direction == PCI_DMA_TODEVICE) || + (direction == PCI_DMA_BIDIRECTIONAL)) { + DPRINTK("%s: copy out from unsafe %p, to safe %p, size %d\n", + __func__, ptr, buf->safe, size); + memcpy(buf->safe, ptr, size); + } + consistent_sync(buf->safe, size, direction); - return busptr; + dma_addr = buf->safe_dma_addr; + } else { + consistent_sync(ptr, size, direction); } - consistent_sync(virtptr, size, direction); - return busptr; + local_irq_restore(flags); + return dma_addr; } /* - * see if a mapped address was really a "safe" buffer and if so, - * copy the data from the safe buffer back to the unsafe buffer - * and free up the safe buffer. - * (basically return things back to the way they should be) + * see if a mapped address was really a "safe" buffer and if so, copy + * the data from the safe buffer back to the unsafe buffer and free up + * the safe buffer. (basically return things back to the way they + * should be) */ + void -sa1111_unmap_single(struct pci_dev *hwdev, dma_addr_t dma_addr, - size_t size, int direction) +sa1111_unmap_single(dma_addr_t dma_addr, size_t size, int direction) { - char *safe, *unsafe; - void *buf; + unsigned long flags; + struct safe_buffer *buf; - /* hack; usb-ohci.c never sends hwdev==NULL, all others do */ - if (hwdev == NULL) { - return; + DPRINTK("%s(ptr=%p,size=%d,dir=%x)\n", + __func__, (void *) dma_addr, size, direction); + + BUG_ON(direction == PCI_DMA_NONE); + + local_irq_save(flags); + + buf = find_safe_buffer(dma_addr); + if (buf) { + BUG_ON(buf->size != size); + BUG_ON(buf->direction != direction); + + DPRINTK("%s: unsafe buffer %p (phy=%p) mapped to %p (phy=%p)\n", + __func__, + buf->ptr, (void *) virt_to_bus(buf->ptr), + buf->safe, (void *) buf->safe_dma_addr); + + if ((direction == PCI_DMA_FROMDEVICE) || + (direction == PCI_DMA_BIDIRECTIONAL)) { + DPRINTK("%s: copy back from safe %p, to unsafe %p size %d\n", + __func__, buf->safe, buf->ptr, size); + memcpy(buf->ptr, buf->safe, size); + } + free_safe_buffer(buf); } - mapped_alloc_size -= size; + local_irq_restore(flags); +} + +int +sa1111_map_sg(struct scatterlist *sg, int nents, int direction) +{ + BUG(); /* Not implemented. */ + + return -1; +} + +void +sa1111_unmap_sg(struct scatterlist *sg, int nents, int direction) +{ + BUG(); /* Not implemented. */ +} + +void +sa1111_dma_sync_single(dma_addr_t dma_addr, size_t size, int direction) +{ + unsigned long flags; + struct safe_buffer *buf; + + DPRINTK("%s(ptr=%p,size=%d,dir=%x)\n", + __func__, (void *) dma_addr, size, direction); - if (0) printk("pci_unmap_single(hwdev=%p,ptr=%p,size=%d,dir=%x) " - "alloced=%ld\n", - hwdev, (void *)dma_addr, size, direction, - mapped_alloc_size); - - if ((safe = find_safe_buffer((void *)dma_addr, &unsafe))) { - if (0) printk("copyback unsafe %p, safe %p, size %d\n", - unsafe, safe, size); - - consistent_sync(safe, size, PCI_DMA_FROMDEVICE); - memcpy(unsafe, safe, size); - free_safe_buffer(safe); + local_irq_save(flags); + + buf = find_safe_buffer(dma_addr); + if (buf) { + BUG_ON(buf->size != size); + BUG_ON(buf->direction != direction); + + DPRINTK("%s: unsafe buffer %p (phy=%p) mapped to %p (phy=%p)\n", + __func__, + buf->ptr, (void *) virt_to_bus(buf->ptr), + buf->safe, (void *) buf->safe_dma_addr); + + switch (direction) { + case PCI_DMA_FROMDEVICE: + DPRINTK("%s: copy back from safe %p, to unsafe %p size %d\n", + __func__, buf->safe, buf->ptr, size); + memcpy(buf->ptr, buf->safe, size); + break; + case PCI_DMA_TODEVICE: + DPRINTK("%s: copy out from unsafe %p, to safe %p, size %d\n", + __func__,buf->ptr, buf->safe, size); + memcpy(buf->safe, buf->ptr, size); + break; + case PCI_DMA_BIDIRECTIONAL: + BUG(); /* is this allowed? what does it mean? */ + default: + BUG(); + } + consistent_sync(buf->safe, size, direction); } else { - /* assume this is normal memory */ - buf = bus_to_virt(dma_addr); - consistent_sync(buf, size, PCI_DMA_FROMDEVICE); + consistent_sync(bus_to_virt(dma_addr), size, direction); } + + local_irq_restore(flags); +} + +void +sa1111_dma_sync_sg(struct scatterlist *sg, int nelems, int direction) +{ + BUG(); /* Not implemented. */ } EXPORT_SYMBOL(sa1111_map_single); EXPORT_SYMBOL(sa1111_unmap_single); +EXPORT_SYMBOL(sa1111_map_sg); +EXPORT_SYMBOL(sa1111_unmap_sg); +EXPORT_SYMBOL(sa1111_dma_sync_single); +EXPORT_SYMBOL(sa1111_dma_sync_sg); -static int __init sa1111_init_safe_buffers(void) +/* **************************************** */ + +static int __init sa1111_pcibuf_init(void) { - printk("Initializing SA1111 buffer pool for DMA workaround\n"); - init_safe_buffers(NULL); - return 0; + int ret; + + printk(KERN_DEBUG + "sa1111_pcibuf: initializing SA-1111 DMA workaround\n"); + + ret = create_safe_buffer_pools(); + + return ret; } +module_init(sa1111_pcibuf_init); -static void free_safe_buffers(void) +static void __exit sa1111_pcibuf_exit(void) { - pci_pool_destroy(small_buffer_cache); - pci_pool_destroy(large_buffer_cache); + BUG_ON(!list_empty(&safe_buffers)); + + destroy_safe_buffer_pools(); } +module_exit(sa1111_pcibuf_exit); -module_init(sa1111_init_safe_buffers); -module_exit(free_safe_buffers); +MODULE_AUTHOR("Christopher Hoover "); +MODULE_DESCRIPTION("Special pci_{map/unmap/dma_sync}_* routines for SA-1111."); +MODULE_LICENSE("GPL"); diff -urN orig/arch/arm/mach-sa1100/sa1111.c linux/arch/arm/mach-sa1100/sa1111.c --- orig/arch/arm/mach-sa1100/sa1111.c Mon Aug 5 13:29:46 2002 +++ linux/arch/arm/mach-sa1100/sa1111.c Thu Feb 6 11:31:52 2003 @@ -24,6 +24,8 @@ #include #include #include +#include +#include #include #include @@ -35,7 +37,7 @@ #include "sa1111.h" struct resource sa1111_resource = { - name: "SA1111", + .name = "SA1111", }; EXPORT_SYMBOL(sa1111_resource); diff -urN orig/arch/arm/mach-sa1100/simpad.c linux/arch/arm/mach-sa1100/simpad.c --- orig/arch/arm/mach-sa1100/simpad.c Mon Aug 5 13:29:47 2002 +++ linux/arch/arm/mach-sa1100/simpad.c Sun Sep 15 13:04:18 2002 @@ -76,7 +76,7 @@ } static struct sa1100_port_fns simpad_port_fns __initdata = { - pm: simpad_uart_pm, + .pm = simpad_uart_pm, }; static void __init simpad_map_io(void) diff -urN orig/arch/arm/mach-sa1100/simputer.c linux/arch/arm/mach-sa1100/simputer.c --- orig/arch/arm/mach-sa1100/simputer.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-sa1100/simputer.c Mon Aug 5 23:10:38 2002 @@ -0,0 +1,78 @@ +/* + * linux/arch/arm/mach-sa1100/simputer.c + * + * Author: Vivek K S (modified from Nicolas Pitre's Assabet file) + * + * This file contains all Simputer-specific tweaks. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + */ + +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include + +#include +#include +#include +#include + +#include "generic.h" + +static void __init +fixup_simputer(struct machine_desc *desc, struct param_struct *params, + char **cmdline, struct meminfo *mi) +{ +} + + +static struct map_desc simputer_io_desc[] __initdata = { + /* virtual physical length domain r w c b */ + { 0xe8000000, 0x00000000, 0x02000000, DOMAIN_IO, 0, 1, 0, 0 }, /* Flash bank 0 */ + { 0xf2800000, 0x40000000, 0x00100000, DOMAIN_IO, 0, 1, 0, 0 }, /* Simputer Modem */ + { 0xf3800000, 0x48000000, 0x00100000, DOMAIN_IO, 0, 1, 0, 0 }, /* USB */ + LAST_DESC +}; + +static void __init simputer_map_io(void) +{ + sa1100_map_io(); + iotable_init(simputer_io_desc); + + sa1100_register_uart(1, 2); + sa1100_register_uart(0, 1); /* com port */ + sa1100_register_uart(2, 3); /* radio module */ + + /* + * Set up registers for sleep mode. + */ + PWER = PWER_GPIO0; + PGSR = 0; + PCFR = 0; + + /* + * Clear all possible wakeup reasons. + */ + RCSR = RCSR_HWR | RCSR_SWR | RCSR_WDR | RCSR_SMR; +} + + +MACHINE_START(SIMPUTER, "Picopeta-Simputer") + BOOT_MEM(0xc0000000, 0x80000000, 0xf8000000) + BOOT_PARAMS(0xc0000100) + FIXUP(fixup_simputer) + MAPIO(simputer_map_io) + INITIRQ(sa1100_init_irq) +MACHINE_END diff -urN orig/arch/arm/mach-sa1100/sleep.S linux/arch/arm/mach-sa1100/sleep.S --- orig/arch/arm/mach-sa1100/sleep.S Fri Nov 16 10:09:48 2001 +++ linux/arch/arm/mach-sa1100/sleep.S Sun Jun 16 12:54:24 2002 @@ -11,12 +11,17 @@ * 2001-02-06: Cliff Brake Initial code * * 2001-08-29: Nicolas Pitre Simplified. + * + * 2002-05-27: Nicolas Pitre Revisited, more cleanup and simplification. + * Storage is on the stack now. */ #include #include #include -#include "sleep.h" + + + .text /* * sa1100_cpu_suspend() @@ -25,35 +30,25 @@ * */ - .text - -ENTRY(sleep_save) .word 0 @ virtual address of parameter array -ENTRY(sleep_save_p) .word 0 @ physical address of parameter array - ENTRY(sa1100_cpu_suspend) - @ save registers on stack - stmfd sp!, {r4 - r12, lr} - - @ load virtual address for sleep_save array - ldr r4, sleep_save + stmfd sp!, {r4 - r12, lr} @ save registers on stack - @ save stack pointer - str sp, [r4, #(SLEEP_SAVE_SP*4)] - - @ save coprocessor registers - mrc p15, 0, r1, c1, c0, 0 - str r1, [r4, #(SLEEP_SAVE_CP15_R1*4)] - mrc p15, 0, r1, c2, c0, 0 - str r1, [r4, #(SLEEP_SAVE_CP15_R2*4)] - mrc p15, 0, r1, c3, c0, 0 - str r1, [r4, #(SLEEP_SAVE_CP15_R3*4)] - mrc p15, 0, r1, c5, c0, 0 - str r1, [r4, #(SLEEP_SAVE_CP15_R5*4)] - mrc p15, 0, r1, c6, c0, 0 - str r1, [r4, #(SLEEP_SAVE_CP15_R6*4)] - mrc p15, 0, r1, c13, c0, 0 - str r1, [r4, #(SLEEP_SAVE_CP15_R13*4)] + @ get coprocessor registers + mrc p15, 0, r4, c3, c0, 0 @ domain ID + mrc p15, 0, r5, c2, c0, 0 @ translation table base addr + mrc p15, 0, r6, c13, c0, 0 @ PID + mrc p15, 0, r7, c1, c0, 0 @ control reg + + @ store them plus current virtual stack ptr on stack + mov r8, sp + stmfd sp!, {r4 - r8} + + @ preserve phys address of stack + mov r0, sp + bl sleep_phys_sp + ldr r1, =sleep_save_sp + str r0, [r1] @ clean data cache and invalidate WB bl cpu_sa1100_cache_clean_invalidate_all @@ -79,25 +74,24 @@ mov r0, #90 bl SYMBOL_NAME(udelay) - -/* setup up register contents for jump to page containing SA1110 SDRAM controller bug fix suspend code - * - * r0 points to MSC0 register - * r1 points to MSC1 register - * r2 points to MSC2 register - * r3 is MSC0 value - * r4 is MSC1 value - * r5 is MSC2 value - * r6 points to MDREFR register - * r7 is first MDREFR value - * r8 is second MDREFR value - * r9 is pointer to MDCNFG register - * r10 is MDCNFG value - * r11 is third MDREFR value - * r12 is pointer to PMCR register - * r13 is PMCR value (1) - * - */ + /* + * SA1110 SDRAM controller workaround. register values: + * + * r0 = &MSC0 + * r1 = &MSC1 + * r2 = &MSC2 + * r3 = MSC0 value + * r4 = MSC1 value + * r5 = MSC2 value + * r6 = &MDREFR + * r7 = first MDREFR value + * r8 = second MDREFR value + * r9 = &MDCNFG + * r10 = MDCNFG value + * r11 = third MDREFR value + * r12 = &PMCR + * r13 = PMCR value (1) + */ ldr r0, =MSC0 ldr r1, =MSC1 @@ -170,69 +164,49 @@ * * entry point from bootloader into kernel during resume * + * Note: Yes, part of the following code is located into the .data section. + * This is to allow sleep_save_sp to be accessed with a relative load + * while we can't rely on any MMU translation. We could have put + * sleep_save_sp in the .text section as well, but some setups might + * insist on .text to be truely read-only. */ + .data .align 5 ENTRY(sa1100_cpu_resume) - - @ set SVC, irqs off - mov r0, #I_BIT | MODE_SVC + mov r0, #I_BIT | F_BIT | MODE_SVC @ set SVC, irqs off msr cpsr_c, r0 - @ load physical address of sleep_save - ldr r4, sleep_save_p - - @ restore cp15_r3, domain id - ldr r1, [r4, #(SLEEP_SAVE_CP15_R3*4)] - mcr p15, 0, r1, c3, c0 ,0 - - @ restore cp15_r2, translation table base address - ldr r1, [r4, #(SLEEP_SAVE_CP15_R2*4)] - mcr p15, 0, r1, c2, c0 ,0 + ldr r0, sleep_save_sp @ stack phys addr + ldr r2, =resume_after_mmu @ its absolute virtual address + ldmfd r0, {r4 - r7, sp} @ CP regs + virt stack ptr mov r1, #0 mcr p15, 0, r1, c8, c7, 0 @ flush I+D TLBs mcr p15, 0, r1, c7, c7, 0 @ flush I&D cache + mcr p15, 0, r1, c9, c0, 0 @ invalidate RB + mcr p15, 0, r1, c9, c0, 5 @ allow user space to use RB - @ get saved cp15 r1 (control register) - ldr r1, [r4, #(SLEEP_SAVE_CP15_R1*4)] - - @ get address to jump to after turning on MMU - ldr r2, =resume_after_mmu - - cmp r2, #0 - - b resume_turn_on_mmu + mcr p15, 0, r4, c3, c0, 0 @ domain ID + mcr p15, 0, r5, c2, c0, 0 @ translation table base addr + mcr p15, 0, r6, c13, c0, 0 @ PID + b resume_turn_on_mmu @ cache align execution .align 5 resume_turn_on_mmu: - - @ turn on mmu - mcr p15, 0, r1, c1, c0 ,0 - - @ jump to resume_after_mmu - mov pc, r2 + mcr p15, 0, r7, c1, c0, 0 @ turn on MMU, caches, etc. + nop + mov pc, r2 @ jump to virtual addr + nop nop nop - .align 5 -resume_after_mmu: - - @ load virtual address for sleep_save array - ldr r4, sleep_save - - @ restore the rest of CPU state - ldr r1, [r4, #(SLEEP_SAVE_CP15_R13*4)] - mcr p15, 0, r1, c13, c0, 0 - ldr r1, [r4, #(SLEEP_SAVE_CP15_R5*4)] - mcr p15, 0, r1, c5, c0 ,0 - ldr r1, [r4, #(SLEEP_SAVE_CP15_R6*4)] - mcr p15, 0, r1, c6, c0 ,0 - - @ restore stack pointer - ldr sp, [r4, #(SLEEP_SAVE_SP*4)] +sleep_save_sp: + .word 0 @ preserve stack phys ptr here - @ return to caller - ldmfd sp!, {r4 - r12, pc} + .text +resume_after_mmu: + mcr p15, 0, r1, c15, c1, 2 @ enable clock switching + ldmfd sp!, {r4 - r12, pc} @ return to caller diff -urN orig/arch/arm/mach-sa1100/sleep.h linux/arch/arm/mach-sa1100/sleep.h --- orig/arch/arm/mach-sa1100/sleep.h Fri Nov 16 10:09:48 2001 +++ linux/arch/arm/mach-sa1100/sleep.h Thu Jan 1 01:00:00 1970 @@ -1,30 +0,0 @@ -/* - * Offsets for register values preserved in RAM while in sleep mode - */ - -#define SLEEP_SAVE_OSCR 0 -#define SLEEP_SAVE_OSMR0 1 -#define SLEEP_SAVE_OSMR1 2 -#define SLEEP_SAVE_OSMR2 3 -#define SLEEP_SAVE_OSMR3 4 -#define SLEEP_SAVE_OIER 5 -#define SLEEP_SAVE_GPDR 6 -#define SLEEP_SAVE_GRER 7 -#define SLEEP_SAVE_GFER 8 -#define SLEEP_SAVE_GAFR 9 -#define SLEEP_SAVE_PPDR 10 -#define SLEEP_SAVE_PPSR 11 -#define SLEEP_SAVE_PPAR 12 -#define SLEEP_SAVE_PSDR 13 -#define SLEEP_SAVE_Ser1SDCR0 14 -#define SLEEP_SAVE_ICMR 15 -#define SLEEP_SAVE_SP 16 -#define SLEEP_SAVE_CP15_R1 17 -#define SLEEP_SAVE_CP15_R2 18 -#define SLEEP_SAVE_CP15_R3 19 -#define SLEEP_SAVE_CP15_R5 20 -#define SLEEP_SAVE_CP15_R6 21 -#define SLEEP_SAVE_CP15_R13 22 - -#define SLEEP_SAVE_SIZE 23 - diff -urN orig/arch/arm/mach-sa1100/ssp.c linux/arch/arm/mach-sa1100/ssp.c --- orig/arch/arm/mach-sa1100/ssp.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-sa1100/ssp.c Wed Feb 12 13:10:07 2003 @@ -0,0 +1,208 @@ +/* + * linux/arch/arm/mach-sa1100/ssp.c + * + * Copyright (C) 2003 Russell King. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + * + * Generic SSP driver. This provides the generic core for simple + * IO-based SSP applications. + */ +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include + +static void ssp_interrupt(int irq, void *dev_id, struct pt_regs *regs) +{ + unsigned int status = Ser4SSSR; + + if (status & SSSR_ROR) { + printk(KERN_WARNING "SSP: receiver overrun\n"); + } + + Ser4SSSR = SSSR_ROR; +} + +/** + * ssp_write_word - write a word to the SSP port + * @data: 16-bit, MSB justified data to write. + * + * Wait for a free entry in the SSP transmit FIFO, and write a data + * word to the SSP port. + * + * The caller is expected to perform the necessary locking. + * + * Returns: + * %-ETIMEDOUT timeout occurred (for future) + * 0 success + */ +int ssp_write_word(u16 data) +{ + while (!(Ser4SSSR & SSSR_TNF)) + cpu_relax(); + + Ser4SSDR = data; + + return 0; +} + +/** + * ssp_read_word - read a word from the SSP port + * + * Wait for a data word in the SSP receive FIFO, and return the + * received data. Data is LSB justified. + * + * Note: Currently, if data is not expected to be received, this + * function will wait for ever. + * + * The caller is expected to perform the necessary locking. + * + * Returns: + * %-ETIMEDOUT timeout occurred (for future) + * 16-bit data success + */ +int ssp_read_word(void) +{ + while (!(Ser4SSSR & SSSR_RNE)) + cpu_relax(); + + return Ser4SSDR; +} + +/** + * ssp_flush - flush the transmit and receive FIFOs + * + * Wait for the SSP to idle, and ensure that the receive FIFO + * is empty. + * + * The caller is expected to perform the necessary locking. + */ +void ssp_flush(void) +{ + do { + while (Ser4SSSR & SSSR_RNE) { + (void) Ser4SSDR; + } + } while (Ser4SSSR & SSSR_BSY); +} + +/** + * ssp_enable - enable the SSP port + * + * Turn on the SSP port. + */ +void ssp_enable(void) +{ + Ser4SSCR0 |= SSCR0_SSE; +} + +/** + * ssp_disable - shut down the SSP port + * + * Turn off the SSP port, optionally powering it down. + */ +void ssp_disable(void) +{ + Ser4SSCR0 &= ~SSCR0_SSE; +} + +/** + * ssp_save_state - save the SSP configuration + * @ssp: pointer to structure to save SSP configuration + * + * Save the configured SSP state for suspend. + */ +void ssp_save_state(struct ssp_state *ssp) +{ + ssp->cr0 = Ser4SSCR0; + ssp->cr1 = Ser4SSCR1; + + Ser4SSCR0 &= ~SSCR0_SSE; +} + +/** + * ssp_restore_state - restore a previously saved SSP configuration + * @ssp: pointer to configuration saved by ssp_save_state + * + * Restore the SSP configuration saved previously by ssp_save_state. + */ +void ssp_restore_state(struct ssp_state *ssp) +{ + Ser4SSSR = SSSR_ROR; + + Ser4SSCR0 = ssp->cr0 & ~SSCR0_SSE; + Ser4SSCR1 = ssp->cr1; + Ser4SSCR0 = ssp->cr0; +} + +/** + * ssp_init - setup the SSP port + * + * initialise and claim resources for the SSP port. + * + * Returns: + * %-ENODEV if the SSP port is unavailable + * %-EBUSY if the resources are already in use + * %0 on success + */ +int ssp_init(void) +{ + int ret; + + if (!(PPAR & PPAR_SPR) && (Ser4MCCR0 & MCCR0_MCE)) + return -ENODEV; + + if (!request_mem_region(__PREG(Ser4SSCR0), 0x18, "SSP")) { + return -EBUSY; + } + + Ser4SSSR = SSSR_ROR; + + ret = request_irq(IRQ_Ser4SSP, ssp_interrupt, 0, "SSP", NULL); + if (ret) + goto out_region; + + return 0; + + out_region: + release_mem_region(__PREG(Ser4SSCR0), 0x18); + return ret; +} + +/** + * ssp_exit - undo the effects of ssp_init + * + * release and free resources for the SSP port. + */ +void ssp_exit(void) +{ + Ser4SSCR0 &= ~SSCR0_SSE; + + free_irq(IRQ_Ser4SSP, NULL); + release_mem_region(__PREG(Ser4SSCR0), 0x18); +} + +MODULE_AUTHOR("Russell King"); +MODULE_DESCRIPTION("SA11x0 SSP PIO driver"); +MODULE_LICENSE("GPL"); + +EXPORT_SYMBOL(ssp_write_word); +EXPORT_SYMBOL(ssp_read_word); +EXPORT_SYMBOL(ssp_flush); +EXPORT_SYMBOL(ssp_enable); +EXPORT_SYMBOL(ssp_disable); +EXPORT_SYMBOL(ssp_save_state); +EXPORT_SYMBOL(ssp_restore_state); +EXPORT_SYMBOL(ssp_init); +EXPORT_SYMBOL(ssp_exit); diff -urN orig/arch/arm/mach-sa1100/system3.c linux/arch/arm/mach-sa1100/system3.c --- orig/arch/arm/mach-sa1100/system3.c Mon Aug 5 13:29:47 2002 +++ linux/arch/arm/mach-sa1100/system3.c Thu Feb 6 11:33:12 2003 @@ -42,6 +42,8 @@ #include #include #include +#include +#include #include #include @@ -107,19 +109,19 @@ }; static struct sa1100_port_fns system3_port_fns __initdata = { - set_mctrl: system3_set_mctrl, - get_mctrl: system3_get_mctrl, - pm: system3_uart_pm, + .set_mctrl = system3_set_mctrl, + .get_mctrl = system3_get_mctrl, + .pm = system3_uart_pm, }; static struct irqaction system3_irq = { - name: "PT Digital Board SA1111 IRQ", - handler: system3_IRQ_demux, - flags: SA_INTERRUPT + .name = "PT Digital Board SA1111 IRQ", + .handler = system3_IRQ_demux, + .flags = SA_INTERRUPT }; static struct notifier_block system3_clkchg_block = { - notifier_call: sdram_notifier, + .notifier_call = sdram_notifier, }; /********************************************************************** diff -urN orig/arch/arm/mach-sa1100/usb-char.c linux/arch/arm/mach-sa1100/usb-char.c --- orig/arch/arm/mach-sa1100/usb-char.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-sa1100/usb-char.c Wed Feb 27 11:38:38 2002 @@ -0,0 +1,723 @@ +/* + * (C) Copyright 2000-2001 Extenex Corporation + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. + * + * usb-char.c + * + * Miscellaneous character device interface for SA1100 USB function + * driver. + * + * Background: + * The SA1100 function driver ported from the Compaq Itsy project + * has an interface, usb-eth.c, to feed network packets over the + * usb wire and into the Linux TCP/IP stack. + * + * This file replaces that one with a simple character device + * interface that allows unstructured "byte pipe" style reads and + * writes over the USB bulk endpoints by userspace programs. + * + * A new define, CONFIG_SA1100_USB_NETLINK, has been created that, + * when set, (the default) causes the ethernet interface to be used. + * When not set, this more pedestrian character interface is linked + * in instead. + * + * Please see linux/Documentation/arm/SA1100/SA1100_USB for details. + * + * ward.willats@extenex.com + * + * To do: + * - Can't dma into ring buffer directly with pci_map/unmap usb_recv + * uses and get bytes out at the same time DMA is going on. Investigate: + * a) changing usb_recv to use alloc_consistent() at client request; or + * b) non-ring-buffer based data structures. In the meantime, I am using + * a bounce buffer. Simple, but wasteful. + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include + +#include "usb-char.h" +#include "sa1100_usb.h" + + + +////////////////////////////////////////////////////////////////////////////// +// Driver Options +////////////////////////////////////////////////////////////////////////////// + +#define VERSION "0.4" + + +#define VERBOSITY 1 + +#if VERBOSITY +# define PRINTK(x, a...) printk (x, ## a) +#else +# define PRINTK(x, a...) /**/ +#endif + +////////////////////////////////////////////////////////////////////////////// +// Globals - Macros - Enums - Structures +////////////////////////////////////////////////////////////////////////////// +#ifndef MIN +#define MIN( a, b ) ((a)<(b)?(a):(b)) +#endif + +typedef int bool; enum { false = 0, true = 1 }; + +static const char pszMe[] = "usbchr: "; + +static wait_queue_head_t wq_read; +static wait_queue_head_t wq_write; +static wait_queue_head_t wq_poll; + +/* Serialze multiple writers onto the transmit hardware +.. since we sleep the writer during transmit to stay in +.. sync. (Multiple writers don't make much sense, but..) */ +static DECLARE_MUTEX( xmit_sem ); + +// size of usb DATA0/1 packets. 64 is standard maximum +// for bulk transport, though most hosts seem to be able +// to handle larger. +#define TX_PACKET_SIZE 64 +#define RX_PACKET_SIZE 64 +#define RBUF_SIZE (4*PAGE_SIZE) + +static struct wcirc_buf { + char *buf; + int in; + int out; +} rx_ring = { NULL, 0, 0 }; + +static struct { + unsigned long cnt_rx_complete; + unsigned long cnt_rx_errors; + unsigned long bytes_rx; + unsigned long cnt_tx_timeouts; + unsigned long cnt_tx_errors; + unsigned long bytes_tx; +} charstats; + + +static char * tx_buf = NULL; +static char * packet_buffer = NULL; +static int sending = 0; +static int usb_ref_count = 0; +static int last_tx_result = 0; +static int last_rx_result = 0; +static int last_tx_size = 0; +static struct timer_list tx_timer; + +////////////////////////////////////////////////////////////////////////////// +// Prototypes +////////////////////////////////////////////////////////////////////////////// +static char * what_the_f( int e ); +static void free_txrx_buffers( void ); +static void twiddle_descriptors( void ); +static void free_string_descriptors( void ) ; +static int usbc_open( struct inode *pInode, struct file *pFile ); +static void rx_done_callback_packet_buffer( int flag, int size ); + +static void tx_timeout( unsigned long ); +static void tx_done_callback( int flag, int size ); + +static ssize_t usbc_read( struct file *, char *, size_t, loff_t * ); +static ssize_t usbc_write( struct file *, const char *, size_t, loff_t * ); +static unsigned int usbc_poll( struct file *pFile, poll_table * pWait ); +static int usbc_ioctl( struct inode *pInode, struct file *pFile, + unsigned int nCmd, unsigned long argument ); +static int usbc_close( struct inode *pInode, struct file *pFile ); + +#ifdef CONFIG_SA1100_EXTENEX1 +static void extenex_configured_notify_proc( void ); +#endif +////////////////////////////////////////////////////////////////////////////// +// Private Helpers +////////////////////////////////////////////////////////////////////////////// + +static char * what_the_f( int e ) +{ + char * p; + switch( e ) { + case 0: + p = "noErr"; + break; + case -ENODEV: + p = "ENODEV - usb not in config state"; + break; + case -EBUSY: + p = "EBUSY - another request on the hardware"; + break; + case -EAGAIN: + p = "EAGAIN"; + break; + case -EINTR: + p = "EINTR - interrupted\n"; + break; + case -EPIPE: + p = "EPIPE - zero length xfer\n"; + break; + default: + p = "????"; + break; + } + return p; +} + +static void free_txrx_buffers( void ) +{ + if ( rx_ring.buf != NULL ) { + kfree( rx_ring.buf ); + rx_ring.buf = NULL; + } + if ( packet_buffer != NULL ) { + kfree( packet_buffer ); + packet_buffer = NULL; + } + if ( tx_buf != NULL ) { + kfree( tx_buf ); + tx_buf = NULL; + } +} + +/* twiddle_descriptors() + * It is between open() and start(). Setup descriptors. + */ +static void twiddle_descriptors( void ) +{ + desc_t * pDesc = sa1100_usb_get_descriptor_ptr(); + string_desc_t * pString; + + pDesc->b.ep1.wMaxPacketSize = make_word_c( RX_PACKET_SIZE ); + pDesc->b.ep1.bmAttributes = USB_EP_BULK; + pDesc->b.ep2.wMaxPacketSize = make_word_c( TX_PACKET_SIZE ); + pDesc->b.ep2.bmAttributes = USB_EP_BULK; + + if ( machine_is_extenex1() ) { +#ifdef CONFIG_SA1100_EXTENEX1 + pDesc->dev.idVendor = make_word_c( 0xC9F ); + pDesc->dev.idProduct = 1; + pDesc->dev.bcdDevice = make_word_c( 0x0001 ); + pDesc->b.cfg.bmAttributes = USB_CONFIG_SELFPOWERED; + pDesc->b.cfg.MaxPower = 0; + + pString = sa1100_usb_kmalloc_string_descriptor( "Extenex" ); + if ( pString ) { + sa1100_usb_set_string_descriptor( 1, pString ); + pDesc->dev.iManufacturer = 1; + } + + pString = sa1100_usb_kmalloc_string_descriptor( "Handheld Theater" ); + if ( pString ) { + sa1100_usb_set_string_descriptor( 2, pString ); + pDesc->dev.iProduct = 2; + } + + pString = sa1100_usb_kmalloc_string_descriptor( "00000000" ); + if ( pString ) { + sa1100_usb_set_string_descriptor( 3, pString ); + pDesc->dev.iSerialNumber = 3; + } + + pString = sa1100_usb_kmalloc_string_descriptor( "HHT Bulk Transfer" ); + if ( pString ) { + sa1100_usb_set_string_descriptor( 4, pString ); + pDesc->b.intf.iInterface = 4; + } + sa1100_set_configured_callback( extenex_configured_notify_proc ); +#endif + } +} + +static void free_string_descriptors( void ) +{ + if ( machine_is_extenex1() ) { + string_desc_t * pString; + int i; + for( i = 1 ; i <= 4 ; i++ ) { + pString = sa1100_usb_get_string_descriptor( i ); + if ( pString ) + kfree( pString ); + } + } +} + +////////////////////////////////////////////////////////////////////////////// +// ASYNCHRONOUS +////////////////////////////////////////////////////////////////////////////// +static void kick_start_rx( void ) +{ + if ( usb_ref_count ) { + int total_space = CIRC_SPACE( rx_ring.in, rx_ring.out, RBUF_SIZE ); + if ( total_space >= RX_PACKET_SIZE ) { + sa1100_usb_recv( packet_buffer, + RX_PACKET_SIZE, + rx_done_callback_packet_buffer + ); + } + } +} +/* + * rx_done_callback_packet_buffer() + * We have completed a DMA xfer into the temp packet buffer. + * Move to ring. + * + * flag values: + * on init, -EAGAIN + * on reset, -EINTR + * on RPE, -EIO + * on short packet -EPIPE + */ +static void +rx_done_callback_packet_buffer( int flag, int size ) +{ + charstats.cnt_rx_complete++; + + if ( flag == 0 || flag == -EPIPE ) { + size_t n; + + charstats.bytes_rx += size; + + n = CIRC_SPACE_TO_END( rx_ring.in, rx_ring.out, RBUF_SIZE ); + n = MIN( n, size ); + size -= n; + + memcpy( &rx_ring.buf[ rx_ring.in ], packet_buffer, n ); + rx_ring.in = (rx_ring.in + n) & (RBUF_SIZE-1); + memcpy( &rx_ring.buf[ rx_ring.in ], packet_buffer + n, size ); + rx_ring.in = (rx_ring.in + size) & (RBUF_SIZE-1); + + wake_up_interruptible( &wq_read ); + wake_up_interruptible( &wq_poll ); + + last_rx_result = 0; + + kick_start_rx(); + + } else if ( flag != -EAGAIN ) { + charstats.cnt_rx_errors++; + last_rx_result = flag; + wake_up_interruptible( &wq_read ); + wake_up_interruptible( &wq_poll ); + } + else /* init, start a read */ + kick_start_rx(); +} + + +static void tx_timeout( unsigned long unused ) +{ + printk( "%stx timeout\n", pszMe ); + sa1100_usb_send_reset(); + charstats.cnt_tx_timeouts++; +} + + +// on init, -EAGAIN +// on reset, -EINTR +// on TPE, -EIO +static void tx_done_callback( int flags, int size ) +{ + if ( flags == 0 ) + charstats.bytes_tx += size; + else + charstats.cnt_tx_errors++; + last_tx_size = size; + last_tx_result = flags; + sending = 0; + wake_up_interruptible( &wq_write ); + wake_up_interruptible( &wq_poll ); +} + + +////////////////////////////////////////////////////////////////////////////// +// Workers +////////////////////////////////////////////////////////////////////////////// + +static int usbc_open( struct inode *pInode, struct file *pFile ) +{ + int retval = 0; + + PRINTK( KERN_DEBUG "%sopen()\n", pszMe ); + + /* start usb core */ + retval = sa1100_usb_open( "usb-char" ); + if ( retval ) return retval; + + /* allocate memory */ + if ( usb_ref_count == 0 ) { + tx_buf = (char*) kmalloc( TX_PACKET_SIZE, GFP_KERNEL | GFP_DMA ); + if ( tx_buf == NULL ) { + printk( "%sARGHH! COULD NOT ALLOCATE TX BUFFER\n", pszMe ); + goto malloc_fail; + } + rx_ring.buf = + (char*) kmalloc( RBUF_SIZE, GFP_KERNEL ); + + if ( rx_ring.buf == NULL ) { + printk( "%sARGHH! COULD NOT ALLOCATE RX BUFFER\n", pszMe ); + goto malloc_fail; + } + + packet_buffer = + (char*) kmalloc( RX_PACKET_SIZE, GFP_KERNEL | GFP_DMA ); + + if ( packet_buffer == NULL ) { + printk( "%sARGHH! COULD NOT ALLOCATE RX PACKET BUFFER\n", pszMe ); + goto malloc_fail; + } + rx_ring.in = rx_ring.out = 0; + memset( &charstats, 0, sizeof( charstats ) ); + sending = 0; + last_tx_result = 0; + last_tx_size = 0; + } + + /* modify default descriptors */ + twiddle_descriptors(); + + retval = sa1100_usb_start(); + if ( retval ) { + printk( "%sAGHH! Could not USB core\n", pszMe ); + free_txrx_buffers(); + return retval; + } + usb_ref_count++; /* must do _before_ kick_start() */ + MOD_INC_USE_COUNT; + kick_start_rx(); + return 0; + + malloc_fail: + free_txrx_buffers(); + return -ENOMEM; +} + +/* + * Read endpoint. Note that you can issue a read to an + * unconfigured endpoint. Eventually, the host may come along + * and configure underneath this module and data will appear. + */ +static ssize_t usbc_read( struct file *pFile, char *pUserBuffer, + size_t stCount, loff_t *pPos ) +{ + ssize_t retval; + int flags; + DECLARE_WAITQUEUE( wait, current ); + + PRINTK( KERN_DEBUG "%sread()\n", pszMe ); + + local_irq_save( flags ); + if ( last_rx_result == 0 ) { + local_irq_restore( flags ); + } else { /* an error happended and receiver is paused */ + local_irq_restore( flags ); + last_rx_result = 0; + kick_start_rx(); + } + + add_wait_queue( &wq_read, &wait ); + while( 1 ) { + ssize_t bytes_avail; + ssize_t bytes_to_end; + + set_current_state( TASK_INTERRUPTIBLE ); + + /* snap ring buf state */ + local_irq_save( flags ); + bytes_avail = CIRC_CNT( rx_ring.in, rx_ring.out, RBUF_SIZE ); + bytes_to_end = CIRC_CNT_TO_END( rx_ring.in, rx_ring.out, RBUF_SIZE ); + local_irq_restore( flags ); + + if ( bytes_avail != 0 ) { + ssize_t bytes_to_move = MIN( stCount, bytes_avail ); + retval = 0; // will be bytes transfered + if ( bytes_to_move != 0 ) { + size_t n = MIN( bytes_to_end, bytes_to_move ); + if ( copy_to_user( pUserBuffer, + &rx_ring.buf[ rx_ring.out ], + n ) ) { + retval = -EFAULT; + break; + } + bytes_to_move -= n; + retval += n; + // might go 1 char off end, so wrap + rx_ring.out = ( rx_ring.out + n ) & (RBUF_SIZE-1); + if ( copy_to_user( pUserBuffer + n, + &rx_ring.buf[ rx_ring.out ], + bytes_to_move ) + ) { + retval = -EFAULT; + break; + } + rx_ring.out += bytes_to_move; // cannot wrap + retval += bytes_to_move; + kick_start_rx(); + } + break; + } + else if ( last_rx_result ) { + retval = last_rx_result; + break; + } + else if ( pFile->f_flags & O_NONBLOCK ) { // no data, can't sleep + retval = -EAGAIN; + break; + } + else if ( signal_pending( current ) ) { // no data, can sleep, but signal + retval = -ERESTARTSYS; + break; + } + schedule(); // no data, can sleep + } + set_current_state( TASK_RUNNING ); + remove_wait_queue( &wq_read, &wait ); + + if ( retval < 0 ) + printk( "%sread error %d - %s\n", pszMe, retval, what_the_f( retval ) ); + return retval; +} + +/* + * Write endpoint. This routine attempts to break the passed in buffer + * into usb DATA0/1 packet size chunks and send them to the host. + * (The lower-level driver tries to do this too, but easier for us + * to manage things here.) + * + * We are at the mercy of the host here, in that it must send an IN + * token to us to pull this data back, so hopefully some higher level + * protocol is expecting traffic to flow in that direction so the host + * is actually polling us. To guard against hangs, a 5 second timeout + * is used. + * + * This routine takes some care to only report bytes sent that have + * actually made it across the wire. Thus we try to stay in lockstep + * with the completion routine and only have one packet on the xmit + * hardware at a time. Multiple simultaneous writers will get + * "undefined" results. + * + */ +static ssize_t usbc_write( struct file *pFile, const char * pUserBuffer, + size_t stCount, loff_t *pPos ) +{ + ssize_t retval = 0; + ssize_t stSent = 0; + + DECLARE_WAITQUEUE( wait, current ); + + PRINTK( KERN_DEBUG "%swrite() %d bytes\n", pszMe, stCount ); + + down( &xmit_sem ); // only one thread onto the hardware at a time + + while( stCount != 0 && retval == 0 ) { + int nThisTime = MIN( TX_PACKET_SIZE, stCount ); + copy_from_user( tx_buf, pUserBuffer, nThisTime ); + sending = nThisTime; + retval = sa1100_usb_send( tx_buf, nThisTime, tx_done_callback ); + if ( retval < 0 ) { + char * p = what_the_f( retval ); + printk( "%sCould not queue xmission. rc=%d - %s\n", + pszMe, retval, p ); + sending = 0; + break; + } + /* now have something on the diving board */ + add_wait_queue( &wq_write, &wait ); + tx_timer.expires = jiffies + ( HZ * 5 ); + add_timer( &tx_timer ); + while( 1 ) { + set_current_state( TASK_INTERRUPTIBLE ); + if ( sending == 0 ) { /* it jumped into the pool */ + del_timer( &tx_timer ); + retval = last_tx_result; + if ( retval == 0 ) { + stSent += last_tx_size; + pUserBuffer += last_tx_size; + stCount -= last_tx_size; + } + else + printk( "%sxmission error rc=%d - %s\n", + pszMe, retval, what_the_f(retval) ); + break; + } + else if ( signal_pending( current ) ) { + del_timer( &tx_timer ); + printk( "%ssignal\n", pszMe ); + retval = -ERESTARTSYS; + break; + } + schedule(); + } + set_current_state( TASK_RUNNING ); + remove_wait_queue( &wq_write, &wait ); + } + + up( &xmit_sem ); + + if ( 0 == retval ) + retval = stSent; + return retval; +} + +static unsigned int usbc_poll( struct file *pFile, poll_table * pWait ) +{ + unsigned int retval = 0; + + PRINTK( KERN_DEBUG "%poll()\n", pszMe ); + + poll_wait( pFile, &wq_poll, pWait ); + + if ( CIRC_CNT( rx_ring.in, rx_ring.out, RBUF_SIZE ) ) + retval |= POLLIN | POLLRDNORM; + if ( sa1100_usb_xmitter_avail() ) + retval |= POLLOUT | POLLWRNORM; + return retval; +} + +static int usbc_ioctl( struct inode *pInode, struct file *pFile, + unsigned int nCmd, unsigned long argument ) +{ + int retval = 0; + + switch( nCmd ) { + + case USBC_IOC_FLUSH_RECEIVER: + sa1100_usb_recv_reset(); + rx_ring.in = rx_ring.out = 0; + break; + + case USBC_IOC_FLUSH_TRANSMITTER: + sa1100_usb_send_reset(); + break; + + case USBC_IOC_FLUSH_ALL: + sa1100_usb_recv_reset(); + rx_ring.in = rx_ring.out = 0; + sa1100_usb_send_reset(); + break; + + default: + retval = -ENOIOCTLCMD; + break; + + } + return retval; +} + + +static int usbc_close( struct inode *pInode, struct file * pFile ) +{ + PRINTK( KERN_DEBUG "%sclose()\n", pszMe ); + if ( --usb_ref_count == 0 ) { + down( &xmit_sem ); + sa1100_usb_stop(); + free_txrx_buffers(); + free_string_descriptors(); + del_timer( &tx_timer ); + sa1100_usb_close(); + up( &xmit_sem ); + } + MOD_DEC_USE_COUNT; + return 0; +} + +#ifdef CONFIG_SA1100_EXTENEX1 +#include "../../../drivers/char/ex_gpio.h" +void extenex_configured_notify_proc( void ) +{ + if ( exgpio_play_string( "440,1:698,1" ) == -EAGAIN ) + printk( "%sWanted to BEEP but ex_gpio not open\n", pszMe ); +} +#endif +////////////////////////////////////////////////////////////////////////////// +// Initialization +////////////////////////////////////////////////////////////////////////////// + +static struct file_operations usbc_fops = { + owner: THIS_MODULE, + open: usbc_open, + read: usbc_read, + write: usbc_write, + poll: usbc_poll, + ioctl: usbc_ioctl, + release: usbc_close, +}; + +static struct miscdevice usbc_misc_device = { + USBC_MINOR, "usb_char", &usbc_fops +}; + +/* + * usbc_init() + */ + +int __init usbc_init( void ) +{ + int rc; + +#if !defined( CONFIG_ARCH_SA1100 ) + return -ENODEV; +#endif + + if ( (rc = misc_register( &usbc_misc_device )) != 0 ) { + printk( KERN_WARNING "%sCould not register device 10, " + "%d. (%d)\n", pszMe, USBC_MINOR, rc ); + return -EBUSY; + } + + // initialize wait queues + init_waitqueue_head( &wq_read ); + init_waitqueue_head( &wq_write ); + init_waitqueue_head( &wq_poll ); + + // initialize tx timeout timer + init_timer( &tx_timer ); + tx_timer.function = tx_timeout; + + printk( KERN_INFO "USB Function Character Driver Interface" + " - %s, (C) 2001, Extenex Corp.\n", VERSION + ); + + return rc; +} + +void __exit usbc_exit( void ) +{ +} + +EXPORT_NO_SYMBOLS; + +module_init(usbc_init); +module_exit(usbc_exit); + + + +// end: usb-char.c + + + diff -urN orig/arch/arm/mach-sa1100/usb-char.h linux/arch/arm/mach-sa1100/usb-char.h --- orig/arch/arm/mach-sa1100/usb-char.h Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-sa1100/usb-char.h Tue Oct 16 20:02:24 2001 @@ -0,0 +1,34 @@ +/* + * Copyright (C) 2001 Extenex Corporation + * + * usb-char.h + * + * Character device emulation client for SA-1100 client usb core. + * + * + * + */ +#ifndef _USB_CHAR_H +#define _USB_CHAR_H + +#define USBC_MAJOR 10 /* miscellaneous character device */ +#define USBC_MINOR 240 /* in the "reserved for local use" range */ + +#define USBC_MAGIC 0x8E + +/* zap everything in receive ring buffer */ +#define USBC_IOC_FLUSH_RECEIVER _IO( USBC_MAGIC, 0x01 ) + +/* reset transmitter */ +#define USBC_IOC_FLUSH_TRANSMITTER _IO( USBC_MAGIC, 0x02 ) + +/* do both of above */ +#define USBC_IOC_FLUSH_ALL _IO( USBC_MAGIC, 0x03 ) + + + + + + +#endif /* _USB_CHAR_H */ + diff -urN orig/arch/arm/mach-sa1100/usb-eth.c linux/arch/arm/mach-sa1100/usb-eth.c --- orig/arch/arm/mach-sa1100/usb-eth.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-sa1100/usb-eth.c Fri Sep 6 10:53:22 2002 @@ -0,0 +1,447 @@ + /* + * Ethernet driver for the SA1100 USB client function + * Copyright (c) 2001 by Nicolas Pitre + * + * This code was loosely inspired by the original initial ethernet test driver + * Copyright (c) Compaq Computer Corporation, 1999 + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + * + * This is still work in progress... + * + * 19/02/2001 - Now we are compatible with generic usbnet driver. green@iXcelerator.com + * 09/03/2001 - Dropped 'framing' scheme, as it seems to cause a lot of problems with little benefit. + * Now, since we do not know what size of packet we are receiving + * last usb packet in sequence will always be less than max packet + * receive endpoint can accept. + * Now the only way to check correct start of frame is to compare + * MAC address. Also now we are stalling on each receive error. + * + * 15/03/2001 - Using buffer to get data from UDC. DMA needs to have 8 byte + * aligned buffer, but this breaks IP code (unaligned access). + * + * 01/04/2001 - stall endpoint operations appeared to be very unstable, so + * they are disabled now. + * + * 03/06/2001 - Readded "zerocopy" receive path (tunable). + * + */ + +// Define DMA_NO_COPY if you want data to arrive directly into the +// receive network buffers, instead of arriving into bounce buffer +// and then get copied to network buffer. +// This does not work correctly right now. +#undef DMA_NO_COPY + +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include + +#include "sa1100_usb.h" + + +#define ETHERNET_VENDOR_ID 0x49f +#define ETHERNET_PRODUCT_ID 0x505A +#define MAX_PACKET 32768 +#define MIN(a, b) (((a) < (b)) ? (a) : (b)) + +// Should be global, so that insmod can change these +int usb_rsize=64; +int usb_wsize=64; + +static struct usbe_info_t { + struct net_device *dev; + u16 packet_id; + struct net_device_stats stats; +} usbe_info; + +static char usb_eth_name[16] = "usbf"; +static struct net_device usb_eth_device; +static struct sk_buff *cur_tx_skb, *next_tx_skb; +static struct sk_buff *cur_rx_skb, *next_rx_skb; +static volatile int terminating; +#ifndef DMA_NO_COPY +static char *dmabuf; // we need that, as dma expect it's buffers to be aligned on 8 bytes boundary +#endif + +static int usb_change_mtu (struct net_device *net, int new_mtu) +{ + if (new_mtu <= sizeof (struct ethhdr) || new_mtu > MAX_PACKET) + return -EINVAL; + // no second zero-length packet read wanted after mtu-sized packets + if (((new_mtu + sizeof (struct ethhdr)) % usb_rsize) == 0) + return -EDOM; + + net->mtu = new_mtu; + return 0; +} + +static struct sk_buff * +usb_new_recv_skb(void) +{ + struct sk_buff *skb = alloc_skb( 2 + sizeof (struct ethhdr) + usb_eth_device.mtu,GFP_ATOMIC); + + if (skb) { + skb_reserve(skb, 2); + } + return skb; +} + +static u8 bcast_hwaddr[ETH_ALEN]={0xff,0xff,0xff,0xff,0xff,0xff}; +static void +usb_recv_callback(int flag, int size) +{ + struct sk_buff *skb; + + if (terminating) + return; + + skb = cur_rx_skb; + + /* flag validation */ + if (flag == 0) { + if ( skb_tailroom (skb) < size ) { // hey! we are overloaded!!! + usbe_info.stats.rx_over_errors++; + goto error; + } +#ifndef DMA_NO_COPY + memcpy(skb->tail,dmabuf,size); +#endif + skb_put(skb, size); + } else { + if (flag == -EIO) { + usbe_info.stats.rx_errors++; + } + goto error; + } + + /* validate packet length */ + if (size == usb_rsize ) { + /* packet not complete yet */ + skb = NULL; + } + + /* + * At this point skb is non null if we have a complete packet. + * If so take a fresh skb right away and restart USB receive without + * further delays, then process the packet. Otherwise resume USB + * receive on the current skb and exit. + */ + + if (skb) + cur_rx_skb = next_rx_skb; +#ifndef DMA_NO_COPY + sa1100_usb_recv(dmabuf, usb_rsize, + usb_recv_callback); +#else + sa1100_usb_recv(cur_rx_skb->tail, MIN(usb_rsize, skb_tailroom (cur_rx_skb)), + usb_recv_callback); +#endif + if (!skb) + return; + + next_rx_skb = usb_new_recv_skb(); + if (!next_rx_skb) { + /* + * We can't aford loosing buffer space... + * So we drop the current packet and recycle its skb. + */ + printk("%s: can't allocate new skb\n", __FUNCTION__); + usbe_info.stats.rx_dropped++; + skb_trim(skb, 0); + next_rx_skb = skb; + return; + } + if ( skb->len >= sizeof(struct ethhdr)) { + if (memcmp(skb->data,usb_eth_device.dev_addr,ETH_ALEN) && memcmp(skb->data,bcast_hwaddr,ETH_ALEN) ) { + // This frame is not for us. nor it is broadcast + usbe_info.stats.rx_frame_errors++; + kfree_skb(skb); + goto error; + } + } + + if (skb->len) { + int status; +// FIXME: eth_copy_and_csum "small" packets to new SKB (small < ~200 bytes) ? + + skb->dev = &usb_eth_device; + skb->protocol = eth_type_trans (skb, &usb_eth_device); + usbe_info.stats.rx_packets++; + usbe_info.stats.rx_bytes += skb->len; + skb->ip_summed = CHECKSUM_NONE; + status = netif_rx (skb); + if (status != NET_RX_SUCCESS) + printk("netif_rx failed with code %d\n",status); + } else { +error: + /* + * Error due to HW addr mismatch, or IO error. + * Recycle the current skb and reset USB reception. + */ + skb_trim(cur_rx_skb, 0); +// if ( flag == -EINTR || flag == -EAGAIN ) // only if we are coming out of stall +#ifndef DMA_NO_COPY + sa1100_usb_recv(dmabuf, usb_rsize, usb_recv_callback); +#else + sa1100_usb_recv(cur_rx_skb->tail, MIN(usb_rsize, skb_tailroom (cur_rx_skb)), usb_recv_callback); +#endif + } +} + + +static void +usb_send_callback(int flag, int size) +{ + struct net_device *dev = usbe_info.dev; + struct net_device_stats *stats; + struct sk_buff *skb=cur_tx_skb; + int ret; + + if (terminating) + return; + + stats = &usbe_info.stats; + switch (flag) { + case 0: + stats->tx_packets++; + stats->tx_bytes += size; + break; + case -EIO: + stats->tx_errors++; + break; + default: + stats->tx_dropped++; + break; + } + + cur_tx_skb = next_tx_skb; + next_tx_skb = NULL; + dev_kfree_skb_irq(skb); + if (!cur_tx_skb) + return; + + dev->trans_start = jiffies; + ret = sa1100_usb_send(cur_tx_skb->data, cur_tx_skb->len, usb_send_callback); + if (ret) { + /* If the USB core can't accept the packet, we drop it. */ + dev_kfree_skb_irq(cur_tx_skb); + cur_tx_skb = NULL; + usbe_info.stats.tx_carrier_errors++; + } + netif_wake_queue(dev); +} + +static int +usb_eth_xmit(struct sk_buff *skb, struct net_device *dev) +{ + int ret; + unsigned long flags; + + if (next_tx_skb) { + printk("%s: called with next_tx_skb != NULL\n", __FUNCTION__); + return 1; + } + + if (skb_shared (skb)) { + struct sk_buff *skb2 = skb_unshare(skb, GFP_ATOMIC); + if (!skb2) { + usbe_info.stats.tx_dropped++; + dev_kfree_skb(skb); + return 1; + } + skb = skb2; + } + + if ((skb->len % usb_wsize) == 0) { + skb->len++; // other side will ignore this one, anyway. + } + + local_irq_save(flags); + if (cur_tx_skb) { + next_tx_skb = skb; + netif_stop_queue(dev); + } else { + cur_tx_skb = skb; + dev->trans_start = jiffies; + ret = sa1100_usb_send(skb->data, skb->len, usb_send_callback); + if (ret) { + /* If the USB core can't accept the packet, we drop it. */ + dev_kfree_skb(skb); + cur_tx_skb = NULL; + usbe_info.stats.tx_carrier_errors++; + } + } + local_irq_restore(flags); + return 0; +} + +static void +usb_xmit_timeout(struct net_device *dev ) +{ + sa1100_usb_send_reset(); + dev->trans_start = jiffies; + netif_wake_queue(dev); +} + + +static int +usb_eth_open(struct net_device *dev) +{ + terminating = 0; + cur_tx_skb = next_tx_skb = NULL; + cur_rx_skb = usb_new_recv_skb(); + next_rx_skb = usb_new_recv_skb(); + if (!cur_rx_skb || !next_rx_skb) { + printk("%s: can't allocate new skb\n", __FUNCTION__); + if (cur_rx_skb) + kfree_skb(cur_rx_skb); + if (next_rx_skb) + kfree_skb(next_rx_skb); + return -ENOMEM;; + } + + MOD_INC_USE_COUNT; +#ifndef DMA_NO_COPY + sa1100_usb_recv(dmabuf, usb_rsize, usb_recv_callback); +#else + sa1100_usb_recv(cur_rx_skb->tail, MIN(usb_rsize, skb_tailroom (cur_rx_skb)), + usb_recv_callback); +#endif + return 0; +} + +static int +usb_eth_release(struct net_device *dev) +{ + terminating = 1; + sa1100_usb_send_reset(); + sa1100_usb_recv_reset(); + if (cur_tx_skb) + kfree_skb(cur_tx_skb); + if (next_tx_skb) + kfree_skb(next_tx_skb); + if (cur_rx_skb) + kfree_skb(cur_rx_skb); + if (next_rx_skb) + kfree_skb(next_rx_skb); + MOD_DEC_USE_COUNT; + return 0; +} + +static struct net_device_stats * +usb_eth_stats(struct net_device *dev) +{ + struct usbe_info_t *priv = (struct usbe_info_t*) dev->priv; + struct net_device_stats *stats=NULL; + + if (priv) + stats = &priv->stats; + return stats; +} + +static int +usb_eth_probe(struct net_device *dev) +{ + u8 node_id [ETH_ALEN]; + + get_random_bytes (node_id, sizeof node_id); + node_id [0] &= 0xfe; // clear multicast bit + + /* + * Assign the hardware address of the board: + * generate it randomly, as there can be many such + * devices on the bus. + */ + memcpy (dev->dev_addr, node_id, sizeof node_id); + + dev->open = usb_eth_open; + dev->change_mtu = usb_change_mtu; + dev->stop = usb_eth_release; + dev->hard_start_xmit = usb_eth_xmit; + dev->get_stats = usb_eth_stats; + dev->watchdog_timeo = 1*HZ; + dev->tx_timeout = usb_xmit_timeout; + dev->priv = &usbe_info; + + usbe_info.dev = dev; + + /* clear the statistics */ + memset(&usbe_info.stats, 0, sizeof(struct net_device_stats)); + + ether_setup(dev); + dev->flags &= ~IFF_MULTICAST; + dev->flags &= ~IFF_BROADCAST; + //dev->flags |= IFF_NOARP; + + return 0; +} + +#ifdef MODULE +MODULE_PARM(usb_rsize, "1i"); +MODULE_PARM_DESC(usb_rsize, "number of bytes in packets from host to sa1100"); +MODULE_PARM(usb_wsize, "1i"); +MODULE_PARM_DESC(usb_wsize, "number of bytes in packets from sa1100 to host"); +#endif + +static int __init +usb_eth_init(void) +{ + int rc; + +#ifndef DMA_NO_COPY + dmabuf = kmalloc( usb_rsize, GFP_KERNEL | GFP_DMA ); + if (!dmabuf) + return -ENOMEM; +#endif + strncpy(usb_eth_device.name, usb_eth_name, IFNAMSIZ); + usb_eth_device.init = usb_eth_probe; + if (register_netdev(&usb_eth_device) != 0) + return -EIO; + + rc = sa1100_usb_open( "usb-eth" ); + if ( rc == 0 ) { + string_desc_t * pstr; + desc_t * pd = sa1100_usb_get_descriptor_ptr(); + + pd->b.ep1.wMaxPacketSize = make_word( usb_rsize ); + pd->b.ep2.wMaxPacketSize = make_word( usb_wsize ); + pd->dev.idVendor = ETHERNET_VENDOR_ID; + pd->dev.idProduct = ETHERNET_PRODUCT_ID; + pstr = sa1100_usb_kmalloc_string_descriptor( "SA1100 USB NIC" ); + if ( pstr ) { + sa1100_usb_set_string_descriptor( 1, pstr ); + pd->dev.iProduct = 1; + } + rc = sa1100_usb_start(); + } + return rc; +} + +module_init(usb_eth_init); + +static void __exit +usb_eth_cleanup(void) +{ + string_desc_t * pstr; + sa1100_usb_stop(); + sa1100_usb_close(); + if ( (pstr = sa1100_usb_get_string_descriptor(1)) != NULL ) + kfree( pstr ); +#ifndef DMA_NO_COPY + kfree(dmabuf); +#endif + unregister_netdev(&usb_eth_device); +} + +module_exit(usb_eth_cleanup); diff -urN orig/arch/arm/mach-sa1100/usb_ctl.c linux/arch/arm/mach-sa1100/usb_ctl.c --- orig/arch/arm/mach-sa1100/usb_ctl.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-sa1100/usb_ctl.c Tue Feb 5 20:32:51 2002 @@ -0,0 +1,774 @@ + /* + * Copyright (C) Compaq Computer Corporation, 1998, 1999 + * Copyright (C) Extenex Corporation, 2001 + * + * usb_ctl.c + * + * SA1100 USB controller core driver. + * + * This file provides interrupt routing and overall coordination + * of the three endpoints in usb_ep0, usb_receive (1), and usb_send (2). + * + * Please see linux/Documentation/arm/SA1100/SA1100_USB for details. + * + */ +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "sa1100_usb.h" +#include "usb_ctl.h" + +////////////////////////////////////////////////////////////////////////////// +// Prototypes +////////////////////////////////////////////////////////////////////////////// + +int usbctl_next_state_on_event( int event ); +static void udc_int_hndlr(int, void *, struct pt_regs *); +static void initialize_descriptors( void ); +static void soft_connect_hook( int enable ); +static void udc_disable(void); +static void udc_enable(void); + +#if CONFIG_PROC_FS +#define PROC_NODE_NAME "sausb" +static int usbctl_read_proc(char *page, char **start, off_t off, + int count, int *eof, void *data); +#endif + +////////////////////////////////////////////////////////////////////////////// +// Globals +////////////////////////////////////////////////////////////////////////////// +static const char pszMe[] = "usbctl: "; +struct usb_info_t usbd_info; /* global to ep0, usb_recv, usb_send */ + +/* device descriptors */ +static desc_t desc; + +#define MAX_STRING_DESC 8 +static string_desc_t * string_desc_array[ MAX_STRING_DESC ]; +static string_desc_t sd_zero; /* special sd_zero holds language codes */ + +// called when configured +static usb_notify_t configured_callback = NULL; + +enum { kStateZombie = 0, kStateZombieSuspend = 1, + kStateDefault = 2, kStateDefaultSuspend = 3, + kStateAddr = 4, kStateAddrSuspend = 5, + kStateConfig = 6, kStateConfigSuspend = 7 +}; + +static int device_state_machine[8][6] = { +// suspend reset resume adddr config deconfig +/* zombie */ { kStateZombieSuspend, kStateDefault, kError, kError, kError, kError }, +/* zom sus */ { kError, kStateDefault, kStateZombie, kError, kError, kError }, +/* default */ { kStateDefaultSuspend, kError, kStateDefault, kStateAddr, kError, kError }, +/* def sus */ { kError, kStateDefault, kStateDefault, kError, kError, kError }, +/* addr */ { kStateAddrSuspend, kStateDefault, kError, kError, kStateConfig, kError }, +/* addr sus */{ kError, kStateDefault, kStateAddr, kError, kError, kError }, +/* config */ { kStateConfigSuspend, kStateDefault, kError, kError, kError, kStateAddr }, +/* cfg sus */ { kError, kStateDefault, kStateConfig, kError, kError, kError } +}; + +/* "device state" is the usb device framework state, as opposed to the + "state machine state" which is whatever the driver needs and is much + more fine grained +*/ +static int sm_state_to_device_state[8] = +// zombie zom suspend default default sus +{ USB_STATE_POWERED, USB_STATE_SUSPENDED, USB_STATE_DEFAULT, USB_STATE_SUSPENDED, +// addr addr sus config config sus + USB_STATE_ADDRESS, USB_STATE_SUSPENDED, USB_STATE_CONFIGURED, USB_STATE_SUSPENDED +}; + +static char * state_names[8] = +{ "zombie", "zombie suspended", "default", "default suspended", + "address", "address suspended", "configured", "config suspended" +}; + +static char * event_names[6] = +{ "suspend", "reset", "resume", + "address assigned", "configure", "de-configure" +}; + +static char * device_state_names[] = +{ "not attached", "attached", "powered", "default", + "address", "configured", "suspended" }; + +static int sm_state = kStateZombie; + +////////////////////////////////////////////////////////////////////////////// +// Async +////////////////////////////////////////////////////////////////////////////// +static void core_kicker(void); + +static inline void enable_resume_mask_suspend( void ); +static inline void enable_suspend_mask_resume(void); + +static void +udc_int_hndlr(int irq, void *dev_id, struct pt_regs *regs) +{ + __u32 status = Ser0UDCSR; + + /* ReSeT Interrupt Request - UDC has been reset */ + if ( status & UDCSR_RSTIR ) + { + if ( usbctl_next_state_on_event( kEvReset ) != kError ) + { + /* starting 20ms or so reset sequence now... */ + printk("%sResetting\n", pszMe); + ep0_reset(); // just set state to idle + ep1_reset(); // flush dma, clear false stall + ep2_reset(); // flush dma, clear false stall + } + // mask reset ints, they flood during sequence, enable + // suspend and resume + Ser0UDCCR |= UDCCR_REM; // mask reset + Ser0UDCCR &= ~(UDCCR_SUSIM | UDCCR_RESIM); // enable suspend and resume + UDC_flip( Ser0UDCSR, status ); // clear all pending sources + return; // <-- no reason to continue if resetting + } + // else we have done something other than reset, so be sure reset enabled + UDC_clear( Ser0UDCCR, UDCCR_REM ); + + /* RESume Interrupt Request */ + if ( status & UDCSR_RESIR ) + { + usbctl_next_state_on_event( kEvResume ); + core_kicker(); + enable_suspend_mask_resume(); + } + + /* SUSpend Interrupt Request */ + if ( status & UDCSR_SUSIR ) + { + usbctl_next_state_on_event( kEvSuspend ); + enable_resume_mask_suspend(); + } + + UDC_flip(Ser0UDCSR, status); // clear all pending sources + + if (status & UDCSR_EIR) + ep0_int_hndlr(); + + if (status & UDCSR_RIR) + ep1_int_hndlr(status); + + if (status & UDCSR_TIR) + ep2_int_hndlr(status); +} + +static inline void enable_resume_mask_suspend( void ) +{ + int i = 0; + + while( 1 ) { + Ser0UDCCR |= UDCCR_SUSIM; // mask future suspend events + udelay( i ); + if ( (Ser0UDCCR & UDCCR_SUSIM) || (Ser0UDCSR & UDCSR_RSTIR) ) + break; + if ( ++i == 50 ) { + printk( "%senable_resume(): Could not set SUSIM %8.8X\n", + pszMe, Ser0UDCCR ); + break; + } + } + + i = 0; + while( 1 ) { + Ser0UDCCR &= ~UDCCR_RESIM; + udelay( i ); + if ( ( Ser0UDCCR & UDCCR_RESIM ) == 0 + || + (Ser0UDCSR & UDCSR_RSTIR) + ) + break; + if ( ++i == 50 ) { + printk( "%senable_resume(): Could not clear RESIM %8.8X\n", + pszMe, Ser0UDCCR ); + break; + } + } +} + +static inline void enable_suspend_mask_resume(void) +{ + int i = 0; + while( 1 ) { + Ser0UDCCR |= UDCCR_RESIM; // mask future resume events + udelay( i ); + if ( Ser0UDCCR & UDCCR_RESIM || (Ser0UDCSR & UDCSR_RSTIR) ) + break; + if ( ++i == 50 ) { + printk( "%senable_suspend(): Could not set RESIM %8.8X\n", + pszMe, Ser0UDCCR ); + break; + } + } + i = 0; + while( 1 ) { + Ser0UDCCR &= ~UDCCR_SUSIM; + udelay( i ); + if ( ( Ser0UDCCR & UDCCR_SUSIM ) == 0 + || + (Ser0UDCSR & UDCSR_RSTIR) + ) + break; + if ( ++i == 50 ) { + printk( "%senable_suspend(): Could not clear SUSIM %8.8X\n", + pszMe, Ser0UDCCR ); + break; + } + } +} + + +////////////////////////////////////////////////////////////////////////////// +// Public Interface +////////////////////////////////////////////////////////////////////////////// + +/* Open SA usb core on behalf of a client, but don't start running */ + +int +sa1100_usb_open( const char * client ) +{ + if ( usbd_info.client_name != NULL ) + return -EBUSY; + + usbd_info.client_name = (char*) client; + memset(&usbd_info.stats, 0, sizeof(struct usb_stats_t)); + memset(string_desc_array, 0, sizeof(string_desc_array)); + + /* hack to start in zombie suspended state */ + sm_state = kStateZombieSuspend; + usbd_info.state = USB_STATE_SUSPENDED; + + /* create descriptors for enumeration */ + initialize_descriptors(); + + printk( "%sOpened for %s\n", pszMe, client ); + return 0; +} + +/* Start running. Must have called usb_open (above) first */ +int +sa1100_usb_start( void ) +{ + if ( usbd_info.client_name == NULL ) { + printk( "%s%s - no client registered\n", + pszMe, __FUNCTION__ ); + return -EPERM; + } + + /* start UDC internal machinery running */ + udc_enable(); + udelay( 100 ); + + /* clear stall - receiver seems to start stalled? 19Jan01ww */ + /* also clear other stuff just to be thurough 22Feb01ww */ + UDC_clear(Ser0UDCCS1, UDCCS1_FST | UDCCS1_RPE | UDCCS1_RPC ); + UDC_clear(Ser0UDCCS2, UDCCS2_FST | UDCCS2_TPE | UDCCS2_TPC ); + + /* mask everything */ + Ser0UDCCR = 0xFC; + + /* flush DMA and fire through some -EAGAINs */ + ep1_init( usbd_info.dmach_rx ); + ep2_init( usbd_info.dmach_tx ); + + /* give endpoint notification we are starting */ + ep1_state_change_notify( USB_STATE_SUSPENDED ); + ep2_state_change_notify( USB_STATE_SUSPENDED ); + + /* enable any platform specific hardware */ + soft_connect_hook( 1 ); + + /* clear all top-level sources */ + Ser0UDCSR = UDCSR_RSTIR | UDCSR_RESIR | UDCSR_EIR | + UDCSR_RIR | UDCSR_TIR | UDCSR_SUSIR ; + + /* EXERIMENT - a short line in the spec says toggling this + ..bit diddles the internal state machine in the udc to + ..expect a suspend */ + Ser0UDCCR |= UDCCR_RESIM; + /* END EXPERIMENT 10Feb01ww */ + + /* enable any platform specific hardware */ + soft_connect_hook( 1 ); + + /* enable interrupts. If you are unplugged you will + immediately get a suspend interrupt. If you are plugged + and have a soft connect-circuit, you will get a reset + If you are plugged without a soft-connect, I think you + also get suspend. In short, start with suspend masked + and everything else enabled */ + UDC_write( Ser0UDCCR, UDCCR_SUSIM ); + + printk( "%sStarted for %s\n", pszMe, usbd_info.client_name ); + return 0; +} + +/* Stop USB core from running */ +int +sa1100_usb_stop( void ) +{ + if ( usbd_info.client_name == NULL ) { + printk( "%s%s - no client registered\n", + pszMe, __FUNCTION__ ); + return -EPERM; + } + /* mask everything */ + Ser0UDCCR = 0xFC; + ep1_reset(); + ep2_reset(); + udc_disable(); + printk( "%sStopped\n", pszMe ); + return 0; +} + +/* Tell SA core client is through using it */ +int +sa1100_usb_close( void ) +{ + if ( usbd_info.client_name == NULL ) { + printk( "%s%s - no client registered\n", + pszMe, __FUNCTION__ ); + return -EPERM; + } + usbd_info.client_name = NULL; + printk( "%sClosed\n", pszMe ); + return 0; +} + +/* set a proc to be called when device is configured */ +usb_notify_t sa1100_set_configured_callback( usb_notify_t func ) +{ + usb_notify_t retval = configured_callback; + configured_callback = func; + return retval; +} + +/*==================================================== + * Descriptor Manipulation. + * Use these between open() and start() above to setup + * the descriptors for your device. + * + */ + +/* get pointer to static default descriptor */ +desc_t * +sa1100_usb_get_descriptor_ptr( void ) { return &desc; } + +/* optional: set a string descriptor */ +int +sa1100_usb_set_string_descriptor( int i, string_desc_t * p ) +{ + int retval; + if ( i < MAX_STRING_DESC ) { + string_desc_array[i] = p; + retval = 0; + } else { + retval = -EINVAL; + } + return retval; +} + +/* optional: get a previously set string descriptor */ +string_desc_t * +sa1100_usb_get_string_descriptor( int i ) +{ + return ( i < MAX_STRING_DESC ) + ? string_desc_array[i] + : NULL; +} + + +/* optional: kmalloc and unicode up a string descriptor */ +string_desc_t * +sa1100_usb_kmalloc_string_descriptor( const char * p ) +{ + string_desc_t * pResult = NULL; + + if ( p ) { + int len = strlen( p ); + int uni_len = len * sizeof( __u16 ); + pResult = (string_desc_t*) kmalloc( uni_len + 2, GFP_KERNEL ); /* ugh! */ + if ( pResult != NULL ) { + int i; + pResult->bLength = uni_len + 2; + pResult->bDescriptorType = USB_DESC_STRING; + for( i = 0; i < len ; i++ ) { + pResult->bString[i] = make_word( (__u16) p[i] ); + } + } + } + return pResult; +} + +////////////////////////////////////////////////////////////////////////////// +// Exports to rest of driver +////////////////////////////////////////////////////////////////////////////// + +/* called by the int handler here and the two endpoint files when interesting + .."events" happen */ + +int +usbctl_next_state_on_event( int event ) +{ + int next_state = device_state_machine[ sm_state ][ event ]; + if ( next_state != kError ) + { + int next_device_state = sm_state_to_device_state[ next_state ]; + printk( "%s%s --> [%s] --> %s. Device in %s state.\n", + pszMe, state_names[ sm_state ], event_names[ event ], + state_names[ next_state ], device_state_names[ next_device_state ] ); + + sm_state = next_state; + if ( usbd_info.state != next_device_state ) + { + if ( configured_callback != NULL + && + next_device_state == USB_STATE_CONFIGURED + && + usbd_info.state != USB_STATE_SUSPENDED + ) { + configured_callback(); + } + usbd_info.state = next_device_state; + ep1_state_change_notify( next_device_state ); + ep2_state_change_notify( next_device_state ); + } + } +#if 0 + else + printk( "%s%s --> [%s] --> ??? is an error.\n", + pszMe, state_names[ sm_state ], event_names[ event ] ); +#endif + return next_state; +} + +////////////////////////////////////////////////////////////////////////////// +// Private Helpers +////////////////////////////////////////////////////////////////////////////// + +/* setup default descriptors */ + +static void +initialize_descriptors(void) +{ + desc.dev.bLength = sizeof( device_desc_t ); + desc.dev.bDescriptorType = USB_DESC_DEVICE; + desc.dev.bcdUSB = 0x100; /* 1.0 */ + desc.dev.bDeviceClass = 0xFF; /* vendor specific */ + desc.dev.bDeviceSubClass = 0; + desc.dev.bDeviceProtocol = 0; + desc.dev.bMaxPacketSize0 = 8; /* ep0 max fifo size */ + desc.dev.idVendor = 0; /* vendor ID undefined */ + desc.dev.idProduct = 0; /* product */ + desc.dev.bcdDevice = 0; /* vendor assigned device release num */ + desc.dev.iManufacturer = 0; /* index of manufacturer string */ + desc.dev.iProduct = 0; /* index of product description string */ + desc.dev.iSerialNumber = 0; /* index of string holding product s/n */ + desc.dev.bNumConfigurations = 1; + + desc.b.cfg.bLength = sizeof( config_desc_t ); + desc.b.cfg.bDescriptorType = USB_DESC_CONFIG; + desc.b.cfg.wTotalLength = make_word_c( sizeof(struct cdb) ); + desc.b.cfg.bNumInterfaces = 1; + desc.b.cfg.bConfigurationValue = 1; + desc.b.cfg.iConfiguration = 0; + desc.b.cfg.bmAttributes = USB_CONFIG_BUSPOWERED; + desc.b.cfg.MaxPower = USB_POWER( 500 ); + + desc.b.intf.bLength = sizeof( intf_desc_t ); + desc.b.intf.bDescriptorType = USB_DESC_INTERFACE; + desc.b.intf.bInterfaceNumber = 0; /* unique intf index*/ + desc.b.intf.bAlternateSetting = 0; + desc.b.intf.bNumEndpoints = 2; + desc.b.intf.bInterfaceClass = 0xFF; /* vendor specific */ + desc.b.intf.bInterfaceSubClass = 0; + desc.b.intf.bInterfaceProtocol = 0; + desc.b.intf.iInterface = 0; + + desc.b.ep1.bLength = sizeof( ep_desc_t ); + desc.b.ep1.bDescriptorType = USB_DESC_ENDPOINT; + desc.b.ep1.bEndpointAddress = USB_EP_ADDRESS( 1, USB_OUT ); + desc.b.ep1.bmAttributes = USB_EP_BULK; + desc.b.ep1.wMaxPacketSize = make_word_c( 64 ); + desc.b.ep1.bInterval = 0; + + desc.b.ep2.bLength = sizeof( ep_desc_t ); + desc.b.ep2.bDescriptorType = USB_DESC_ENDPOINT; + desc.b.ep2.bEndpointAddress = USB_EP_ADDRESS( 2, USB_IN ); + desc.b.ep2.bmAttributes = USB_EP_BULK; + desc.b.ep2.wMaxPacketSize = make_word_c( 64 ); + desc.b.ep2.bInterval = 0; + + /* set language */ + /* See: http://www.usb.org/developers/data/USB_LANGIDs.pdf */ + sd_zero.bDescriptorType = USB_DESC_STRING; + sd_zero.bLength = sizeof( string_desc_t ); + sd_zero.bString[0] = make_word_c( 0x409 ); /* American English */ + sa1100_usb_set_string_descriptor( 0, &sd_zero ); +} + +/* soft_connect_hook() + * Some devices have platform-specific circuitry to make USB + * not seem to be plugged in, even when it is. This allows + * software to control when a device 'appears' on the USB bus + * (after Linux has booted and this driver has loaded, for + * example). If you have such a circuit, control it here. + */ +static void +soft_connect_hook( int enable ) +{ +#ifdef CONFIG_SA1100_EXTENEX1 + if (machine_is_extenex1() ) { + if ( enable ) { + PPDR |= PPC_USB_SOFT_CON; + PPSR |= PPC_USB_SOFT_CON; + } else { + PPSR &= ~PPC_USB_SOFT_CON; + PPDR &= ~PPC_USB_SOFT_CON; + } + } +#endif +} + +/* disable the UDC at the source */ +static void +udc_disable(void) +{ + soft_connect_hook( 0 ); + UDC_set( Ser0UDCCR, UDCCR_UDD ); +} + + +/* enable the udc at the source */ +static void +udc_enable(void) +{ + UDC_clear(Ser0UDCCR, UDCCR_UDD); +} + +// HACK DEBUG 3Mar01ww +// Well, maybe not, it really seems to help! 08Mar01ww +static void +core_kicker( void ) +{ + __u32 car = Ser0UDCAR; + __u32 imp = Ser0UDCIMP; + __u32 omp = Ser0UDCOMP; + + UDC_set( Ser0UDCCR, UDCCR_UDD ); + udelay( 300 ); + UDC_clear(Ser0UDCCR, UDCCR_UDD); + + Ser0UDCAR = car; + Ser0UDCIMP = imp; + Ser0UDCOMP = omp; +} + +////////////////////////////////////////////////////////////////////////////// +// Proc Filesystem Support +////////////////////////////////////////////////////////////////////////////// + +#if CONFIG_PROC_FS + +#define SAY( fmt, args... ) p += sprintf(p, fmt, ## args ) +#define SAYV( num ) p += sprintf(p, num_fmt, "Value", num ) +#define SAYC( label, yn ) p += sprintf(p, yn_fmt, label, yn ) +#define SAYS( label, v ) p += sprintf(p, cnt_fmt, label, v ) + +static int usbctl_read_proc(char *page, char **start, off_t off, + int count, int *eof, void *data) +{ + const char * num_fmt = "%25.25s: %8.8lX\n"; + const char * cnt_fmt = "%25.25s: %lu\n"; + const char * yn_fmt = "%25.25s: %s\n"; + const char * yes = "YES"; + const char * no = "NO"; + unsigned long v; + char * p = page; + int len; + + SAY( "SA1100 USB Controller Core\n" ); + SAY( "USB state: %s (%s) %d\n", + device_state_names[ sm_state_to_device_state[ sm_state ] ], + state_names[ sm_state ], + sm_state ); + + SAYS( "ep0 bytes read", usbd_info.stats.ep0_bytes_read ); + SAYS( "ep0 bytes written", usbd_info.stats.ep0_bytes_written ); + SAYS( "ep0 FIFO read failures", usbd_info.stats.ep0_fifo_write_failures ); + SAYS( "ep0 FIFO write failures", usbd_info.stats.ep0_fifo_write_failures ); + + SAY( "\n" ); + + v = Ser0UDCAR; + SAY( "%25.25s: 0x%8.8lX - %ld\n", "Address Register", v, v ); + v = Ser0UDCIMP; + SAY( "%25.25s: %ld (%8.8lX)\n", "IN max packet size", v+1, v ); + v = Ser0UDCOMP; + SAY( "%25.25s: %ld (%8.8lX)\n", "OUT max packet size", v+1, v ); + + v = Ser0UDCCR; + SAY( "\nUDC Mask Register\n" ); + SAYV( v ); + SAYC( "UDC Active", ( v & UDCCR_UDA ) ? yes : no ); + SAYC( "Suspend interrupts masked", ( v & UDCCR_SUSIM ) ? yes : no ); + SAYC( "Resume interrupts masked", ( v & UDCCR_RESIM ) ? yes : no ); + SAYC( "Reset interrupts masked", ( v & UDCCR_REM ) ? yes : no ); + + v = Ser0UDCSR; + SAY( "\nUDC Interrupt Request Register\n" ); + SAYV( v ); + SAYC( "Reset pending", ( v & UDCSR_RSTIR ) ? yes : no ); + SAYC( "Suspend pending", ( v & UDCSR_SUSIR ) ? yes : no ); + SAYC( "Resume pending", ( v & UDCSR_RESIR ) ? yes : no ); + SAYC( "ep0 pending", ( v & UDCSR_EIR ) ? yes : no ); + SAYC( "receiver pending", ( v & UDCSR_RIR ) ? yes : no ); + SAYC( "tramsitter pending", ( v & UDCSR_TIR ) ? yes : no ); + +#ifdef CONFIG_SA1100_EXTENEX1 + SAYC( "\nSoft connect", (PPSR & PPC_USB_SOFT_CON) ? "Visible" : "Hidden" ); +#endif + +#if 0 + v = Ser0UDCCS0; + SAY( "\nUDC Endpoint Zero Status Register\n" ); + SAYV( v ); + SAYC( "Out Packet Ready", ( v & UDCCS0_OPR ) ? yes : no ); + SAYC( "In Packet Ready", ( v & UDCCS0_IPR ) ? yes : no ); + SAYC( "Sent Stall", ( v & UDCCS0_SST ) ? yes : no ); + SAYC( "Force Stall", ( v & UDCCS0_FST ) ? yes : no ); + SAYC( "Data End", ( v & UDCCS0_DE ) ? yes : no ); + SAYC( "Data Setup End", ( v & UDCCS0_SE ) ? yes : no ); + SAYC( "Serviced (SO)", ( v & UDCCS0_SO ) ? yes : no ); + + v = Ser0UDCCS1; + SAY( "\nUDC Receiver Status Register\n" ); + SAYV( v ); + SAYC( "Receive Packet Complete", ( v & UDCCS1_RPC ) ? yes : no ); + SAYC( "Sent Stall", ( v & UDCCS1_SST ) ? yes : no ); + SAYC( "Force Stall", ( v & UDCCS1_FST ) ? yes : no ); + SAYC( "Receive Packet Error", ( v & UDCCS1_RPE ) ? yes : no ); + SAYC( "Receive FIFO not empty", ( v & UDCCS1_RNE ) ? yes : no ); + + v = Ser0UDCCS2; + SAY( "\nUDC Transmitter Status Register\n" ); + SAYV( v ); + SAYC( "FIFO has < 8 of 16 chars", ( v & UDCCS2_TFS ) ? yes : no ); + SAYC( "Transmit Packet Complete", ( v & UDCCS2_TPC ) ? yes : no ); + SAYC( "Transmit FIFO underrun", ( v & UDCCS2_TUR ) ? yes : no ); + SAYC( "Transmit Packet Error", ( v & UDCCS2_TPE ) ? yes : no ); + SAYC( "Sent Stall", ( v & UDCCS2_SST ) ? yes : no ); + SAYC( "Force Stall", ( v & UDCCS2_FST ) ? yes : no ); +#endif + + len = ( p - page ) - off; + if ( len < 0 ) + len = 0; + *eof = ( len <=count ) ? 1 : 0; + *start = page + off; + return len; +} + +#endif /* CONFIG_PROC_FS */ + +////////////////////////////////////////////////////////////////////////////// +// Module Initialization and Shutdown +////////////////////////////////////////////////////////////////////////////// +/* + * usbctl_init() + * Module load time. Allocate dma and interrupt resources. Setup /proc fs + * entry. Leave UDC disabled. + */ +int __init usbctl_init( void ) +{ + int retval = 0; + + udc_disable(); + + memset( &usbd_info, 0, sizeof( usbd_info ) ); + +#if CONFIG_PROC_FS + create_proc_read_entry ( PROC_NODE_NAME, 0, NULL, usbctl_read_proc, NULL); +#endif + + /* setup rx dma */ + retval = sa1100_request_dma(&usbd_info.dmach_rx, "USB receive", DMA_Ser0UDCRd); + if (retval) { + printk("%sunable to register for rx dma rc=%d\n", pszMe, retval ); + goto err_rx_dma; + } + + /* setup tx dma */ + retval = sa1100_request_dma(&usbd_info.dmach_tx, "USB transmit", DMA_Ser0UDCWr); + if (retval) { + printk("%sunable to register for tx dma rc=%d\n",pszMe,retval); + goto err_tx_dma; + } + + /* now allocate the IRQ. */ + retval = request_irq(IRQ_Ser0UDC, udc_int_hndlr, SA_INTERRUPT, + "SA USB core", NULL); + if (retval) { + printk("%sCouldn't request USB irq rc=%d\n",pszMe, retval); + goto err_irq; + } + + printk( "SA1100 USB Controller Core Initialized\n"); + return 0; + +err_irq: + sa1100_free_dma(usbd_info.dmach_tx); + usbd_info.dmach_tx = 0; +err_tx_dma: + sa1100_free_dma(usbd_info.dmach_rx); + usbd_info.dmach_rx = 0; +err_rx_dma: + return retval; +} +/* + * usbctl_exit() + * Release DMA and interrupt resources + */ +void __exit usbctl_exit( void ) +{ + printk("Unloading SA1100 USB Controller\n"); + + udc_disable(); + +#if CONFIG_PROC_FS + remove_proc_entry ( PROC_NODE_NAME, NULL); +#endif + + sa1100_free_dma(usbd_info.dmach_rx); + sa1100_free_dma(usbd_info.dmach_tx); + free_irq(IRQ_Ser0UDC, NULL); +} + +EXPORT_SYMBOL( sa1100_usb_open ); +EXPORT_SYMBOL( sa1100_usb_start ); +EXPORT_SYMBOL( sa1100_usb_stop ); +EXPORT_SYMBOL( sa1100_usb_close ); + + +EXPORT_SYMBOL( sa1100_usb_get_descriptor_ptr ); +EXPORT_SYMBOL( sa1100_usb_set_string_descriptor ); +EXPORT_SYMBOL( sa1100_usb_get_string_descriptor ); +EXPORT_SYMBOL( sa1100_usb_kmalloc_string_descriptor ); + + +module_init( usbctl_init ); +module_exit( usbctl_exit ); diff -urN orig/arch/arm/mach-sa1100/usb_ctl.h linux/arch/arm/mach-sa1100/usb_ctl.h --- orig/arch/arm/mach-sa1100/usb_ctl.h Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-sa1100/usb_ctl.h Tue Oct 16 20:02:24 2001 @@ -0,0 +1,123 @@ +/* + * Copyright (C) Compaq Computer Corporation, 1998, 1999 + * Copyright (C) Extenex Corporation 2001 + * + * usb_ctl.h + * + * PRIVATE interface used to share info among components of the SA-1100 USB + * core: usb_ctl, usb_ep0, usb_recv and usb_send. Clients of the USB core + * should use sa1100_usb.h. + * + */ + +#ifndef _USB_CTL_H +#define _USB_CTL_H + +#include /* dmach_t */ + + +/* + * These states correspond to those in the USB specification v1.0 + * in chapter 8, Device Framework. + */ +enum { USB_STATE_NOTATTACHED=0, USB_STATE_ATTACHED=1,USB_STATE_POWERED=2, + USB_STATE_DEFAULT=3, USB_STATE_ADDRESS=4, USB_STATE_CONFIGURED=5, + USB_STATE_SUSPENDED=6}; + +struct usb_stats_t { + unsigned long ep0_fifo_write_failures; + unsigned long ep0_bytes_written; + unsigned long ep0_fifo_read_failures; + unsigned long ep0_bytes_read; +}; + +struct usb_info_t +{ + char * client_name; + dmach_t dmach_tx, dmach_rx; + int state; + unsigned char address; + struct usb_stats_t stats; +}; + +/* in usb_ctl.c */ +extern struct usb_info_t usbd_info; + +/* + * Function Prototypes + */ +enum { kError=-1, kEvSuspend=0, kEvReset=1, + kEvResume=2, kEvAddress=3, kEvConfig=4, kEvDeConfig=5 }; +int usbctl_next_state_on_event( int event ); + +/* endpoint zero */ +void ep0_reset(void); +void ep0_int_hndlr(void); + +/* receiver */ +void ep1_state_change_notify( int new_state ); +int ep1_recv(void); +int ep1_init(int chn); +void ep1_int_hndlr(int status); +void ep1_reset(void); +void ep1_stall(void); + +/* xmitter */ +void ep2_state_change_notify( int new_state ); +void ep2_reset(void); +int ep2_init(int chn); +void ep2_int_hndlr(int status); +void ep2_stall(void); + +#define UDC_write(reg, val) { \ + int i = 10000; \ + do { \ + (reg) = (val); \ + if (i-- <= 0) { \ + printk( "%s [%d]: write %#x to %p (%#x) failed\n", \ + __FUNCTION__, __LINE__, (val), &(reg), (reg)); \ + break; \ + } \ + } while((reg) != (val)); \ +} + +#define UDC_set(reg, val) { \ + int i = 10000; \ + do { \ + (reg) |= (val); \ + if (i-- <= 0) { \ + printk( "%s [%d]: set %#x of %p (%#x) failed\n", \ + __FUNCTION__, __LINE__, (val), &(reg), (reg)); \ + break; \ + } \ + } while(!((reg) & (val))); \ +} + +#define UDC_clear(reg, val) { \ + int i = 10000; \ + do { \ + (reg) &= ~(val); \ + if (i-- <= 0) { \ + printk( "%s [%d]: clear %#x of %p (%#x) failed\n", \ + __FUNCTION__, __LINE__, (val), &(reg), (reg)); \ + break; \ + } \ + } while((reg) & (val)); \ +} + +#define UDC_flip(reg, val) { \ + int i = 10000; \ + (reg) = (val); \ + do { \ + (reg) = (val); \ + if (i-- <= 0) { \ + printk( "%s [%d]: flip %#x of %p (%#x) failed\n", \ + __FUNCTION__, __LINE__, (val), &(reg), (reg)); \ + break; \ + } \ + } while(((reg) & (val))); \ +} + + +#define CHECK_ADDRESS { if ( Ser0UDCAR == 1 ) { printk("%s:%d I lost my address!!!\n",__FUNCTION__, __LINE__);}} +#endif /* _USB_CTL_H */ diff -urN orig/arch/arm/mach-sa1100/usb_ep0.c linux/arch/arm/mach-sa1100/usb_ep0.c --- orig/arch/arm/mach-sa1100/usb_ep0.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-sa1100/usb_ep0.c Tue Oct 16 20:02:24 2001 @@ -0,0 +1,911 @@ +/* + * Copyright (C) Extenex Corporation 2001 + * Much folklore gleaned from original code: + * Copyright (C) Compaq Computer Corporation, 1998, 1999 + * + * usb_ep0.c - SA1100 USB controller driver. + * Endpoint zero management + * + * Please see: + * linux/Documentation/arm/SA1100/SA1100_USB + * for details. (Especially since Intel docs are full of + * errors about ep0 operation.) ward.willats@extenex.com. + * + * Intel also has a "Universal Serial Bus Client Device + * Validation for the StrongARM SA-1100 Microprocessor" + * document, which has flow charts and assembler test driver, + * but be careful, since it is just for validation and not + * a "real world" solution. + * + * A summary of three types of data-returning setups: + * + * 1. Setup request <= 8 bytes. That is, requests that can + * be fullfilled in one write to the FIFO. DE is set + * with IPR in queue_and_start_write(). (I don't know + * if there really are any of these!) + * + * 2. Setup requests > 8 bytes (requiring more than one + * IN to get back to the host), and we have at least + * as much or more data than the host requested. In + * this case we pump out everything we've got, and + * when the final interrupt comes in due to the UDC + * clearing the last IPR, we just set DE. + * + * 3. Setup requests > 8 bytes, but we don't have enough + * data to satisfy the request. In this case, we send + * everything we've got, and when the final interrupt + * comes in due to the UDC clearing the last IPR + * we write nothing to the FIFO and set both IPR and DE + * so the UDC sends an empty packet and forces the host + * to perform short packet retirement instead of stalling + * out. + * + */ + +#include +#include "sa1100_usb.h" /* public interface */ +#include "usb_ctl.h" /* private stuff */ + + +// 1 == lots of trace noise, 0 = only "important' stuff +#define VERBOSITY 0 + +enum { true = 1, false = 0 }; +typedef int bool; +#ifndef MIN +#define MIN( a, b ) ((a)<(b)?(a):(b)) +#endif + +#if 1 && !defined( ASSERT ) +# define ASSERT(expr) \ + if(!(expr)) { \ + printk( "Assertion failed! %s,%s,%s,line=%d\n",\ + #expr,__FILE__,__FUNCTION__,__LINE__); \ + } +#else +# define ASSERT(expr) +#endif + +#if VERBOSITY +#define PRINTKD(fmt, args...) printk( fmt , ## args) +#else +#define PRINTKD(fmt, args...) +#endif + +/*================================================ + * USB Protocol Stuff + */ + +/* Request Codes */ +enum { GET_STATUS=0, CLEAR_FEATURE=1, SET_FEATURE=3, + SET_ADDRESS=5, GET_DESCRIPTOR=6, SET_DESCRIPTOR=7, + GET_CONFIGURATION=8, SET_CONFIGURATION=9, GET_INTERFACE=10, + SET_INTERFACE=11 }; + + +/* USB Device Requests */ +typedef struct +{ + __u8 bmRequestType; + __u8 bRequest; + __u16 wValue; + __u16 wIndex; + __u16 wLength; +} usb_dev_request_t __attribute__ ((packed)); + +/*************************************************************************** +Prototypes +***************************************************************************/ +/* "setup handlers" -- the main functions dispatched to by the + .. isr. These represent the major "modes" of endpoint 0 operaton */ +static void sh_setup_begin(void); /* setup begin (idle) */ +static void sh_write( void ); /* writing data */ +static void sh_write_with_empty_packet( void ); /* empty packet at end of xfer*/ +/* called before both sh_write routines above */ +static void common_write_preamble( void ); + +/* other subroutines */ +static __u32 queue_and_start_write( void * p, int req, int act ); +static void write_fifo( void ); +static int read_fifo( usb_dev_request_t * p ); +static void get_descriptor( usb_dev_request_t * pReq ); + +/* some voodo helpers 01Mar01ww */ +static void set_cs_bits( __u32 set_bits ); +static void set_de( void ); +static void set_ipr( void ); +static void set_ipr_and_de( void ); +static bool clear_opr( void ); + +/*************************************************************************** +Inline Helpers +***************************************************************************/ + +/* Data extraction from usb_request_t fields */ +enum { kTargetDevice=0, kTargetInterface=1, kTargetEndpoint=2 }; +static inline int request_target( __u8 b ) { return (int) ( b & 0x0F); } + +static inline int windex_to_ep_num( __u16 w ) { return (int) ( w & 0x000F); } +inline int type_code_from_request( __u8 by ) { return (( by >> 4 ) & 3); } + +/* following is hook for self-powered flag in GET_STATUS. Some devices + .. might like to override and return real info */ +static inline bool self_powered_hook( void ) { return true; } + +/* print string descriptor */ +static inline void psdesc( string_desc_t * p ) +{ + int i; + int nchars = ( p->bLength - 2 ) / sizeof( __u16 ); + printk( "'" ); + for( i = 0 ; i < nchars ; i++ ) { + printk( "%c", (char) p->bString[i] ); + } + printk( "'\n" ); +} + + +#if VERBOSITY +/* "pcs" == "print control status" */ +static inline void pcs( void ) +{ + __u32 foo = Ser0UDCCS0; + printk( "%8.8X: %s %s %s %s\n", + foo, + foo & UDCCS0_SE ? "SE" : "", + foo & UDCCS0_OPR ? "OPR" : "", + foo & UDCCS0_IPR ? "IPR" : "", + foo & UDCCS0_SST ? "SST" : "" + ); +} +static inline void preq( usb_dev_request_t * pReq ) +{ + static char * tnames[] = { "dev", "intf", "ep", "oth" }; + static char * rnames[] = { "std", "class", "vendor", "???" }; + char * psz; + switch( pReq->bRequest ) { + case GET_STATUS: psz = "get stat"; break; + case CLEAR_FEATURE: psz = "clr feat"; break; + case SET_FEATURE: psz = "set feat"; break; + case SET_ADDRESS: psz = "set addr"; break; + case GET_DESCRIPTOR: psz = "get desc"; break; + case SET_DESCRIPTOR: psz = "set desc"; break; + case GET_CONFIGURATION: psz = "get cfg"; break; + case SET_CONFIGURATION: psz = "set cfg"; break; + case GET_INTERFACE: psz = "get intf"; break; + case SET_INTERFACE: psz = "set intf"; break; + default: psz = "unknown"; break; + } + printk( "- [%s: %s req to %s. dir=%s]\n", psz, + rnames[ (pReq->bmRequestType >> 5) & 3 ], + tnames[ pReq->bmRequestType & 3 ], + ( pReq->bmRequestType & 0x80 ) ? "in" : "out" ); +} + +#else +static inline void pcs( void ){} +static inline void preq( void ){} +#endif + +/*************************************************************************** +Globals +***************************************************************************/ +static const char pszMe[] = "usbep0: "; + +/* pointer to current setup handler */ +static void (*current_handler)(void) = sh_setup_begin; + +/* global write struct to keep write + ..state around across interrupts */ +static struct { + unsigned char *p; + int bytes_left; +} wr; + +/*************************************************************************** +Public Interface +***************************************************************************/ + +/* reset received from HUB (or controller just went nuts and reset by itself!) + so udc core has been reset, track this state here */ +void +ep0_reset(void) +{ + /* reset state machine */ + current_handler = sh_setup_begin; + wr.p = NULL; + wr.bytes_left = 0; + usbd_info.address=0; +} + +/* handle interrupt for endpoint zero */ +void +ep0_int_hndlr( void ) +{ + PRINTKD( "/\\(%d)\n", Ser0UDCAR ); + pcs(); + + /* if not in setup begin, we are returning data. + execute a common preamble to both write handlers + */ + if ( current_handler != sh_setup_begin ) + common_write_preamble(); + + (*current_handler)(); + + PRINTKD( "---\n" ); + pcs(); + PRINTKD( "\\/\n" ); +} + +/*************************************************************************** +Setup Handlers +***************************************************************************/ +/* + * sh_setup_begin() + * This setup handler is the "idle" state of endpoint zero. It looks for OPR + * (OUT packet ready) to see if a setup request has been been received from the + * host. Requests without a return data phase are immediately handled. Otherwise, + * in the case of GET_XXXX the handler may be set to one of the sh_write_xxxx + * data pumpers if more than 8 bytes need to get back to the host. + * + */ +static void +sh_setup_begin( void ) +{ + unsigned char status_buf[2]; /* returned in GET_STATUS */ + usb_dev_request_t req; + int request_type; + int n; + __u32 cs_bits; + __u32 address; + __u32 cs_reg_in = Ser0UDCCS0; + + if (cs_reg_in & UDCCS0_SST) { + PRINTKD( "%ssetup begin: sent stall. Continuing\n", pszMe ); + set_cs_bits( UDCCS0_SST ); + } + + if ( cs_reg_in & UDCCS0_SE ) { + PRINTKD( "%ssetup begin: Early term of setup. Continuing\n", pszMe ); + set_cs_bits( UDCCS0_SSE ); /* clear setup end */ + } + + /* Be sure out packet ready, otherwise something is wrong */ + if ( (cs_reg_in & UDCCS0_OPR) == 0 ) { + /* we can get here early...if so, we'll int again in a moment */ + PRINTKD( "%ssetup begin: no OUT packet available. Exiting\n", pszMe ); + goto sh_sb_end; + } + + /* read the setup request */ + n = read_fifo( &req ); + if ( n != sizeof( req ) ) { + printk( "%ssetup begin: fifo READ ERROR wanted %d bytes got %d. " + " Stalling out...\n", + pszMe, sizeof( req ), n ); + /* force stall, serviced out */ + set_cs_bits( UDCCS0_FST | UDCCS0_SO ); + goto sh_sb_end; + } + + /* Is it a standard request? (not vendor or class request) */ + request_type = type_code_from_request( req.bmRequestType ); + if ( request_type != 0 ) { + printk( "%ssetup begin: unsupported bmRequestType: %d ignored\n", + pszMe, request_type ); + set_cs_bits( UDCCS0_DE | UDCCS0_SO ); + goto sh_sb_end; + } + +#if VERBOSITY + { + unsigned char * pdb = (unsigned char *) &req; + PRINTKD( "%2.2X %2.2X %2.2X %2.2X %2.2X %2.2X %2.2X %2.2X ", + pdb[0], pdb[1], pdb[2], pdb[3], pdb[4], pdb[5], pdb[6], pdb[7] + ); + preq( &req ); + } +#endif + + /* Handle it */ + switch( req.bRequest ) { + + /* This first bunch have no data phase */ + + case SET_ADDRESS: + address = (__u32) (req.wValue & 0x7F); + /* when SO and DE sent, UDC will enter status phase and ack, + ..propagating new address to udc core. Next control transfer + ..will be on the new address. You can't see the change in a + ..read back of CAR until then. (about 250us later, on my box). + ..The original Intel driver sets S0 and DE and code to check + ..that address has propagated here. I tried this, but it + ..would only work sometimes! The rest of the time it would + ..never propagate and we'd spin forever. So now I just set + ..it and pray... + */ + Ser0UDCAR = address; + usbd_info.address = address; + usbctl_next_state_on_event( kEvAddress ); + set_cs_bits( UDCCS0_SO | UDCCS0_DE ); /* no data phase */ + printk( "%sI have been assigned address: %d\n", pszMe, address ); + break; + + + case SET_CONFIGURATION: + if ( req.wValue == 1 ) { + /* configured */ + if (usbctl_next_state_on_event( kEvConfig ) != kError){ + /* (re)set the out and in max packet sizes */ + desc_t * pDesc = sa1100_usb_get_descriptor_ptr(); + __u32 out = __le16_to_cpu( pDesc->b.ep1.wMaxPacketSize ); + __u32 in = __le16_to_cpu( pDesc->b.ep2.wMaxPacketSize ); + Ser0UDCOMP = ( out - 1 ); + Ser0UDCIMP = ( in - 1 ); + printk( "%sConfigured (OMP=%8.8X IMP=%8.8X)\n", pszMe, out, in ); + } + } else if ( req.wValue == 0 ) { + /* de-configured */ + if (usbctl_next_state_on_event( kEvDeConfig ) != kError ) + printk( "%sDe-Configured\n", pszMe ); + } else { + printk( "%ssetup phase: Unknown " + "\"set configuration\" data %d\n", + pszMe, req.wValue ); + } + set_cs_bits( UDCCS0_SO | UDCCS0_DE ); /* no data phase */ + break; + + case CLEAR_FEATURE: + /* could check data length, direction...26Jan01ww */ + if ( req.wValue == 0 ) { /* clearing ENDPOINT_HALT/STALL */ + int ep = windex_to_ep_num( req.wIndex ); + if ( ep == 1 ) { + printk( "%sclear feature \"endpoint halt\" " + " on receiver\n", pszMe ); + ep1_reset(); + } + else if ( ep == 2 ) { + printk( "%sclear feature \"endpoint halt\" " + "on xmitter\n", pszMe ); + ep2_reset(); + } else { + printk( "%sclear feature \"endpoint halt\" " + "on unsupported ep # %d\n", + pszMe, ep ); + } + } else { + printk( "%sUnsupported feature selector (%d) " + "in clear feature. Ignored.\n" , + pszMe, req.wValue ); + } + set_cs_bits( UDCCS0_SO | UDCCS0_DE ); /* no data phase */ + break; + + case SET_FEATURE: + if ( req.wValue == 0 ) { /* setting ENDPOINT_HALT/STALL */ + int ep = windex_to_ep_num( req.wValue ); + if ( ep == 1 ) { + printk( "%set feature \"endpoint halt\" " + "on receiver\n", pszMe ); + ep1_stall(); + } + else if ( ep == 2 ) { + printk( "%sset feature \"endpoint halt\" " + " on xmitter\n", pszMe ); + ep2_stall(); + } else { + printk( "%sset feature \"endpoint halt\" " + "on unsupported ep # %d\n", + pszMe, ep ); + } + } + else { + printk( "%sUnsupported feature selector " + "(%d) in set feature\n", + pszMe, req.wValue ); + } + set_cs_bits( UDCCS0_SO | UDCCS0_DE ); /* no data phase */ + break; + + + /* The rest have a data phase that writes back to the host */ + case GET_STATUS: + /* return status bit flags */ + status_buf[0] = status_buf[1] = 0; + n = request_target(req.bmRequestType); + switch( n ) { + case kTargetDevice: + if ( self_powered_hook() ) + status_buf[0] |= 1; + break; + case kTargetInterface: + break; + case kTargetEndpoint: + /* return stalled bit */ + n = windex_to_ep_num( req.wIndex ); + if ( n == 1 ) + status_buf[0] |= (Ser0UDCCS1 & UDCCS1_FST) >> 4; + else if ( n == 2 ) + status_buf[0] |= (Ser0UDCCS2 & UDCCS2_FST) >> 5; + else { + printk( "%sUnknown endpoint (%d) " + "in GET_STATUS\n", pszMe, n ); + } + break; + default: + printk( "%sUnknown target (%d) in GET_STATUS\n", + pszMe, n ); + /* fall thru */ + break; + } + cs_bits = queue_and_start_write( status_buf, + req.wLength, + sizeof( status_buf ) ); + set_cs_bits( cs_bits ); + break; + case GET_DESCRIPTOR: + get_descriptor( &req ); + break; + + case GET_CONFIGURATION: + status_buf[0] = (usbd_info.state == USB_STATE_CONFIGURED) + ? 1 + : 0; + cs_bits = queue_and_start_write( status_buf, req.wLength, 1 ); + set_cs_bits( cs_bits ); + break; + case GET_INTERFACE: + printk( "%sfixme: get interface not supported\n", pszMe ); + cs_bits = queue_and_start_write( NULL, req.wLength, 0 ); + set_cs_bits( cs_bits ); + break; + case SET_INTERFACE: + printk( "%sfixme: set interface not supported\n", pszMe ); + set_cs_bits( UDCCS0_DE | UDCCS0_SO ); + break; + default : + printk("%sunknown request 0x%x\n", pszMe, req.bRequest); + break; + } /* switch( bRequest ) */ + +sh_sb_end: + return; + +} +/* + * common_wrtie_preamble() + * Called before execution of sh_write() or sh_write_with_empty_packet() + * Handles common abort conditions. + * + */ +static void common_write_preamble( void ) +{ + /* If "setup end" has been set, the usb controller has + ..terminated a setup transaction before we set DE. This + ..happens during enumeration with some hosts. For example, + ..the host will ask for our device descriptor and specify + ..a return of 64 bytes. When we hand back the first 8, the + ..host will know our max packet size and turn around and + ..issue a new setup immediately. This causes the UDC to auto-ack + ..the new setup and set SE. We must then "unload" (process) + ..the new setup, which is what will happen after this preamble + ..is finished executing. + */ + __u32 cs_reg_in = Ser0UDCCS0; + + if ( cs_reg_in & UDCCS0_SE ) { + PRINTKD( "%swrite_preamble(): Early termination of setup\n", pszMe ); + Ser0UDCCS0 = UDCCS0_SSE; /* clear setup end */ + current_handler = sh_setup_begin; + } + + if ( cs_reg_in & UDCCS0_SST ) { + PRINTKD( "%swrite_preamble(): UDC sent stall\n", pszMe ); + Ser0UDCCS0 = UDCCS0_SST; /* clear setup end */ + current_handler = sh_setup_begin; + } + + if ( cs_reg_in & UDCCS0_OPR ) { + PRINTKD( "%swrite_preamble(): see OPR. Stopping write to " + "handle new SETUP\n", pszMe ); + /* very rarely, you can get OPR and leftover IPR. Try to clear */ + UDC_clear( Ser0UDCCS0, UDCCS0_IPR ); + current_handler = sh_setup_begin; + } +} + +/* + * sh_write() + * This is the setup handler when we are in the data return phase of + * a setup request and have as much (or more) data than the host + * requested. If we enter this routine and bytes left is zero, the + * last data packet has gone (int is because IPR was just cleared) + * so we just set DE and reset. Otheriwse, we write another packet + * and set IPR. + */ +static void sh_write() +{ + PRINTKD( "W\n" ); + + if ( Ser0UDCCS0 & UDCCS0_IPR ) { + PRINTKD( "%ssh_write(): IPR set, exiting\n", pszMe ); + return; + } + + /* If bytes left is zero, we are coming in on the + ..interrupt after the last packet went out. And + ..we know we don't have to empty packet this transfer + ..so just set DE and we are done */ + + if ( 0 == wr.bytes_left ) { + /* that's it, so data end */ + set_de(); + wr.p = NULL; /* be anal */ + current_handler = sh_setup_begin; + } else { + /* Otherwise, more data to go */ + write_fifo(); + set_ipr(); + } +} +/* + * sh_write_with_empty_packet() + * This is the setup handler when we don't have enough data to + * satisfy the host's request. After we send everything we've got + * we must send an empty packet (by setting IPR and DE) so the + * host can perform "short packet retirement" and not stall. + * + */ +static void sh_write_with_empty_packet( void ) +{ + __u32 cs_reg_out = 0; + PRINTKD( "WE\n" ); + + if ( Ser0UDCCS0 & UDCCS0_IPR ) { + PRINTKD( "%ssh_write(): IPR set, exiting\n", pszMe ); + return; + } + + /* If bytes left is zero, we are coming in on the + ..interrupt after the last packet went out. + ..we must do short packet suff, so set DE and IPR + */ + + if ( 0 == wr.bytes_left ) { + set_ipr_and_de(); + wr.p = NULL; + current_handler = sh_setup_begin; + PRINTKD( "%ssh_write empty() Sent empty packet \n", pszMe ); + } else { + write_fifo(); /* send data */ + set_ipr(); /* flag a packet is ready */ + } + Ser0UDCCS0 = cs_reg_out; +} + +/*************************************************************************** +Other Private Subroutines +***************************************************************************/ +/* + * queue_and_start_write() + * p == data to send + * req == bytes host requested + * act == bytes we actually have + * Returns: bits to be flipped in ep0 control/status register + * + * Called from sh_setup_begin() to begin a data return phase. Sets up the + * global "wr"-ite structure and load the outbound FIFO with data. + * If can't send all the data, set appropriate handler for next interrupt. + * + */ +static __u32 queue_and_start_write( void * in, int req, int act ) +{ + __u32 cs_reg_bits = UDCCS0_IPR; + unsigned char * p = (unsigned char*) in; + + PRINTKD( "Qr=%d a=%d\n",req,act ); + + /* thou shalt not enter data phase until the serviced OUT is clear */ + if ( ! clear_opr() ) { + printk( "%sSO did not clear OPR\n", pszMe ); + return ( UDCCS0_DE | UDCCS0_SO ) ; + } + wr.p = p; + wr.bytes_left = MIN( act, req ); + + write_fifo(); + + if ( 0 == wr.bytes_left ) { + cs_reg_bits |= UDCCS0_DE; /* out in 1 so data end */ + wr.p = NULL; /* be anal */ + } + else if ( act < req ) /* we are going to short-change host */ + current_handler = sh_write_with_empty_packet; /* so need nul to not stall */ + else /* we have as much or more than requested */ + current_handler = sh_write; + + return cs_reg_bits; /* note: IPR was set uncondtionally at start of routine */ +} +/* + * write_fifo() + * Stick bytes in the 8 bytes endpoint zero FIFO. + * This version uses a variety of tricks to make sure the bytes + * are written correctly. 1. The count register is checked to + * see if the byte went in, and the write is attempted again + * if not. 2. An overall counter is used to break out so we + * don't hang in those (rare) cases where the UDC reverses + * direction of the FIFO underneath us without notification + * (in response to host aborting a setup transaction early). + * + */ +static void write_fifo( void ) +{ + int bytes_this_time = MIN( wr.bytes_left, 8 ); + int bytes_written = 0; + int i=0; + + PRINTKD( "WF=%d: ", bytes_this_time ); + + while( bytes_this_time-- ) { + PRINTKD( "%2.2X ", *wr.p ); + i = 0; + do { + Ser0UDCD0 = *wr.p; + udelay( 20 ); /* voodo 28Feb01ww */ + i++; + } while( Ser0UDCWC == bytes_written && i < 10 ); + if ( i == 50 ) { + printk( "%swrite_fifo: write failure\n", pszMe ); + usbd_info.stats.ep0_fifo_write_failures++; + } + + wr.p++; + bytes_written++; + } + wr.bytes_left -= bytes_written; + + /* following propagation voodo so maybe caller writing IPR in + ..a moment might actually get it to stick 28Feb01ww */ + udelay( 300 ); + + usbd_info.stats.ep0_bytes_written += bytes_written; + PRINTKD( "L=%d WCR=%8.8X\n", wr.bytes_left, Ser0UDCWC ); +} +/* + * read_fifo() + * Read 1-8 bytes out of FIFO and put in request. + * Called to do the initial read of setup requests + * from the host. Return number of bytes read. + * + * Like write fifo above, this driver uses multiple + * reads checked agains the count register with an + * overall timeout. + * + */ +static int +read_fifo( usb_dev_request_t * request ) +{ + int bytes_read = 0; + int fifo_count; + int i; + + unsigned char * pOut = (unsigned char*) request; + + fifo_count = ( Ser0UDCWC & 0xFF ); + + ASSERT( fifo_count <= 8 ); + PRINTKD( "RF=%d ", fifo_count ); + + while( fifo_count-- ) { + i = 0; + do { + *pOut = (unsigned char) Ser0UDCD0; + udelay( 10 ); + } while( ( Ser0UDCWC & 0xFF ) != fifo_count && i < 10 ); + if ( i == 10 ) { + printk( "%sread_fifo(): read failure\n", pszMe ); + usbd_info.stats.ep0_fifo_read_failures++; + } + pOut++; + bytes_read++; + } + + PRINTKD( "fc=%d\n", bytes_read ); + usbd_info.stats.ep0_bytes_read++; + return bytes_read; +} + +/* + * get_descriptor() + * Called from sh_setup_begin to handle data return + * for a GET_DESCRIPTOR setup request. + */ +static void get_descriptor( usb_dev_request_t * pReq ) +{ + __u32 cs_bits = 0; + string_desc_t * pString; + ep_desc_t * pEndpoint; + + desc_t * pDesc = sa1100_usb_get_descriptor_ptr(); + int type = pReq->wValue >> 8; + int idx = pReq->wValue & 0xFF; + + switch( type ) { + case USB_DESC_DEVICE: + cs_bits = + queue_and_start_write( &pDesc->dev, + pReq->wLength, + pDesc->dev.bLength ); + break; + + // return config descriptor buffer, cfg, intf, 2 ep + case USB_DESC_CONFIG: + cs_bits = + queue_and_start_write( &pDesc->b, + pReq->wLength, + sizeof( struct cdb ) ); + break; + + // not quite right, since doesn't do language code checking + case USB_DESC_STRING: + pString = sa1100_usb_get_string_descriptor( idx ); + if ( pString ) { + if ( idx != 0 ) { // if not language index + printk( "%sReturn string %d: ", pszMe, idx ); + psdesc( pString ); + } + cs_bits = + queue_and_start_write( pString, + pReq->wLength, + pString->bLength ); + } + else { + printk("%sunkown string index %d Stall.\n", pszMe, idx ); + cs_bits = ( UDCCS0_DE | UDCCS0_SO | UDCCS0_FST ); + } + break; + + case USB_DESC_INTERFACE: + if ( idx == pDesc->b.intf.bInterfaceNumber ) { + cs_bits = + queue_and_start_write( &pDesc->b.intf, + pReq->wLength, + pDesc->b.intf.bLength ); + } + break; + + case USB_DESC_ENDPOINT: /* correct? 21Feb01ww */ + if ( idx == 1 ) + pEndpoint = &pDesc->b.ep1; + else if ( idx == 2 ) + pEndpoint = &pDesc->b.ep2; + else + pEndpoint = NULL; + if ( pEndpoint ) { + cs_bits = + queue_and_start_write( pEndpoint, + pReq->wLength, + pEndpoint->bLength ); + } else { + printk("%sunkown endpoint index %d Stall.\n", pszMe, idx ); + cs_bits = ( UDCCS0_DE | UDCCS0_SO | UDCCS0_FST ); + } + break; + + + default : + printk("%sunknown descriptor type %d. Stall.\n", pszMe, type ); + cs_bits = ( UDCCS0_DE | UDCCS0_SO | UDCCS0_FST ); + break; + + } + set_cs_bits( cs_bits ); +} + + +/* some voodo I am adding, since the vanilla macros just aren't doing it 1Mar01ww */ + +#define ABORT_BITS ( UDCCS0_SST | UDCCS0_SE ) +#define OK_TO_WRITE (!( Ser0UDCCS0 & ABORT_BITS )) +#define BOTH_BITS (UDCCS0_IPR | UDCCS0_DE) + +static void set_cs_bits( __u32 bits ) +{ + if ( bits & ( UDCCS0_SO | UDCCS0_SSE | UDCCS0_FST ) ) + Ser0UDCCS0 = bits; + else if ( (bits & BOTH_BITS) == BOTH_BITS ) + set_ipr_and_de(); + else if ( bits & UDCCS0_IPR ) + set_ipr(); + else if ( bits & UDCCS0_DE ) + set_de(); +} + +static void set_de( void ) +{ + int i = 1; + while( 1 ) { + if ( OK_TO_WRITE ) { + Ser0UDCCS0 |= UDCCS0_DE; + } else { + PRINTKD( "%sQuitting set DE because SST or SE set\n", pszMe ); + break; + } + if ( Ser0UDCCS0 & UDCCS0_DE ) + break; + udelay( i ); + if ( ++i == 50 ) { + printk( "%sDangnabbbit! Cannot set DE! (DE=%8.8X CCS0=%8.8X)\n", + pszMe, UDCCS0_DE, Ser0UDCCS0 ); + break; + } + } +} + +static void set_ipr( void ) +{ + int i = 1; + while( 1 ) { + if ( OK_TO_WRITE ) { + Ser0UDCCS0 |= UDCCS0_IPR; + } else { + PRINTKD( "%sQuitting set IPR because SST or SE set\n", pszMe ); + break; + } + if ( Ser0UDCCS0 & UDCCS0_IPR ) + break; + udelay( i ); + if ( ++i == 50 ) { + printk( "%sDangnabbbit! Cannot set IPR! (IPR=%8.8X CCS0=%8.8X)\n", + pszMe, UDCCS0_IPR, Ser0UDCCS0 ); + break; + } + } +} + + + +static void set_ipr_and_de( void ) +{ + int i = 1; + while( 1 ) { + if ( OK_TO_WRITE ) { + Ser0UDCCS0 |= BOTH_BITS; + } else { + PRINTKD( "%sQuitting set IPR/DE because SST or SE set\n", pszMe ); + break; + } + if ( (Ser0UDCCS0 & BOTH_BITS) == BOTH_BITS) + break; + udelay( i ); + if ( ++i == 50 ) { + printk( "%sDangnabbbit! Cannot set DE/IPR! (DE=%8.8X IPR=%8.8X CCS0=%8.8X)\n", + pszMe, UDCCS0_DE, UDCCS0_IPR, Ser0UDCCS0 ); + break; + } + } +} + +static bool clear_opr( void ) +{ + int i = 10000; + bool is_clear; + do { + Ser0UDCCS0 = UDCCS0_SO; + is_clear = ! ( Ser0UDCCS0 & UDCCS0_OPR ); + if ( i-- <= 0 ) { + printk( "%sclear_opr(): failed\n", pszMe ); + break; + } + } while( ! is_clear ); + return is_clear; +} + + + + + +/* end usb_ep0.c */ + diff -urN orig/arch/arm/mach-sa1100/usb_recv.c linux/arch/arm/mach-sa1100/usb_recv.c --- orig/arch/arm/mach-sa1100/usb_recv.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-sa1100/usb_recv.c Tue Oct 16 20:02:24 2001 @@ -0,0 +1,205 @@ +/* + * Generic receive layer for the SA1100 USB client function + * Copyright (c) 2001 by Nicolas Pitre + * + * This code was loosely inspired by the original version which was + * Copyright (c) Compaq Computer Corporation, 1998-1999 + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + * + * This is still work in progress... + * + * Please see linux/Documentation/arm/SA1100/SA1100_USB for details. + */ + +#include +#include +#include +#include +#include + +#include "sa1100_usb.h" +#include "usb_ctl.h" + + +static char *ep1_buf; +static int ep1_len; +static usb_callback_t ep1_callback; +static char *ep1_curdmabuf; +static dma_addr_t ep1_curdmapos; +static int ep1_curdmalen; +static int ep1_remain; +static int dmachn_rx; +static int rx_pktsize; + +static int naking; + +static void +ep1_start(void) +{ + sa1100_dma_flush_all(dmachn_rx); + if (!ep1_curdmalen) { + ep1_curdmalen = rx_pktsize; + if (ep1_curdmalen > ep1_remain) + ep1_curdmalen = ep1_remain; + ep1_curdmapos = pci_map_single(NULL, ep1_curdmabuf, ep1_curdmalen, + PCI_DMA_FROMDEVICE); + } + sa1100_dma_queue_buffer(dmachn_rx, NULL, ep1_curdmapos, ep1_curdmalen); + if ( naking ) { + /* turn off NAK of OUT packets, if set */ + UDC_flip( Ser0UDCCS1, UDCCS1_RPC ); + naking = 0; + } +} + +static void +ep1_done(int flag) +{ + int size = ep1_len - ep1_remain; + + if (!ep1_len) + return; + if (ep1_curdmalen) + pci_unmap_single(NULL, ep1_curdmapos, ep1_curdmalen, + PCI_DMA_FROMDEVICE); + ep1_len = ep1_curdmalen = 0; + if (ep1_callback) { + ep1_callback(flag, size); + } +} + +void +ep1_state_change_notify( int new_state ) +{ + +} + +void +ep1_stall( void ) +{ + /* SET_FEATURE force stall at UDC */ + UDC_set( Ser0UDCCS1, UDCCS1_FST ); +} + +int +ep1_init(int chn) +{ + desc_t * pd = sa1100_usb_get_descriptor_ptr(); + rx_pktsize = __le16_to_cpu( pd->b.ep1.wMaxPacketSize ); + dmachn_rx = chn; + sa1100_dma_flush_all(dmachn_rx); + ep1_done(-EAGAIN); + return 0; +} + +void +ep1_reset(void) +{ + desc_t * pd = sa1100_usb_get_descriptor_ptr(); + rx_pktsize = __le16_to_cpu( pd->b.ep1.wMaxPacketSize ); + sa1100_dma_flush_all(dmachn_rx); + UDC_clear(Ser0UDCCS1, UDCCS1_FST); + ep1_done(-EINTR); +} + +void +ep1_int_hndlr(int udcsr) +{ + dma_addr_t dma_addr; + unsigned int len; + int status = Ser0UDCCS1; + + if ( naking ) printk( "%sEh? in ISR but naking = %d\n", "usbrx: ", naking ); + + if (status & UDCCS1_RPC) { + + if (!ep1_curdmalen) { + printk("usb_recv: RPC for non-existent buffer\n"); + naking=1; + return; + } + + sa1100_dma_stop(dmachn_rx); + + if (status & UDCCS1_SST) { + printk("usb_recv: stall sent OMP=%d\n",Ser0UDCOMP); + UDC_flip(Ser0UDCCS1, UDCCS1_SST); + ep1_done(-EIO); // UDC aborted current transfer, so we do + return; + } + + if (status & UDCCS1_RPE) { + printk("usb_recv: RPError %x\n", status); + UDC_flip(Ser0UDCCS1, UDCCS1_RPC); + ep1_done(-EIO); + return; + } + + sa1100_dma_get_current(dmachn_rx, NULL, &dma_addr); + pci_unmap_single(NULL, ep1_curdmapos, ep1_curdmalen, + PCI_DMA_FROMDEVICE); + len = dma_addr - ep1_curdmapos; + if (len < ep1_curdmalen) { + char *buf = ep1_curdmabuf + len; + while (Ser0UDCCS1 & UDCCS1_RNE) { + if (len >= ep1_curdmalen) { + printk("usb_recv: too much data in fifo\n"); + break; + } + *buf++ = Ser0UDCDR; + len++; + } + } else if (Ser0UDCCS1 & UDCCS1_RNE) { + printk("usb_recv: fifo screwed, shouldn't contain data\n"); + len = 0; + } + ep1_curdmalen = 0; /* dma unmap already done */ + ep1_remain -= len; + naking = 1; + ep1_done((len) ? 0 : -EPIPE); + } + /* else, you can get here if we are holding NAK */ +} + +int +sa1100_usb_recv(char *buf, int len, usb_callback_t callback) +{ + int flags; + + if (ep1_len) + return -EBUSY; + + local_irq_save(flags); + ep1_buf = buf; + ep1_len = len; + ep1_callback = callback; + ep1_remain = len; + ep1_curdmabuf = buf; + ep1_curdmalen = 0; + ep1_start(); + local_irq_restore(flags); + + return 0; +} + +EXPORT_SYMBOL(sa1100_usb_recv); + +void +sa1100_usb_recv_reset(void) +{ + ep1_reset(); +} + +EXPORT_SYMBOL(sa1100_usb_recv_reset); + +void +sa1100_usb_recv_stall(void) +{ + ep1_stall(); +} + +EXPORT_SYMBOL(sa1100_usb_recv_stall); + diff -urN orig/arch/arm/mach-sa1100/usb_send.c linux/arch/arm/mach-sa1100/usb_send.c --- orig/arch/arm/mach-sa1100/usb_send.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mach-sa1100/usb_send.c Tue Oct 16 20:02:24 2001 @@ -0,0 +1,205 @@ +/* + * Generic xmit layer for the SA1100 USB client function + * Copyright (c) 2001 by Nicolas Pitre + * + * This code was loosely inspired by the original version which was + * Copyright (c) Compaq Computer Corporation, 1998-1999 + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + * + * This is still work in progress... + * + * Please see linux/Documentation/arm/SA1100/SA1100_USB for details. + * 15/03/2001 - ep2_start now sets UDCAR to overcome something that is hardware + * bug, I think. green@iXcelerator.com + */ + +#include +#include +#include +#include // for the massive_attack hack 28Feb01ww +#include +#include +#include +#include + +#include "sa1100_usb.h" +#include "usb_ctl.h" + + +static char *ep2_buf; +static int ep2_len; +static usb_callback_t ep2_callback; +static dma_addr_t ep2_dma; +static dma_addr_t ep2_curdmapos; +static int ep2_curdmalen; +static int ep2_remain; +static int dmachn_tx; +static int tx_pktsize; + +/* device state is changing, async */ +void +ep2_state_change_notify( int new_state ) +{ +} + +/* set feature stall executing, async */ +void +ep2_stall( void ) +{ + UDC_set( Ser0UDCCS2, UDCCS2_FST ); /* force stall at UDC */ +} + +static void +ep2_start(void) +{ + if (!ep2_len) + return; + + ep2_curdmalen = tx_pktsize; + if (ep2_curdmalen > ep2_remain) + ep2_curdmalen = ep2_remain; + + /* must do this _before_ queue buffer.. */ + UDC_flip( Ser0UDCCS2,UDCCS2_TPC ); /* stop NAKing IN tokens */ + UDC_write( Ser0UDCIMP, ep2_curdmalen-1 ); + + /* Remove if never seen...8Mar01ww */ + { + int massive_attack = 20; + while ( Ser0UDCIMP != ep2_curdmalen-1 && massive_attack-- ) { + printk( "usbsnd: Oh no you don't! Let me spin..." ); + udelay( 500 ); + printk( "and try again...\n" ); + UDC_write( Ser0UDCIMP, ep2_curdmalen-1 ); + } + if ( massive_attack != 20 ) { + if ( Ser0UDCIMP != ep2_curdmalen-1 ) + printk( "usbsnd: Massive attack FAILED :-( %d\n", + 20 - massive_attack ); + else + printk( "usbsnd: Massive attack WORKED :-) %d\n", + 20 - massive_attack ); + } + } + /* End remove if never seen... 8Mar01ww */ + + Ser0UDCAR = usbd_info.address; // fighting stupid silicon bug + sa1100_dma_queue_buffer(dmachn_tx, NULL, ep2_curdmapos, ep2_curdmalen); +} + +static void +ep2_done(int flag) +{ + int size = ep2_len - ep2_remain; + if (ep2_len) { + pci_unmap_single(NULL, ep2_dma, ep2_len, PCI_DMA_TODEVICE); + ep2_len = 0; + if (ep2_callback) + ep2_callback(flag, size); + } +} + +int +ep2_init(int chn) +{ + desc_t * pd = sa1100_usb_get_descriptor_ptr(); + tx_pktsize = __le16_to_cpu( pd->b.ep2.wMaxPacketSize ); + dmachn_tx = chn; + sa1100_dma_flush_all(dmachn_tx); + ep2_done(-EAGAIN); + return 0; +} + +void +ep2_reset(void) +{ + desc_t * pd = sa1100_usb_get_descriptor_ptr(); + tx_pktsize = __le16_to_cpu( pd->b.ep2.wMaxPacketSize ); + UDC_clear(Ser0UDCCS2, UDCCS2_FST); + sa1100_dma_flush_all(dmachn_tx); + ep2_done(-EINTR); +} + +void +ep2_int_hndlr(int udcsr) +{ + int status = Ser0UDCCS2; + + if (Ser0UDCAR != usbd_info.address) // check for stupid silicon bug. + Ser0UDCAR = usbd_info.address; + + UDC_flip(Ser0UDCCS2, UDCCS2_SST); + + if (status & UDCCS2_TPC) { + sa1100_dma_flush_all(dmachn_tx); + + if (status & (UDCCS2_TPE | UDCCS2_TUR)) { + printk("usb_send: transmit error %x\n", status); + ep2_done(-EIO); + } else { +#if 1 // 22Feb01ww/Oleg + ep2_curdmapos += ep2_curdmalen; + ep2_remain -= ep2_curdmalen; +#else + ep2_curdmapos += Ser0UDCIMP + 1; // this is workaround + ep2_remain -= Ser0UDCIMP + 1; // for case when setting of Ser0UDCIMP was failed +#endif + + if (ep2_remain != 0) { + ep2_start(); + } else { + ep2_done(0); + } + } + } else { + printk("usb_send: Not TPC: UDCCS2 = %x\n", status); + } +} + +int +sa1100_usb_send(char *buf, int len, usb_callback_t callback) +{ + int flags; + + if (usbd_info.state != USB_STATE_CONFIGURED) + return -ENODEV; + + if (ep2_len) + return -EBUSY; + + local_irq_save(flags); + ep2_buf = buf; + ep2_len = len; + ep2_dma = pci_map_single(NULL, buf, len, PCI_DMA_TODEVICE); + ep2_callback = callback; + ep2_remain = len; + ep2_curdmapos = ep2_dma; + ep2_start(); + local_irq_restore(flags); + + return 0; +} + + +void +sa1100_usb_send_reset(void) +{ + ep2_reset(); +} + +int sa1100_usb_xmitter_avail( void ) +{ + if (usbd_info.state != USB_STATE_CONFIGURED) + return -ENODEV; + if (ep2_len) + return -EBUSY; + return 0; +} + + +EXPORT_SYMBOL(sa1100_usb_xmitter_avail); +EXPORT_SYMBOL(sa1100_usb_send); +EXPORT_SYMBOL(sa1100_usb_send_reset); diff -urN orig/arch/arm/mach-shark/pci.c linux/arch/arm/mach-shark/pci.c --- orig/arch/arm/mach-shark/pci.c Tue Oct 3 20:08:19 2000 +++ linux/arch/arm/mach-shark/pci.c Sun Sep 15 12:56:16 2002 @@ -21,7 +21,7 @@ } struct hw_pci shark_pci __initdata = { - init: via82c505_init, - swizzle: no_swizzle, - map_irq: shark_map_irq + .init = via82c505_init, + .swizzle = no_swizzle, + .map_irq = shark_map_irq, }; diff -urN orig/arch/arm/mm/Makefile linux/arch/arm/mm/Makefile --- orig/arch/arm/mm/Makefile Mon Aug 5 13:29:47 2002 +++ linux/arch/arm/mm/Makefile Thu Dec 12 22:40:17 2002 @@ -36,8 +36,10 @@ p-$(CONFIG_CPU_ARM720T) += proc-arm720.o p-$(CONFIG_CPU_ARM920T) += proc-arm920.o p-$(CONFIG_CPU_ARM922T) += proc-arm922.o +p-$(CONFIG_CPU_ARM925T) += proc-arm925.o p-$(CONFIG_CPU_ARM926T) += proc-arm926.o p-$(CONFIG_CPU_ARM1020) += proc-arm1020.o +p-$(CONFIG_CPU_ARM1026) += proc-arm1026.o p-$(CONFIG_CPU_SA110) += proc-sa110.o p-$(CONFIG_CPU_SA1100) += proc-sa110.o diff -urN orig/arch/arm/mm/alignment.c linux/arch/arm/mm/alignment.c --- orig/arch/arm/mm/alignment.c Mon Aug 5 13:29:47 2002 +++ linux/arch/arm/mm/alignment.c Mon Aug 5 22:25:06 2002 @@ -426,13 +426,13 @@ eaddr += 4; /* - * For alignment faults on the ARM922T the MMU makes + * For alignment faults on the ARM922T/ARM920T the MMU makes * the FSR (and hence addr) equal to the updated base address * of the multiple access rather than the restored value. - * Switch this messsage off if we've got a ARM922, otherwise + * Switch this messsage off if we've got a ARM92[02], otherwise * [ls]dm alignment faults are noisy! */ -#if !(defined CONFIG_CPU_ARM922T) +#if !(defined CONFIG_CPU_ARM922T) && !(defined CONFIG_CPU_ARM920T) /* * This is a "hint" - we already have eaddr worked out by the * processor for us. diff -urN orig/arch/arm/mm/consistent.c linux/arch/arm/mm/consistent.c --- orig/arch/arm/mm/consistent.c Wed Jul 4 19:53:31 2001 +++ linux/arch/arm/mm/consistent.c Thu Dec 12 22:33:33 2002 @@ -30,14 +30,20 @@ * now, we expressly forbid it, especially as some of the stuff we do * here is not interrupt context safe. * + * We should allow this function to be called from interrupt context. + * However, we call ioremap, which needs to fiddle around with various + * things (like the vmlist_lock, and allocating page tables). These + * things aren't interrupt safe (yet). + * * Note that this does *not* zero the allocated area! */ void *consistent_alloc(int gfp, size_t size, dma_addr_t *dma_handle) { struct page *page, *end, *free; unsigned long order; - void *ret, *virt; + void *ret; + /* FIXME */ if (in_interrupt()) BUG(); @@ -48,22 +54,22 @@ if (!page) goto no_page; - /* - * We could do with a page_to_phys and page_to_bus here. - */ - virt = page_address(page); - *dma_handle = virt_to_bus(virt); - ret = __ioremap(virt_to_phys(virt), size, 0); + *dma_handle = page_to_bus(page); + ret = __ioremap(page_to_pfn(page) << PAGE_SHIFT, size, 0); if (!ret) goto no_remap; #if 0 /* ioremap_does_flush_cache_all */ - /* - * we need to ensure that there are no cachelines in use, or - * worse dirty in this area. Really, we don't need to do - * this since __ioremap does a flush_cache_all() anyway. --rmk - */ - invalidate_dcache_range(virt, virt + size); + { + void *virt = page_address(page); + + /* + * we need to ensure that there are no cachelines in use, or + * worse dirty in this area. Really, we don't need to do + * this since __ioremap does a flush_cache_all() anyway. --rmk + */ + invalidate_dcache_range(virt, virt + size); + } #endif /* @@ -72,7 +78,6 @@ * We also mark the pages in use as reserved so that * remap_page_range works. */ - page = virt_to_page(virt); free = page + (size >> PAGE_SHIFT); end = page + (1 << order); @@ -93,18 +98,15 @@ void *pci_alloc_consistent(struct pci_dev *hwdev, size_t size, dma_addr_t *handle) { - void *__ret; - int __gfp = GFP_KERNEL; + int gfp = GFP_KERNEL; -#ifdef CONFIG_PCI - if ((hwdev) == NULL || +#if defined(CONFIG_PCI) || defined(CONFIG_SA1111) + if ((hwdev) == NULL || dev_is_sa1111(hwdev) || (hwdev)->dma_mask != 0xffffffff) #endif - __gfp |= GFP_DMA; + gfp |= GFP_DMA; - __ret = consistent_alloc(__gfp, (size), - (handle)); - return __ret; + return consistent_alloc(gfp, size, handle); } /* @@ -114,19 +116,16 @@ void consistent_free(void *vaddr, size_t size, dma_addr_t handle) { struct page *page, *end; - void *virt; if (in_interrupt()) BUG(); - virt = bus_to_virt(handle); - /* * More messing around with the MM internals. This is * sick, but then so is remap_page_range(). */ size = PAGE_ALIGN(size); - page = virt_to_page(virt); + page = virt_to_page(bus_to_virt(handle)); end = page + (size >> PAGE_SHIFT); for (; page < end; page++) diff -urN orig/arch/arm/mm/discontig.c linux/arch/arm/mm/discontig.c --- orig/arch/arm/mm/discontig.c Mon Sep 3 14:14:41 2001 +++ linux/arch/arm/mm/discontig.c Fri Jun 21 14:11:37 2002 @@ -20,7 +20,7 @@ #endif /* - * Our node_data structure for discontigous memory. + * Our node_data structure for discontiguous memory. */ static bootmem_data_t node_bootmem_data[NR_NODES]; diff -urN orig/arch/arm/mm/fault-armv.c linux/arch/arm/mm/fault-armv.c --- orig/arch/arm/mm/fault-armv.c Mon Aug 5 13:29:47 2002 +++ linux/arch/arm/mm/fault-armv.c Wed Feb 12 17:21:26 2003 @@ -28,7 +28,6 @@ #include #include -extern void die_if_kernel(const char *str, struct pt_regs *regs, int err); extern void show_pte(struct mm_struct *mm, unsigned long addr); extern int do_page_fault(unsigned long addr, int error_code, struct pt_regs *regs); @@ -236,9 +235,13 @@ */ void update_mmu_cache(struct vm_area_struct *vma, unsigned long addr, pte_t pte) { - struct page *page = pte_page(pte); + unsigned long pfn = pte_pfn(pte); + struct page *page; - if (VALID_PAGE(page) && page->mapping) { + if (!pfn_valid(pfn)) + return; + page = pfn_to_page(pfn); + if (page->mapping) { if (test_and_clear_bit(PG_dcache_dirty, &page->flags)) { unsigned long kvirt = (unsigned long)page_address(page); cpu_cache_clean_invalidate_range(kvirt, kvirt + PAGE_SIZE, 0); diff -urN orig/arch/arm/mm/fault-common.c linux/arch/arm/mm/fault-common.c --- orig/arch/arm/mm/fault-common.c Mon Aug 5 13:29:47 2002 +++ linux/arch/arm/mm/fault-common.c Tue Dec 24 19:32:16 2002 @@ -43,52 +43,77 @@ #define READ_FAULT(m) (!(m)) #endif -NORET_TYPE void die(const char *msg, struct pt_regs *regs, int err) ATTRIB_NORET; - /* * This is useful to dump out the page tables associated with * 'addr' in mm 'mm'. */ void show_pte(struct mm_struct *mm, unsigned long addr) { - pgd_t *pgd; + mm_segment_t fs; if (!mm) mm = &init_mm; - printk(KERN_ALERT "pgd = %p\n", mm->pgd); - pgd = pgd_offset(mm, addr); - printk(KERN_ALERT "*pgd = %08lx", pgd_val(*pgd)); + printk(KERN_ALERT "mm = %p pgd = %p\n", mm, mm->pgd); + fs = get_fs(); + set_fs(get_ds()); do { - pmd_t *pmd; - pte_t *pte; + pgd_t pg, *pgd = pgd_offset(mm, addr); + pmd_t pm, *pmd; + pte_t pt, *pte; + + printk(KERN_ALERT "*pgd = "); - if (pgd_none(*pgd)) + if (__get_user(pgd_val(pg), (unsigned long *)pgd)) { + printk("(faulted)"); break; + } - if (pgd_bad(*pgd)) { + printk("%08lx", pgd_val(pg)); + + if (pgd_none(pg)) + break; + + if (pgd_bad(pg)) { printk("(bad)"); break; } pmd = pmd_offset(pgd, addr); - printk(", *pmd = %08lx", pmd_val(*pmd)); - if (pmd_none(*pmd)) + printk(", *pmd = "); + + if (__get_user(pmd_val(pm), (unsigned long *)pmd)) { + printk("(faulted)"); + break; + } + + printk("%08lx", pmd_val(pm)); + + if (pmd_none(pm)) break; - if (pmd_bad(*pmd)) { + if (pmd_bad(pm)) { printk("(bad)"); break; } pte = pte_offset(pmd, addr); - printk(", *pte = %08lx", pte_val(*pte)); + + printk(", *pte = "); + + if (__get_user(pte_val(pt), (unsigned long *)pte)) { + printk("(faulted)"); + break; + } + + printk("%08lx", pte_val(pt)); #ifdef CONFIG_CPU_32 printk(", *ppte = %08lx", pte_val(pte[-PTRS_PER_PTE])); #endif } while(0); + set_fs(fs); printk("\n"); } @@ -346,6 +371,9 @@ offset = __pgd_offset(addr); + /* + * FIXME: CP15 C1 is write only on ARMv3 architectures. + */ pgd = cpu_get_pgd() + offset; pgd_k = init_mm.pgd + offset; diff -urN orig/arch/arm/mm/init.c linux/arch/arm/mm/init.c --- orig/arch/arm/mm/init.c Fri Oct 26 16:45:58 2001 +++ linux/arch/arm/mm/init.c Thu Jan 23 23:56:26 2003 @@ -1,7 +1,7 @@ /* * linux/arch/arm/mm/init.c * - * Copyright (C) 1995-2000 Russell King + * Copyright (C) 1995-2002 Russell King * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as @@ -46,11 +46,13 @@ #define TABLE_OFFSET 0 #endif -#define TABLE_SIZE ((TABLE_OFFSET + PTRS_PER_PTE) * sizeof(void *)) +#define TABLE_SIZE ((TABLE_OFFSET + PTRS_PER_PTE) * sizeof(pte_t)) static unsigned long totalram_pages; extern pgd_t swapper_pg_dir[PTRS_PER_PGD]; extern char _stext, _text, _etext, _end, __init_begin, __init_end; +extern unsigned long phys_initrd_start; +extern unsigned long phys_initrd_size; /* * The sole use of this is to pass memory configuration @@ -292,6 +294,7 @@ * also get rid of some of the stuff above as well. */ max_low_pfn = memend_pfn - O_PFN_DOWN(PHYS_OFFSET); +// max_pfn = memend_pfn - O_PFN_DOWN(PHYS_OFFSET); mi->end = memend_pfn << PAGE_SHIFT; return bootmem_pages; @@ -300,18 +303,17 @@ static int __init check_initrd(struct meminfo *mi) { int initrd_node = -2; + unsigned long end = phys_initrd_start + phys_initrd_size; #ifdef CONFIG_BLK_DEV_INITRD /* * Make sure that the initrd is within a valid area of * memory. */ - if (initrd_start) { - unsigned long phys_initrd_start, phys_initrd_end; + if (phys_initrd_size) { unsigned int i; - phys_initrd_start = __pa(initrd_start); - phys_initrd_end = __pa(initrd_end); + initrd_node = -1; for (i = 0; i < mi->nr_banks; i++) { unsigned long bank_end; @@ -319,7 +321,7 @@ bank_end = mi->bank[i].start + mi->bank[i].size; if (mi->bank[i].start <= phys_initrd_start && - phys_initrd_end <= bank_end) + end <= bank_end) initrd_node = mi->bank[i].node; } } @@ -327,8 +329,8 @@ if (initrd_node == -1) { printk(KERN_ERR "initrd (0x%08lx - 0x%08lx) extends beyond " "physical memory - disabling initrd\n", - initrd_start, initrd_end); - initrd_start = initrd_end = 0; + phys_initrd_start, end); + phys_initrd_start = phys_initrd_size = 0; } #endif @@ -354,7 +356,7 @@ * and can only be in node 0. */ reserve_bootmem_node(pgdat, __pa(swapper_pg_dir), - PTRS_PER_PGD * sizeof(void *)); + PTRS_PER_PGD * sizeof(pgd_t)); #endif /* * And don't forget to reserve the allocator bitmap, @@ -378,7 +380,7 @@ */ if (machine_is_archimedes() || machine_is_a5k()) reserve_bootmem_node(pgdat, 0x02000000, 0x00080000); - if (machine_is_edb7211()) + if (machine_is_edb7211() || machine_is_fortunet()) reserve_bootmem_node(pgdat, 0xc0000000, 0x00020000); if (machine_is_p720t()) reserve_bootmem_node(pgdat, PHYS_OFFSET, 0x00014000); @@ -469,9 +471,12 @@ #ifdef CONFIG_BLK_DEV_INITRD - if (initrd_node >= 0) - reserve_bootmem_node(NODE_DATA(initrd_node), __pa(initrd_start), - initrd_end - initrd_start); + if (phys_initrd_size && initrd_node >= 0) { + reserve_bootmem_node(NODE_DATA(initrd_node), phys_initrd_start, + phys_initrd_size); + initrd_start = __phys_to_virt(phys_initrd_start); + initrd_end = initrd_start + phys_initrd_size; + } #endif if (map_pg != bootmap_pfn + bootmap_pages) @@ -501,6 +506,7 @@ memtable_init(mi); if (mdesc->map_io) mdesc->map_io(); + flush_cache_all(); flush_tlb_all(); /* @@ -583,7 +589,7 @@ } if (size && s) - printk("Freeing %s memory: %dK\n", s, size); + printk(KERN_INFO "Freeing %s memory: %dK\n", s, size); } /* diff -urN orig/arch/arm/mm/ioremap.c linux/arch/arm/mm/ioremap.c --- orig/arch/arm/mm/ioremap.c Sat Apr 28 11:24:54 2001 +++ linux/arch/arm/mm/ioremap.c Fri Sep 6 11:25:14 2002 @@ -8,24 +8,19 @@ * Hacked for ARM by Phil Blundell * Hacked to allow all architectures to build, and various cleanups * by Russell King - */ - -/* + * * This allows a driver to remap an arbitrary region of bus memory into * virtual space. One should *only* use readl, writel, memcpy_toio and * so on with such remapped areas. * - * Because the ARM only has a 32-bit address space we can't address the - * whole of the (physical) PCI space at once. PCI huge-mode addressing - * allows us to circumvent this restriction by splitting PCI space into - * two 2GB chunks and mapping only one at a time into processor memory. - * We use MMU protection domains to trap any attempt to access the bank - * that is not currently mapped. (This isn't fully implemented yet.) - * - * DC21285 currently has a bug in that the PCI address extension - * register affects the address of any writes waiting in the outbound - * FIFO. Unfortunately, it is not possible to tell the DC21285 to - * flush this - flushing the area causes the bus to lock. + * ioremap support tweaked to allow support for large page mappings. We + * have several issues that needs to be resolved first however: + * + * 1. We need set_pte, or something like set_pte to understand large + * page mappings. + * + * 2. we need the unmap_* functions to likewise understand large page + * mappings. */ #include #include @@ -35,8 +30,9 @@ #include #include -static inline void remap_area_pte(pte_t * pte, unsigned long address, unsigned long size, - unsigned long phys_addr, pgprot_t pgprot) +static inline void +remap_area_pte(pte_t * pte, unsigned long address, unsigned long size, + unsigned long pfn, pgprot_t pgprot) { unsigned long end; @@ -44,22 +40,26 @@ end = address + size; if (end > PMD_SIZE) end = PMD_SIZE; - if (address >= end) - BUG(); + BUG_ON(address >= end); do { - if (!pte_none(*pte)) { - printk("remap_area_pte: page already exists\n"); - BUG(); - } - set_pte(pte, mk_pte_phys(phys_addr, pgprot)); + if (!pte_none(*pte)) + goto bad; + + set_pte(pte, pfn_pte(pfn, pgprot)); address += PAGE_SIZE; - phys_addr += PAGE_SIZE; + pfn++; pte++; } while (address && (address < end)); + return; + + bad: + printk("remap_area_pte: page already exists\n"); + BUG(); } -static inline int remap_area_pmd(pmd_t * pmd, unsigned long address, unsigned long size, - unsigned long phys_addr, unsigned long flags) +static inline int +remap_area_pmd(pmd_t * pmd, unsigned long address, unsigned long size, + unsigned long pfn, unsigned long flags) { unsigned long end; pgprot_t pgprot; @@ -70,34 +70,33 @@ if (end > PGDIR_SIZE) end = PGDIR_SIZE; - phys_addr -= address; - if (address >= end) - BUG(); + pfn -= address >> PAGE_SHIFT; + BUG_ON(address >= end); pgprot = __pgprot(L_PTE_PRESENT | L_PTE_YOUNG | L_PTE_DIRTY | L_PTE_WRITE | flags); do { pte_t * pte = pte_alloc(&init_mm, pmd, address); if (!pte) return -ENOMEM; - remap_area_pte(pte, address, end - address, address + phys_addr, pgprot); + remap_area_pte(pte, address, end - address, pfn + (address >> PAGE_SHIFT), pgprot); address = (address + PMD_SIZE) & PMD_MASK; pmd++; } while (address && (address < end)); return 0; } -static int remap_area_pages(unsigned long address, unsigned long phys_addr, - unsigned long size, unsigned long flags) +static int +remap_area_pages(unsigned long address, unsigned long pfn, + unsigned long size, unsigned long flags) { int error; pgd_t * dir; unsigned long end = address + size; - phys_addr -= address; + pfn -= address >> PAGE_SHIFT; dir = pgd_offset(&init_mm, address); flush_cache_all(); - if (address >= end) - BUG(); + BUG_ON(address >= end); spin_lock(&init_mm.page_table_lock); do { pmd_t *pmd; @@ -106,7 +105,7 @@ if (!pmd) break; if (remap_area_pmd(pmd, address, end - address, - phys_addr + address, flags)) + pfn + (address >> PAGE_SHIFT), flags)) break; error = 0; address = (address + PGDIR_SIZE) & PGDIR_MASK; @@ -154,7 +153,7 @@ if (!area) return NULL; addr = area->addr; - if (remap_area_pages(VMALLOC_VMADDR(addr), phys_addr, size, flags)) { + if (remap_area_pages(VMALLOC_VMADDR(addr), phys_addr >> PAGE_SHIFT, size, flags)) { vfree(addr); return NULL; } diff -urN orig/arch/arm/mm/mm-armv.c linux/arch/arm/mm/mm-armv.c --- orig/arch/arm/mm/mm-armv.c Mon Aug 5 13:29:47 2002 +++ linux/arch/arm/mm/mm-armv.c Thu Jan 23 23:56:36 2003 @@ -200,7 +200,7 @@ } ptep = pte_offset(pmdp, virt); - set_pte(ptep, mk_pte_phys(phys, __pgprot(prot))); + set_pte(ptep, pfn_pte(phys >> PAGE_SHIFT, __pgprot(prot))); } /* @@ -388,8 +388,6 @@ init_maps->bufferable = 0; create_mapping(init_maps); - - flush_cache_all(); } /* diff -urN orig/arch/arm/mm/mm-riscstation.c linux/arch/arm/mm/mm-riscstation.c --- orig/arch/arm/mm/mm-riscstation.c Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mm/mm-riscstation.c Sun Sep 15 20:18:44 2002 @@ -0,0 +1,35 @@ +/* + * linux/arch/arm/mm/mm-rpc.c + * linux/arch/arm/mm/mm-riscstation.c + * + * Copyright (C) 1998-1999 Russell King + * Copyright (C) 2002 Simtec Electronics / Ben Dooks + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + * + * Extra MM routines for RiscStation + */ +#include +#include + +#include +#include +#include +#include + +#include + +/* map the EASI space to use for the ISA slot */ + +static struct map_desc riscstation_io_desc[] __initdata = { + { IO_BASE, IO_START, IO_SIZE , DOMAIN_IO, 0, 1, 0, 0 }, /* IO space */ + { EASI_BASE, EASI_START, EASI_SIZE, DOMAIN_IO, 0, 1, 0, 0 }, /* EASI space */ + LAST_DESC +}; + +void __init riscstation_map_io(void) +{ + iotable_init(riscstation_io_desc); +} diff -urN orig/arch/arm/mm/proc-arm1020.S linux/arch/arm/mm/proc-arm1020.S --- orig/arch/arm/mm/proc-arm1020.S Mon Aug 5 13:29:47 2002 +++ linux/arch/arm/mm/proc-arm1020.S Thu Jan 23 20:40:31 2003 @@ -135,7 +135,7 @@ .align 5 ENTRY(cpu_arm1020_do_idle) mcr p15, 0, r0, c7, c0, 4 @ Wait for interrupt -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE mov r0, r0 mov r0, r0 #endif @@ -156,7 +156,7 @@ ENTRY(cpu_arm1020_cache_clean_invalidate_all) mov r2, #1 cpu_arm1020_cache_clean_invalidate_all_r2: -#ifdef CONFIG_CPU_ARM1020_D_CACHE_ON +#ifndef CONFIG_CPU_DCACHE_DISABLE mcr p15, 0, ip, c7, c10, 4 mov r1, #0xf @ 16 segments @@ -165,7 +165,7 @@ orr ip, ip, r1, LSL #5 @ shift in/up index mcr p15, 0, ip, c7, c14, 2 @ Clean & Inval DCache entry mcr p15, 0, ip, c7, c10, 4 @ drain WB -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE mov ip, ip #endif subs r3, r3, #1 @@ -176,12 +176,12 @@ bge 1b @ segments 7 to 0 #endif -#ifdef CONFIG_CPU_ARM1020_I_CACHE_ON +#ifndef CONFIG_CPU_ICACHE_DISABLE teq r2, #0 mcrne p15, 0, ip, c7, c5, 0 @ invalidate I cache #endif mcr p15, 0, ip, c7, c10, 4 @ drain WB -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE mov ip, ip mov ip, ip #endif @@ -203,13 +203,13 @@ cmp r3, #MAX_AREA_SIZE bgt cpu_arm1020_cache_clean_invalidate_all_r2 mcr p15, 0, r3, c7, c10, 4 -#ifdef CONFIG_CPU_ARM1020_D_CACHE_ON +#ifndef CONFIG_CPU_DCACHE_DISABLE 1: mcr p15, 0, r0, c7, c14, 1 @ clean and invalidate D entry mcr p15, 0, r3, c7, c10, 4 @ drain WB add r0, r0, #DCACHELINESIZE mcr p15, 0, r0, c7, c14, 1 @ clean and invalidate D entry mcr p15, 0, r3, c7, c10, 4 @ drain WB -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE mov r0, r0 #endif add r0, r0, #DCACHELINESIZE @@ -217,11 +217,11 @@ blt 1b #endif -#ifdef CONFIG_CPU_ARM1020_I_CACHE_ON +#ifndef CONFIG_CPU_ICACHE_DISABLE teq r2, #0 movne r0, #0 mcrne p15, 0, r0, c7, c5, 0 @ invalidate I cache -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE mov r0, r0 mov r0, r0 #endif @@ -238,14 +238,14 @@ .align 5 ENTRY(cpu_arm1020_flush_ram_page) mcr p15, 0, r1, c7, c10, 4 -#ifdef CONFIG_CPU_ARM1020_D_CACHE_ON +#ifndef CONFIG_CPU_DCACHE_DISABLE mov r1, #PAGESIZE 1: mcr p15, 0, r0, c7, c10, 1 @ clean D entry mcr p15, 0, r0, c7, c10, 4 @ drain WB add r0, r0, #DCACHELINESIZE mcr p15, 0, r0, c7, c10, 1 @ clean D entry mcr p15, 0, r0, c7, c10, 4 @ drain WB -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE mov r0, r0 mov r0, r0 #endif @@ -254,7 +254,7 @@ mov r0, #0 #endif -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE mov r0, r0 mov r0, r0 #endif @@ -275,14 +275,14 @@ */ .align 5 ENTRY(cpu_arm1020_dcache_invalidate_range) -#ifdef CONFIG_CPU_ARM1020_D_CACHE_ON +#ifndef CONFIG_CPU_DCACHE_DISABLE /* D cache are on */ tst r0, #DCACHELINESIZE - 1 bic r0, r0, #DCACHELINESIZE - 1 mcrne p15, 0, r0, c7, c10, 4 mcrne p15, 0, r0, c7, c10, 1 @ clean D entry at start mcrne p15, 0, r0, c7, c10, 4 @ drain WB -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE mov r0, r0 mov r0, r0 #endif @@ -290,14 +290,14 @@ mcrne p15, 0, r1, c7, c10, 4 mcrne p15, 0, r1, c7, c10, 1 @ clean D entry at end mcrne p15, 0, r1, c7, c10, 4 @ drain WB -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE mov r1, r1 mov r1, r1 mov r1, r1 #endif 1: mcr p15, 0, r0, c7, c6, 1 @ invalidate D entry -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE mov r0, r0 #endif add r0, r0, #DCACHELINESIZE @@ -306,7 +306,7 @@ #else /* D cache off, but still drain the write buffer */ mcr p15, 0, r0, c7, c10, 4 @ Drain write buffer -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE mov r0, r0 mov r0, r0 #endif @@ -330,13 +330,13 @@ cmp r3, #MAX_AREA_SIZE bgt cpu_arm1020_cache_clean_invalidate_all_r2 mcr p15, 0, r3, c7, c10, 4 -#ifdef CONFIG_CPU_ARM1020_D_CACHE_ON +#ifndef CONFIG_CPU_DCACHE_DISABLE 1: mcr p15, 0, r0, c7, c14, 1 @ clean and invalidate D entry mcr p15, 0, r3, c7, c10, 4 @ drain WB add r0, r0, #DCACHELINESIZE mcr p15, 0, r0, c7, c14, 1 @ clean and invalidate D entry mcr p15, 0, r3, c7, c10, 4 @ drain WB -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE mov r0, r0 #endif add r0, r0, #DCACHELINESIZE @@ -344,7 +344,7 @@ blt 1b #endif -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE mov r0, r0 mov r0, r0 #endif @@ -367,23 +367,23 @@ ENTRY(cpu_arm1020_dcache_clean_page) mov r1, #PAGESIZE mcr p15, 0, r0, c7, c10, 4 -#ifdef CONFIG_CPU_ARM1020_D_CACHE_ON +#ifndef CONFIG_CPU_DCACHE_DISABLE 1: mcr p15, 0, r0, c7, c10, 1 @ clean D entry (drain is done by TLB fns) mcr p15, 0, r0, c7, c10, 4 @ drain WB -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE mov r0, r0 #endif add r0, r0, #DCACHELINESIZE mcr p15, 0, r0, c7, c10, 1 @ clean D entry mcr p15, 0, r0, c7, c10, 4 @ drain WB -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE mov r0, r0 #endif add r0, r0, #DCACHELINESIZE subs r1, r1, #2 * DCACHELINESIZE bhi 1b #endif -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE mov r0, r0 mov r0, r0 #endif @@ -401,14 +401,14 @@ ENTRY(cpu_arm1020_dcache_clean_entry) mov r1, #0 mcr p15, 0, r1, c7, c10, 4 -#ifdef CONFIG_CPU_ARM1020_D_CACHE_ON +#ifndef CONFIG_CPU_DCACHE_DISABLE mcr p15, 0, r0, c7, c10, 1 @ clean single D entry mcr p15, 0, r1, c7, c10, 4 @ drain WB #endif -#ifdef CONFIG_CPU_ARM1020_I_CACHE_ON +#ifndef CONFIG_CPU_ICACHE_DISABLE mcr p15, 0, r1, c7, c5, 1 @ invalidate I entry #endif -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE mov r1, r1 mov r1, r1 #endif @@ -427,14 +427,14 @@ .align 5 ENTRY(cpu_arm1020_icache_invalidate_range) 1: mcr p15, 0, r0, c7, c10, 4 -#ifdef CONFIG_CPU_ARM1020_D_CACHE_ON +#ifndef CONFIG_CPU_DCACHE_DISABLE mcr p15, 0, r0, c7, c10, 1 @ Clean D entry mcr p15, 0, r0, c7, c10, 4 @ drain WB add r0, r0, #DCACHELINESIZE mcr p15, 0, r0, c7, c10, 1 @ Clean D entry mcr p15, 0, r0, c7, c10, 4 @ drain WB #endif -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE mov r0, r0 #endif add r0, r0, #DCACHELINESIZE @@ -442,7 +442,7 @@ blo 1b ENTRY(cpu_arm1020_icache_invalidate_page) mcr p15, 0, r0, c7, c5, 0 @ invalidate I cache -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE mov r0, r0 mov r0, r0 #endif @@ -460,7 +460,7 @@ mov r0, #0 mcr p15, 0, r0, c7, c10, 4 @ drain WB mcr p15, 0, r0, c8, c7, 0 @ invalidate I & D tlbs -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE mov r0, r0 mov r0, r0 #endif @@ -476,12 +476,15 @@ */ .align 5 ENTRY(cpu_arm1020_tlb_invalidate_range) + sub r3, r1, r0 + cmp r3, #256 * PAGESIZE + bhi cpu_arm1020_tlb_invalidate_all mov r3, #0 mcr p15, 0, r3, c7, c10, 4 @ drain WB 1: mcr p15, 0, r0, c8, c6, 1 @ invalidate D TLB entry mcr p15, 0, r0, c8, c5, 1 @ invalidate I TLB entry add r0, r0, #PAGESIZE -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE mov r0, r0 #endif cmp r0, r1 @@ -500,14 +503,14 @@ ENTRY(cpu_arm1020_tlb_invalidate_page) mov r3, #0 mcr p15, 0, r3, c7, c10, 4 @ drain WB -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE mov r0, r0 mov r0, r0 #endif teq r1, #0 mcr p15, 0, r0, c8, c6, 1 @ invalidate D TLB entry mcrne p15, 0, r0, c8, c5, 1 @ invalidate I TLB entry -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE mov r0, r0 mov r0, r0 #endif @@ -524,7 +527,7 @@ */ .align 5 ENTRY(cpu_arm1020_set_pgd) -#ifdef CONFIG_CPU_ARM1020_D_CACHE_ON +#ifndef CONFIG_CPU_DCACHE_DISABLE mcr p15, 0, r3, c7, c10, 4 mov r1, #0xF @ 16 segments 1: mov r3, #0x3F @ 64 entries @@ -533,7 +536,7 @@ mcr p15, 0, ip, c7, c14, 2 @ Clean & Inval DCache entry mov ip, #0 mcr p15, 0, ip, c7, c10, 4 -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE mov ip, ip #endif subs r3, r3, #1 @@ -545,13 +548,13 @@ #endif mov r1, #0 -#ifdef CONFIG_CPU_ARM1020_I_CACHE_ON +#ifndef CONFIG_CPU_ICACHE_DISABLE mcr p15, 0, r1, c7, c5, 0 @ invalidate I cache #endif mcr p15, 0, r1, c7, c10, 4 @ drain WB mcr p15, 0, r0, c2, c0, 0 @ load page table pointer mcr p15, 0, r1, c8, c7, 0 @ invalidate I & D TLBs -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE mov ip, ip mov ip, ip #endif @@ -568,18 +571,18 @@ */ .align 5 ENTRY(cpu_arm1020_set_pmd) -#ifdef CONFIG_CPU_ARM1020_FORCE_WRITE_THROUGH +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH eor r2, r1, #0x0a @ C & Section tst r2, #0x0b biceq r1, r1, #4 @ clear bufferable bit #endif str r1, [r0] -#ifdef CONFIG_CPU_ARM1020_D_CACHE_ON +#ifndef CONFIG_CPU_DCACHE_DISABLE mcr p15, 0, r0, c7, c10, 4 mcr p15, 0, r0, c7, c10, 1 @ clean D entry (drain is done by TLB fns) #endif mcr p15, 0, r0, c7, c10, 4 @ drain WB -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE mov r0, r0 mov r0, r0 #endif @@ -609,47 +612,42 @@ tst r1, #LPTE_PRESENT | LPTE_YOUNG @ Present and Young? movne r2, #0 -#ifdef CONFIG_CPU_ARM1020_FORCE_WRITE_THROUGH +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH eor r3, r1, #0x0a @ C & small page? tst r3, #0x0b biceq r2, r2, #4 #endif str r2, [r0] @ hardware version mov r0, r0 -#ifdef CONFIG_CPU_ARM1020_D_CACHE_ON +#ifndef CONFIG_CPU_DCACHE_DISABLE mcr p15, 0, r0, c7, c10, 4 mcr p15, 0, r0, c7, c10, 1 @ clean D entry #endif mcr p15, 0, r0, c7, c10, 4 @ drain WB -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE mov r0, r0 mov r0, r0 #endif mov pc, lr -cpu_manu_name: - .asciz "ARM/VLSI" ENTRY(cpu_arm1020_name) .ascii "Arm1020" -#if defined(CONFIG_CPU_ARM1020_CPU_IDLE) - .ascii "s" -#endif -#if defined(CONFIG_CPU_ARM1020_I_CACHE_ON) +#ifndef CONFIG_CPU_ICACHE_DISABLE .ascii "i" #endif -#if defined(CONFIG_CPU_ARM1020_D_CACHE_ON) +#ifndef CONFIG_CPU_DCACHE_DISABLE .ascii "d" -#if defined(CONFIG_CPU_ARM1020_FORCE_WRITE_THROUGH) +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH .ascii "(wt)" #else .ascii "(wb)" #endif #endif -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE .ascii "B" #endif -#ifdef CONFIG_CPU_ARM1020_ROUND_ROBIN +#ifdef CONFIG_CPU_CACHE_ROUND_ROBIN .ascii "RR" #endif .ascii "\0" @@ -679,16 +677,16 @@ orr r0, r0, #0x0031 @ ..........DP...M orr r0, r0, #0x0100 @ .......S........ -#ifdef CONFIG_CPU_ARM1020_ROUND_ROBIN +#ifdef CONFIG_CPU_CACHE_ROUND_ROBIN orr r0, r0, #0x4000 @ .R.............. #endif -#ifdef CONFIG_CPU_ARM1020_BRANCH_PREDICTION +#ifndef CONFIG_CPU_BPREDICT_DISABLE orr r0, r0, #0x0800 @ ....Z........... #endif -#ifdef CONFIG_CPU_ARM1020_D_CACHE_ON +#ifndef CONFIG_CPU_DCACHE_DISABLE orr r0, r0, #0x0004 @ Enable D cache #endif -#ifdef CONFIG_CPU_ARM1020_I_CACHE_ON +#ifndef CONFIG_CPU_ICACHE_DISABLE orr r0, r0, #0x1000 @ I Cache on #endif mov pc, lr @@ -736,7 +734,7 @@ .type cpu_arm1020_info, #object cpu_arm1020_info: - .long cpu_manu_name + .long 0 .long cpu_arm1020_name .size cpu_arm1020_info, . - cpu_arm1020_info @@ -761,7 +759,7 @@ b __arm1020_setup .long cpu_arch_name .long cpu_elf_name - .long HWCAP_SWP | HWCAP_HALF | HWCAP_26BIT + .long HWCAP_SWP | HWCAP_HALF | HWCAP_THUMB .long cpu_arm1020_info .long arm1020_processor_functions .size __arm1020_proc_info, . - __arm1020_proc_info diff -urN orig/arch/arm/mm/proc-arm1026.S linux/arch/arm/mm/proc-arm1026.S --- orig/arch/arm/mm/proc-arm1026.S Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mm/proc-arm1026.S Thu Jan 23 20:40:31 2003 @@ -0,0 +1,660 @@ +/* + * linux/arch/arm/mm/arm1026.S: MMU functions for ARM1026EJ-S + * + * Copyright (C) 2002 ARM Limited + * Copyright (C) 2002 Deep Blue Solutions Ltd. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + * + * + * These are the low level assembler for performing cache and TLB + * functions on the arm1026. + */ +#include +#include +#include +#include +#include +#include + +/* + * This is the maximum size of an area which will be invalidated + * using the single invalidate entry instructions. Anything larger + * than this, and we go for the whole cache. + * + * This value should be chosen such that we choose the cheapest + * alternative. + */ +#define MAX_AREA_SIZE 16384 + +/* + * the cache line size of the I and D cache + */ +#define DCACHELINESIZE 32 +#define ICACHELINESIZE 32 + +/* + * and the page size + */ +#define PAGESIZE 4096 + + .text + +/* + * cpu_arm1026_data_abort() + * + * obtain information about current aborted instruction + * Note: we read user space. This means we might cause a data + * abort here if the I-TLB and D-TLB aren't seeing the same + * picture. Unfortunately, this does happen. We live with it. + * + * Inputs: + * r2 = address of abort + * r3 = cpsr of abort + * + * Returns: + * r0 = address of abort + * r1 != 0 if writing + * r3 = FSR + * r4 = corrupted + */ + .align 5 +ENTRY(cpu_arm1026_data_abort) + mrc p15, 0, r3, c5, c0, 0 @ get FSR + and r2, r3, #0b1101 @ Check for translation error + sub r1, r2, #0b0101 + + and r3, r3, #255 + mrc p15, 0, r0, c6, c0, 0 @ get FAR + + mov pc, lr + +/* + * cpu_arm1026_check_bugs() + */ +ENTRY(cpu_arm1026_check_bugs) + mrs ip, cpsr + bic ip, ip, #F_BIT + msr cpsr, ip + mov pc, lr + +/* + * cpu_arm1026_proc_init() + */ +ENTRY(cpu_arm1026_proc_init) + mov pc, lr + +/* + * cpu_arm1026_proc_fin() + */ +ENTRY(cpu_arm1026_proc_fin) + stmfd sp!, {lr} + mov ip, #F_BIT | I_BIT | SVC_MODE + msr cpsr_c, ip + bl cpu_arm1026_cache_clean_invalidate_all + mrc p15, 0, r0, c1, c0, 0 @ ctrl register + bic r0, r0, #0x1000 @ ...i............ + bic r0, r0, #0x000e @ ............wca. + mcr p15, 0, r0, c1, c0, 0 @ disable caches + ldmfd sp!, {pc} + +/* + * cpu_arm1026_reset(loc) + * + * Perform a soft reset of the system. Put the CPU into the + * same state as it would be if it had been reset, and branch + * to what would be the reset vector. + * + * loc: location to jump to for soft reset + */ + .align 5 +ENTRY(cpu_arm1026_reset) + mov ip, #0 + mcr p15, 0, ip, c7, c7, 0 @ invalidate I,D caches + mcr p15, 0, ip, c7, c10, 4 @ drain WB + mcr p15, 0, ip, c8, c7, 0 @ invalidate I & D TLBs + mrc p15, 0, ip, c1, c0, 0 @ ctrl register + bic ip, ip, #0x000f @ ............wcam + bic ip, ip, #0x1100 @ ...i...s........ + mcr p15, 0, ip, c1, c0, 0 @ ctrl register + mov pc, r0 + +/* + * cpu_arm1026_do_idle() + */ + .align 5 +ENTRY(cpu_arm1026_do_idle) + mcr p15, 0, r0, c7, c0, 4 @ Wait for interrupt + mov pc, lr + +/* ================================= CACHE ================================ */ + + +/* + * cpu_arm1026_cache_clean_invalidate_all() + * + * clean and invalidate all cache lines + * + * Note: + * 1. we should preserve r0 at all times + */ + .align 5 +ENTRY(cpu_arm1026_cache_clean_invalidate_all) + mov r2, #1 +cpu_arm1026_cache_clean_invalidate_all_r2: + mov ip, #0 +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH + mcr p15, 0, ip, c7, c6, 0 @ invalidate D cache +#else +1: mrc p15, 0, r15, c7, c14, 3 @ test,clean,invalidate + bne 1b +#endif + teq r2, #0 + mcrne p15, 0, ip, c7, c5, 0 @ invalidate I cache + mcr p15, 0, ip, c7, c10, 4 @ drain WB + mov pc, lr + +/* + * cpu_arm1026_cache_clean_invalidate_range(start, end, flags) + * + * clean and invalidate all cache lines associated with this area of memory + * + * This is a little misleading, it is not intended to clean out + * the i-cache but to make sure that any data written to the + * range is made consistant. This means that when we execute code + * in that region, everything works as we expect. + * + * This generally means writing back data in the Dcache and + * write buffer and flushing the Icache over that region + * start: Area start address + * end: Area end address + * flags: nonzero for I cache as well + */ + .align 5 +ENTRY(cpu_arm1026_cache_clean_invalidate_range) + bic r0, r0, #DCACHELINESIZE - 1 @ && added by PGM + sub r3, r1, r0 + cmp r3, #MAX_AREA_SIZE + bhi cpu_arm1026_cache_clean_invalidate_all_r2 + +1: teq r2, #0 +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH + mcr p15, 0, r0, c7, c6, 1 @ invalidate D entry + mcrne p15, 0, r0, c7, c5, 1 @ invalidate I entry + add r0, r0, #DCACHELINESIZE + mcr p15, 0, r0, c7, c6, 1 @ invalidate D entry + mcrne p15, 0, r0, c7, c5, 1 @ invalidate I entry + add r0, r0, #DCACHELINESIZE +#else + mcr p15, 0, r0, c7, c14, 1 @ clean and invalidate D entry + mcrne p15, 0, r0, c7, c5, 1 @ invalidate I entry + add r0, r0, #DCACHELINESIZE + mcr p15, 0, r0, c7, c14, 1 @ clean and invalidate D entry + mcrne p15, 0, r0, c7, c5, 1 @ invalidate I entry + add r0, r0, #DCACHELINESIZE +#endif + + cmp r0, r1 + blo 1b + + mcr p15, 0, r1, c7, c10, 4 @ drain WB + + mov pc, lr + +/* + * cpu_arm1026_flush_ram_page(page) + * + * clean and invalidate all cache lines associated with this area of memory + * + * page: page to clean and invalidate + */ + .align 5 +ENTRY(cpu_arm1026_flush_ram_page) + mov r1, #PAGESIZE +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH +1: mcr p15, 0, r0, c7, c6, 1 @ invalidate D entry + add r0, r0, #DCACHELINESIZE + mcr p15, 0, r0, c7, c6, 1 @ invalidate D entry + add r0, r0, #DCACHELINESIZE +#else +1: mcr p15, 0, r0, c7, c14, 1 @ clean and invalidate D entry + add r0, r0, #DCACHELINESIZE + mcr p15, 0, r0, c7, c14, 1 @ clean and invalidate D entry + add r0, r0, #DCACHELINESIZE +#endif + subs r1, r1, #2 * DCACHELINESIZE + bne 1b + mcr p15, 0, r1, c7, c10, 4 @ drain WB + mov pc, lr + +/* ================================ D-CACHE =============================== */ + +/* + * cpu_arm1026_dcache_invalidate_range(start, end) + * + * throw away all D-cached data in specified region without an obligation + * to write them back. Note however that we must clean the D-cached entries + * around the boundaries if the start and/or end address are not cache + * aligned. + * + * start: virtual start address + * end: virtual end address + */ + .align 5 +ENTRY(cpu_arm1026_dcache_invalidate_range) +#ifndef CONFIG_CPU_ARM1026_WRITETHROUGH + tst r0, #DCACHELINESIZE - 1 + mcrne p15, 0, r0, c7, c10, 1 @ clean D entry + tst r1, #DCACHELINESIZE - 1 + mcrne p15, 0, r1, c7, c10, 1 @ clean D entry +#endif + bic r0, r0, #DCACHELINESIZE - 1 +1: mcr p15, 0, r0, c7, c6, 1 @ invalidate D entry + add r0, r0, #DCACHELINESIZE + cmp r0, r1 + blo 1b + mov pc, lr + +/* + * cpu_arm1026_dcache_clean_range(start, end) + * + * For the specified virtual address range, ensure that all caches contain + * clean data, such that peripheral accesses to the physical RAM fetch + * correct data. + * + * start: virtual start address + * end: virtual end address + */ + .align 5 +ENTRY(cpu_arm1026_dcache_clean_range) +#ifndef CONFIG_CPU_ARM1026_WRITETHROUGH + bic r0, r0, #DCACHELINESIZE - 1 + sub r3, r1, r0 + cmp r3, #MAX_AREA_SIZE + mov r2, #0 + bhi cpu_arm1026_cache_clean_invalidate_all_r2 + +1: mcr p15, 0, r0, c7, c10, 1 @ clean D entry + add r0, r0, #DCACHELINESIZE + cmp r0, r1 + blo 1b +#endif + mcr p15, 0, r2, c7, c10, 4 @ drain WB + mov pc, lr + +/* + * cpu_arm1026_dcache_clean_page(page) + * + * Cleans a single page of dcache so that if we have any future aliased + * mappings, they will be consistent at the time that they are created. + * + * page: virtual address of page to clean from dcache + * + * Note: + * 1. we don't need to flush the write buffer in this case. + * 2. we don't invalidate the entries since when we write the page + * out to disk, the entries may get reloaded into the cache. + */ + .align 5 +ENTRY(cpu_arm1026_dcache_clean_page) +#ifndef CONFIG_CPU_ARM1026_WRITETHROUGH + mov r1, #PAGESIZE +1: mcr p15, 0, r0, c7, c10, 1 @ clean D entry + add r0, r0, #DCACHELINESIZE + mcr p15, 0, r0, c7, c10, 1 @ clean D entry + add r0, r0, #DCACHELINESIZE + subs r1, r1, #2 * DCACHELINESIZE + bne 1b +#endif + mov pc, lr + +/* + * cpu_arm1026_dcache_clean_entry(addr) + * + * Clean the specified entry of any caches such that the MMU + * translation fetches will obtain correct data. + * + * addr: cache-unaligned virtual address + */ + .align 5 +ENTRY(cpu_arm1026_dcache_clean_entry) +#ifndef CONFIG_CPU_ARM1026_WRITETHROUGH + mcr p15, 0, r0, c7, c10, 1 @ clean D entry +#endif + mcr p15, 0, r0, c7, c10, 4 @ drain WB + mov pc, lr + +/* ================================ I-CACHE =============================== */ + +/* + * cpu_arm1026_icache_invalidate_range(start, end) + * + * This *is not* just icache. It is to make data written to memory + * consistent such that instructions fetched from the region are what + * we expect. + * + * This is typically used after we have copied a module into kernel space, + * and we're about to start executing code from that module. + * + * start: virtual start address + * end: virtual end address + */ + .align 5 +ENTRY(cpu_arm1026_icache_invalidate_range) + bic r0, r0, #DCACHELINESIZE - 1 @ Safety check + sub r3, r1, r0 + cmp r3, #MAX_AREA_SIZE + bhi cpu_arm1026_cache_clean_invalidate_all_r2 + +1: mcr p15, 0, r0, c7, c5, 1 @ clean I entries + add r0, r0, #DCACHELINESIZE + cmp r0, r1 + blo 1b + + mov r0, #0 + mcr p15, 0, r0, c7, c10, 4 @ drain WB + mov pc, lr + +ENTRY(cpu_arm1026_icache_invalidate_page) + mcr p15, 0, r0, c7, c5, 0 @ invalidate I cache + mov pc, lr + + +/* ================================== TLB ================================= */ + +/* + * cpu_arm1026_tlb_invalidate_all() + * + * Invalidate all TLB entries + */ + .align 5 +ENTRY(cpu_arm1026_tlb_invalidate_all) + mov r0, #0 + mcr p15, 0, r0, c7, c10, 4 @ drain WB + mcr p15, 0, r0, c8, c7, 0 @ invalidate I & D TLBs + mov pc, lr + +/* + * cpu_arm1026_tlb_invalidate_range(start, end) + * + * invalidate TLB entries covering the specified range + * + * start: range start address + * end: range end address + */ + .align 5 +ENTRY(cpu_arm1026_tlb_invalidate_range) + sub r3, r1, r0 + cmp r3, #256 * PAGESIZE @ arbitary, should be tuned + bhi cpu_arm1026_tlb_invalidate_all + + mov r3, #0 + mcr p15, 0, r3, c7, c10, 4 @ drain WB + + bic r0, r0, #(PAGESIZE - 1) & 0x00ff + bic r0, r0, #(PAGESIZE - 1) & 0xff00 + +1: mcr p15, 0, r0, c8, c6, 1 @ invalidate D TLB entry + mcr p15, 0, r0, c8, c5, 1 @ invalidate I TLB entry + add r0, r0, #PAGESIZE + cmp r0, r1 + blo 1b + mov pc, lr + +/* + * cpu_arm1026_tlb_invalidate_page(page, flags) + * + * invalidate the TLB entries for the specified page. + * + * page: page to invalidate + * flags: non-zero if we include the I TLB + */ + .align 5 +ENTRY(cpu_arm1026_tlb_invalidate_page) + mov r3, #0 + mcr p15, 0, r3, c7, c10, 4 @ drain WB + teq r1, #0 + mcr p15, 0, r0, c8, c6, 1 @ invalidate D TLB entry + mcrne p15, 0, r0, c8, c5, 1 @ invalidate I TLB entry + mov pc, lr + +/* =============================== PageTable ============================== */ + +/* + * cpu_arm1026_set_pgd(pgd) + * + * Set the translation base pointer to be as described by pgd. + * + * pgd: new page tables + */ + .align 5 +ENTRY(cpu_arm1026_set_pgd) + mov ip, #0 +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH + /* Any reason why we don't use mcr p15, 0, r0, c7, c7, 0 here? --rmk */ + mcr p15, 0, ip, c7, c6, 0 @ invalidate D cache +#else +@ && 'Clean & Invalidate whole DCache' +1: mrc p15, 0, r15, c7, c14, 3 @ test,clean,invalidate + bne 1b +#endif + mcr p15, 0, ip, c7, c5, 0 @ invalidate I cache + mcr p15, 0, ip, c7, c10, 4 @ drain WB + mcr p15, 0, r0, c2, c0, 0 @ load page table pointer + mcr p15, 0, ip, c8, c7, 0 @ invalidate I & D TLBs + mov pc, lr + +/* + * cpu_arm1026_set_pmd(pmdp, pmd) + * + * Set a level 1 translation table entry, and clean it out of + * any caches such that the MMUs can load it correctly. + * + * pmdp: pointer to PMD entry + * pmd: PMD value to store + */ + .align 5 +ENTRY(cpu_arm1026_set_pmd) +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH + eor r2, r1, #0x0a @ C & Section + tst r2, #0x0b + biceq r1, r1, #4 @ clear bufferable bit +#endif + str r1, [r0] +#ifndef CONFIG_CPU_ARM1026_WRITETHROUGH + mcr p15, 0, r0, c7, c10, 1 @ clean D entry +#endif + mcr p15, 0, r0, c7, c10, 4 @ drain WB + mov pc, lr + +/* + * cpu_arm1026_set_pte(ptep, pte) + * + * Set a PTE and flush it out + */ + .align 5 +ENTRY(cpu_arm1026_set_pte) + str r1, [r0], #-1024 @ linux version + + eor r1, r1, #LPTE_PRESENT | LPTE_YOUNG | LPTE_WRITE | LPTE_DIRTY + + bic r2, r1, #0xff0 + bic r2, r2, #3 + orr r2, r2, #HPTE_TYPE_SMALL + + tst r1, #LPTE_USER | LPTE_EXEC @ User or Exec? + orrne r2, r2, #HPTE_AP_READ + + tst r1, #LPTE_WRITE | LPTE_DIRTY @ Write and Dirty? + orreq r2, r2, #HPTE_AP_WRITE + + tst r1, #LPTE_PRESENT | LPTE_YOUNG @ Present and Young? + movne r2, #0 + +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH + eor r3, r2, #0x0a @ C & small page? + tst r3, #0x0b + biceq r2, r2, #4 +#endif + str r2, [r0] @ hardware version + mov r0, r0 +#ifndef CONFIG_CPU_DCACHE_WRITETHROUGH + mcr p15, 0, r0, c7, c10, 1 @ clean D entry +#endif + mcr p15, 0, r0, c7, c10, 4 @ drain WB + mov pc, lr + + +ENTRY(cpu_arm1026_name) + .ascii "ARM1026EJ-S" +#ifndef CONFIG_CPU_ICACHE_DISABLE + .ascii "i" +#endif +#ifndef CONFIG_CPU_DCACHE_DISABLE + .ascii "d" +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH + .ascii "(wt)" +#else + .ascii "(wb)" +#endif +#ifndef CONFIG_CPU_BPREDICT_DISABLE + .ascii "B" +#endif +#ifdef CONFIG_CPU_CACHE_ROUND_ROBIN + .ascii "RR" +#endif +#endif + .ascii "\0" + .align + + .section ".text.init", #alloc, #execinstr + +__arm1026_setup: + mov r0, #0 + mcr p15, 0, r0, c7, c7 @ invalidate I,D caches on v4 + mcr p15, 0, r0, c7, c10, 4 @ drain write buffer on v4 + mcr p15, 0, r0, c8, c7 @ invalidate I,D TLBs on v4 + mcr p15, 0, r4, c2, c0 @ load page table pointer + + +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH + mov r0, #4 @ disable write-back on caches explicitly + mcr p15, 7, r0, c15, c0, 0 +#endif + + mov r0, #0x1f @ Domains 0, 1 = client + mcr p15, 0, r0, c3, c0 @ load domain access register + mrc p15, 0, r0, c1, c0 @ get control register v4 +/* + * Clear out 'unwanted' bits (then put them in if we need them) + */ + @ VI ZFRS BLDP WCAM + bic r0, r0, #0x0e00 + bic r0, r0, #0x0002 + bic r0, r0, #0x000c + bic r0, r0, #0x1000 @ ...0 000. .... 000. +/* + * Turn on what we want + */ + orr r0, r0, #0x0031 + orr r0, r0, #0x2100 @ ..1. ...1 ..11 ...1 + +#ifdef CONFIG_CPU_CACHE_ROUND_ROBIN + orr r0, r0, #0x4000 @ .1.. .... .... .... +#endif +#ifndef CONFIG_CPU_BPREDICT_DISABLE + orr r0, r0, #0x0800 @ ....Z........... +#endif +#ifndef CONFIG_CPU_DCACHE_DISABLE + orr r0, r0, #0x0004 @ .... .... .... .1.. +#endif +#ifndef CONFIG_CPU_ICACHE_DISABLE + orr r0, r0, #0x1000 @ ...1 .... .... .... +#endif + mov pc, lr + + .text + +/* + * Purpose : Function pointers used to access above functions - all calls + * come through these + */ + .type arm1026_processor_functions, #object +arm1026_processor_functions: + .word cpu_arm1026_data_abort + .word cpu_arm1026_check_bugs + .word cpu_arm1026_proc_init + .word cpu_arm1026_proc_fin + .word cpu_arm1026_reset + .word cpu_arm1026_do_idle + + /* cache */ + .word cpu_arm1026_cache_clean_invalidate_all + .word cpu_arm1026_cache_clean_invalidate_range + .word cpu_arm1026_flush_ram_page + + /* dcache */ + .word cpu_arm1026_dcache_invalidate_range + .word cpu_arm1026_dcache_clean_range + .word cpu_arm1026_dcache_clean_page + .word cpu_arm1026_dcache_clean_entry + + /* icache */ + .word cpu_arm1026_icache_invalidate_range + .word cpu_arm1026_icache_invalidate_page + + /* tlb */ + .word cpu_arm1026_tlb_invalidate_all + .word cpu_arm1026_tlb_invalidate_range + .word cpu_arm1026_tlb_invalidate_page + + /* pgtable */ + .word cpu_arm1026_set_pgd + .word cpu_arm1026_set_pmd + .word cpu_arm1026_set_pte + .size arm1026_processor_functions, . - arm1026_processor_functions + + .type cpu_arm1026_info, #object +cpu_arm1026_info: + .long 0 + .long cpu_arm1026_name + .size cpu_arm1026_info, . - cpu_arm1026_info + + .type cpu_arch_name, #object +cpu_arch_name: + .asciz "armv5EJ" + .size cpu_arch_name, . - cpu_arch_name + + .type cpu_elf_name, #object +cpu_elf_name: + .asciz "v5EJ" + .size cpu_elf_name, . - cpu_elf_name + .align + + .section ".proc.info", #alloc, #execinstr + + .type __arm1026_proc_info,#object +__arm1026_proc_info: + .long 0x4100a260 + .long 0xff00fff0 + .long 0x00000c1e @ mmuflags + b __arm1026_setup + .long cpu_arch_name + .long cpu_elf_name + .long HWCAP_SWP | HWCAP_HALF | HWCAP_THUMB | \ + HWCAP_FAST_MULT + .long cpu_arm1026_info + .long arm1026_processor_functions + .size __arm1026_proc_info, . - __arm1026_proc_info diff -urN orig/arch/arm/mm/proc-arm2,3.S linux/arch/arm/mm/proc-arm2,3.S --- orig/arch/arm/mm/proc-arm2,3.S Sat Mar 31 23:47:15 2001 +++ linux/arch/arm/mm/proc-arm2,3.S Thu Jan 23 20:40:31 2003 @@ -275,7 +275,6 @@ _arm2_3_check_bugs: bics pc, lr, #0x04000000 @ Clear FIQ disable bit -armvlsi_name: .asciz "ARM/VLSI" _arm2_name: .asciz "ARM 2" _arm250_name: .asciz "ARM 250" _arm3_name: .asciz "ARM 3" @@ -295,7 +294,7 @@ .word _arm2_xchg_4 cpu_arm2_info: - .long armvlsi_name + .long 0 .long _arm2_name .globl SYMBOL_NAME(arm250_processor_functions) @@ -308,7 +307,7 @@ .word _arm3_xchg_4 cpu_arm250_info: - .long armvlsi_name + .long 0 .long _arm250_name .globl SYMBOL_NAME(arm3_processor_functions) @@ -321,7 +320,7 @@ .word _arm3_xchg_4 cpu_arm3_info: - .long armvlsi_name + .long 0 .long _arm3_name arm2_arch_name: .asciz "armv1" diff -urN orig/arch/arm/mm/proc-arm6,7.S linux/arch/arm/mm/proc-arm6,7.S --- orig/arch/arm/mm/proc-arm6,7.S Mon Sep 3 14:14:41 2001 +++ linux/arch/arm/mm/proc-arm6,7.S Thu Jan 23 20:40:31 2003 @@ -68,13 +68,27 @@ * Purpose : flush a TLB entry */ ENTRY(cpu_arm6_tlb_invalidate_range) -ENTRY(cpu_arm7_tlb_invalidate_range) -1: mcr p15, 0, r0, c6, c0, 0 @ flush TLB + @ arm 6 tbl purge reg is bits 12 - 31 +1: mcr p15, 0, r0, c6, c0, 0 @ purge TLB add r0, r0, #4096 cmp r0, r1 blt 1b mov pc, lr +ENTRY(cpu_arm7_tlb_invalidate_range) +#if defined(CONFIG_ARCH_RISCSTATION) + mov r0, #0 + mcr p15, 0, r0, c5, c0, 0 @ flush TLB + mov pc, lr +#else + @ arm 7 tbl purge reg is bits 14 - 31 +1: mcr p15, 0, r0, c6, c0, 0 @ purge TLB + add r0, r0, #0x4000 + cmp r0, r1 + blt 1b + mov pc, lr +#endif + /* * Function: arm6_7_tlb_invalidate_page (unsigned long address, int flags) * @@ -143,7 +157,7 @@ movs pc, lr b Ldata_unknown - b Ldata_unknown + b Ldata_simple @ swp b Ldata_unknown b Ldata_unknown b Ldata_lateldrpostconst @ ldr rd, [rn], #m @@ -160,7 +174,7 @@ Ldata_unknown: @ Part of jumptable mov r0, r2 mov r1, r4 - mov r2, r3 + mov r2, r5 bl baddataabort b ret_from_exception @@ -258,7 +272,6 @@ ENTRY(cpu_arm6_do_idle) ENTRY(cpu_arm7_do_idle) - mov r0, #-EINVAL mov pc, lr /* @@ -349,8 +362,6 @@ mcr p15, 0, r1, c1, c0, 0 @ turn off MMU etc mov pc, r0 -cpu_armvlsi_name: - .asciz "ARM/VLSI" cpu_arm6_name: .asciz "ARM 6" cpu_arm610_name: .asciz "ARM 610" @@ -461,25 +472,25 @@ .type cpu_arm6_info, #object cpu_arm6_info: - .long cpu_armvlsi_name + .long 0 .long cpu_arm6_name .size cpu_arm6_info, . - cpu_arm6_info .type cpu_arm610_info, #object cpu_arm610_info: - .long cpu_armvlsi_name + .long 0 .long cpu_arm610_name .size cpu_arm610_info, . - cpu_Arm610_info .type cpu_arm7_info, #object cpu_arm7_info: - .long cpu_armvlsi_name + .long 0 .long cpu_arm7_name .size cpu_arm7_info, . - cpu_arm7_info .type cpu_arm710_info, #object cpu_arm710_info: - .long cpu_armvlsi_name + .long 0 .long cpu_arm710_name .size cpu_arm710_info, . - cpu_arm710_info diff -urN orig/arch/arm/mm/proc-arm720.S linux/arch/arm/mm/proc-arm720.S --- orig/arch/arm/mm/proc-arm720.S Mon Aug 5 13:29:47 2002 +++ linux/arch/arm/mm/proc-arm720.S Thu Jan 23 20:40:32 2003 @@ -37,6 +37,11 @@ #include /* + * and the page size + */ +#define PAGESIZE 4096 + +/* * Function: arm720_cache_clean_invalidate_all (void) * : arm720_cache_clean_invalidate_page (unsigned long address, int size, * int flags) @@ -86,8 +91,11 @@ * Purpose : flush a TLB entry */ ENTRY(cpu_arm720_tlb_invalidate_range) + sub r3, r1, r0 + cmp r3, #256 * PAGESIZE @ arbitary, should be tuned + bhi cpu_arm720_tlb_invalidate_all 1: mcr p15, 0, r0, c8, c7, 1 @ flush TLB (v4) - add r0, r0, #4096 + add r0, r0, #PAGESIZE cmp r0, r1 blt 1b mov pc, lr @@ -438,8 +446,6 @@ mov pc, r0 -cpu_armvlsi_name: - .asciz "ARM" cpu_arm720_name: .asciz "ARM720T" .align @@ -500,7 +506,7 @@ .type cpu_arm720_info, #object cpu_arm720_info: - .long cpu_armvlsi_name + .long 0 .long cpu_arm720_name .size cpu_arm720_info, . - cpu_arm720_info @@ -527,7 +533,7 @@ b __arm720_setup @ cpu_flush .long cpu_arch_name @ arch_name .long cpu_elf_name @ elf_name - .long HWCAP_SWP | HWCAP_HALF | HWCAP_THUMB | HWCAP_26BIT @ elf_hwcap + .long HWCAP_SWP | HWCAP_HALF | HWCAP_THUMB @ elf_hwcap .long cpu_arm720_info @ info .long arm720_processor_functions .size __arm720_proc_info, . - __arm720_proc_info diff -urN orig/arch/arm/mm/proc-arm920.S linux/arch/arm/mm/proc-arm920.S --- orig/arch/arm/mm/proc-arm920.S Mon Aug 5 13:29:47 2002 +++ linux/arch/arm/mm/proc-arm920.S Thu Jan 23 20:40:32 2003 @@ -134,9 +134,7 @@ */ .align 5 ENTRY(cpu_arm920_do_idle) -#if defined(CONFIG_CPU_ARM920_CPU_IDLE) mcr p15, 0, r0, c7, c0, 4 @ Wait for interrupt -#endif mov pc, lr /* ================================= CACHE ================================ */ @@ -155,7 +153,7 @@ mov r2, #1 cpu_arm920_cache_clean_invalidate_all_r2: mov ip, #0 -#ifdef CONFIG_CPU_ARM920_WRITETHROUGH +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH mcr p15, 0, ip, c7, c6, 0 @ invalidate D cache #else /* @@ -193,7 +191,7 @@ cmp r3, #MAX_AREA_SIZE bgt cpu_arm920_cache_clean_invalidate_all_r2 1: teq r2, #0 -#ifdef CONFIG_CPU_ARM920_WRITETHROUGH +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH mcr p15, 0, r0, c7, c6, 1 @ invalidate D entry mcrne p15, 0, r0, c7, c5, 1 @ invalidate I entry add r0, r0, #DCACHELINESIZE @@ -224,7 +222,7 @@ .align 5 ENTRY(cpu_arm920_flush_ram_page) mov r1, #PAGESIZE -#ifdef CONFIG_CPU_ARM920_WRITETHROUGH +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH 1: mcr p15, 0, r0, c7, c6, 1 @ invalidate D entry add r0, r0, #DCACHELINESIZE mcr p15, 0, r0, c7, c6, 1 @ invalidate D entry @@ -410,6 +408,10 @@ */ .align 5 ENTRY(cpu_arm920_tlb_invalidate_range) + sub r3, r1, r0 + cmp r3, #256 * PAGESIZE @ arbitary, should be tuned + bhi cpu_arm920_tlb_invalidate_all + mov r3, #0 mcr p15, 0, r3, c7, c10, 4 @ drain WB @@ -454,7 +456,7 @@ .align 5 ENTRY(cpu_arm920_set_pgd) mov ip, #0 -#ifdef CONFIG_CPU_ARM920_WRITETHROUGH +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH /* Any reason why we don't use mcr p15, 0, r0, c7, c7, 0 here? --rmk */ mcr p15, 0, ip, c7, c6, 0 @ invalidate D cache #else @@ -487,7 +489,7 @@ */ .align 5 ENTRY(cpu_arm920_set_pmd) -#ifdef CONFIG_CPU_ARM920_WRITETHROUGH +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH eor r2, r1, #0x0a @ C & Section tst r2, #0x0b biceq r1, r1, #4 @ clear bufferable bit @@ -521,7 +523,7 @@ tst r1, #LPTE_PRESENT | LPTE_YOUNG @ Present and Young? movne r2, #0 -#ifdef CONFIG_CPU_ARM920_WRITETHROUGH +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH eor r3, r2, #0x0a @ C & small page? tst r3, #0x0b biceq r2, r2, #4 @@ -533,19 +535,14 @@ mov pc, lr -cpu_manu_name: - .asciz "ARM/CIRRUS" ENTRY(cpu_arm920_name) .ascii "Arm920T" -#if defined(CONFIG_CPU_ARM920_CPU_IDLE) - .ascii "s" -#endif -#if defined(CONFIG_CPU_ARM920_I_CACHE_ON) +#ifndef CONFIG_CPU_ICACHE_DISABLE .ascii "i" #endif -#if defined(CONFIG_CPU_ARM920_D_CACHE_ON) +#ifndef CONFIG_CPU_DCACHE_DISABLE .ascii "d" -#if defined(CONFIG_CPU_ARM920_WRITETHROUGH) +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH .ascii "(wt)" #else .ascii "(wb)" @@ -579,10 +576,10 @@ orr r0, r0, #0x0031 orr r0, r0, #0x2100 @ ..1. ...1 ..11 ...1 -#ifdef CONFIG_CPU_ARM920_D_CACHE_ON +#ifndef CONFIG_CPU_DCACHE_DISABLE orr r0, r0, #0x0004 @ .... .... .... .1.. #endif -#ifdef CONFIG_CPU_ARM920_I_CACHE_ON +#ifndef CONFIG_CPU_ICACHE_DISABLE orr r0, r0, #0x1000 @ ...1 .... .... .... #endif mov pc, lr @@ -630,7 +627,7 @@ .type cpu_arm920_info, #object cpu_arm920_info: - .long cpu_manu_name + .long 0 .long cpu_arm920_name .size cpu_arm920_info, . - cpu_arm920_info @@ -655,7 +652,7 @@ b __arm920_setup .long cpu_arch_name .long cpu_elf_name - .long HWCAP_SWP | HWCAP_HALF | HWCAP_26BIT + .long HWCAP_SWP | HWCAP_HALF | HWCAP_THUMB .long cpu_arm920_info .long arm920_processor_functions .size __arm920_proc_info, . - __arm920_proc_info diff -urN orig/arch/arm/mm/proc-arm922.S linux/arch/arm/mm/proc-arm922.S --- orig/arch/arm/mm/proc-arm922.S Mon Aug 5 13:29:47 2002 +++ linux/arch/arm/mm/proc-arm922.S Thu Jan 23 20:40:32 2003 @@ -130,9 +130,7 @@ */ .align 5 ENTRY(cpu_arm922_do_idle) -#if defined(CONFIG_CPU_ARM922_CPU_IDLE) mcr p15, 0, r0, c7, c0, 4 @ Wait for interrupt -#endif mov pc, lr /* ================================= CACHE ================================ */ @@ -151,7 +149,7 @@ mov r2, #1 cpu_arm922_cache_clean_invalidate_all_r2: mov ip, #0 -#ifdef CONFIG_CPU_ARM922_WRITETHROUGH +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH mcr p15, 0, ip, c7, c6, 0 @ invalidate D cache #else /* @@ -189,7 +187,7 @@ cmp r3, #MAX_AREA_SIZE bgt cpu_arm922_cache_clean_invalidate_all_r2 1: teq r2, #0 -#ifdef CONFIG_CPU_ARM922_WRITETHROUGH +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH mcr p15, 0, r0, c7, c6, 1 @ invalidate D entry mcrne p15, 0, r0, c7, c5, 1 @ invalidate I entry add r0, r0, #DCACHELINESIZE @@ -220,7 +218,7 @@ .align 5 ENTRY(cpu_arm922_flush_ram_page) mov r1, #PAGESIZE -#ifdef CONFIG_CPU_ARM922_WRITETHROUGH +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH 1: mcr p15, 0, r0, c7, c6, 1 @ invalidate D entry add r0, r0, #DCACHELINESIZE mcr p15, 0, r0, c7, c6, 1 @ invalidate D entry @@ -406,6 +404,10 @@ */ .align 5 ENTRY(cpu_arm922_tlb_invalidate_range) + sub r3, r1, r0 + cmp r3, #256 * PAGESIZE @ arbitary, should be tuned + bhi cpu_arm922_tlb_invalidate_all + mov r3, #0 mcr p15, 0, r3, c7, c10, 4 @ drain WB @@ -450,7 +452,7 @@ .align 5 ENTRY(cpu_arm922_set_pgd) mov ip, #0 -#ifdef CONFIG_CPU_ARM922_WRITETHROUGH +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH /* Any reason why we don't use mcr p15, 0, r0, c7, c7, 0 here? --rmk */ mcr p15, 0, ip, c7, c6, 0 @ invalidate D cache #else @@ -483,7 +485,7 @@ */ .align 5 ENTRY(cpu_arm922_set_pmd) -#ifdef CONFIG_CPU_ARM922_WRITETHROUGH +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH eor r2, r1, #0x0a @ C & Section tst r2, #0x0b biceq r1, r1, #4 @ clear bufferable bit @@ -517,7 +519,7 @@ tst r1, #LPTE_PRESENT | LPTE_YOUNG @ Present and Young? movne r2, #0 -#ifdef CONFIG_CPU_ARM922_WRITETHROUGH +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH eor r3, r2, #0x0a @ C & small page? tst r3, #0x0b biceq r2, r2, #4 @@ -529,19 +531,14 @@ mov pc, lr -cpu_manu_name: - .asciz "ARM/ALTERA" ENTRY(cpu_arm922_name) .ascii "Arm922T" -#if defined(CONFIG_CPU_ARM922_CPU_IDLE) - .ascii "s" -#endif -#if defined(CONFIG_CPU_ARM922_I_CACHE_ON) +#ifndef CONFIG_CPU_ICACHE_DISABLE .ascii "i" #endif -#if defined(CONFIG_CPU_ARM922_D_CACHE_ON) +#ifndef CONFIG_CPU_DCACHE_DISABLE .ascii "d" -#if defined(CONFIG_CPU_ARM922_WRITETHROUGH) +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH .ascii "(wt)" #else .ascii "(wb)" @@ -575,10 +572,10 @@ orr r0, r0, #0x0031 orr r0, r0, #0x2100 @ ..1. ...1 ..11 ...1 -#ifdef CONFIG_CPU_ARM922_D_CACHE_ON +#ifndef CONFIG_CPU_DCACHE_DISABLE orr r0, r0, #0x0004 @ .... .... .... .1.. #endif -#ifdef CONFIG_CPU_ARM922_I_CACHE_ON +#ifndef CONFIG_CPU_ICACHE_DISABLE orr r0, r0, #0x1000 @ ...1 .... .... .... #endif mov pc, lr @@ -626,7 +623,7 @@ .type cpu_arm922_info, #object cpu_arm922_info: - .long cpu_manu_name + .long 0 .long cpu_arm922_name .size cpu_arm922_info, . - cpu_arm922_info @@ -651,7 +648,7 @@ b __arm922_setup .long cpu_arch_name .long cpu_elf_name - .long HWCAP_SWP | HWCAP_HALF | HWCAP_26BIT + .long HWCAP_SWP | HWCAP_HALF | HWCAP_THUMB .long cpu_arm922_info .long arm922_processor_functions .size __arm922_proc_info, . - __arm922_proc_info diff -urN orig/arch/arm/mm/proc-arm925.S linux/arch/arm/mm/proc-arm925.S --- orig/arch/arm/mm/proc-arm925.S Thu Jan 1 01:00:00 1970 +++ linux/arch/arm/mm/proc-arm925.S Thu Jan 23 20:40:32 2003 @@ -0,0 +1,732 @@ +/* + * linux/arch/arm/mm/arm925.S: MMU functions for ARM925 + * + * Copyright (C) 1999-2001 ARM Limited + * Copyright (C) 2000 Deep Blue Solutions Ltd. + * Copyright (C) 2002 RidgeRun, Inc. + * Copyright (C) 2002 MontaVista Software, Inc. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + * + * + * These are the low level assembler for performing cache and TLB + * functions on the arm925. + */ +#include +#include +#include +#include +#include +#include + + +/* + * This is the maximum size of an area which will be invalidated + * using the single invalidate entry instructions. Anything larger + * than this, and we go for the whole cache. + * + * This value should be chosen such that we choose the cheapest + * alternative. + */ +#define MAX_AREA_SIZE (2 * 1024) + +/* + * the cache line size of the I and D cache + */ +#define DCACHELINESIZE 16 +#define ICACHELINESIZE 16 + +/* + * and the page size + */ +#define PAGESIZE 4096 + + .text + +/* + * cpu_arm925_data_abort() + * + * obtain information about current aborted instruction + * Note: we read user space. This means we might cause a data + * abort here if the I-TLB and D-TLB aren't seeing the same + * picture. Unfortunately, this does happen. We live with it. + * + * Inputs: + * r2 = address of abort + * r3 = cpsr of abort + * + * Returns: + * r0 = address of abort + * r1 != 0 if writing + * r3 = FSR + * r4 = corrupted + */ + .align 5 +ENTRY(cpu_arm925_data_abort) + mrc p15, 0, r0, c6, c0, 0 @ get FAR + mrc p15, 0, r4, c5, c0, 0 @ get FSR + + tst r3, #1<<5 @ Check for Thumb-bit (NE -> found) + ldrneh r1, [r2] @ Read aborted Thumb instruction + tstne r1, r1, lsr #12 @ C = bit 11 + + ldreq r1, [r2] @ Read aborted ARM instruction + tsteq r1, r1, lsr #21 @ C = bit 20 + + sbc r1, r1, r1 @ r1 = C - 1 + and r3, r4, #255 + mov pc, lr + +/* + * cpu_arm925_check_bugs() + */ +ENTRY(cpu_arm925_check_bugs) + mrs ip, cpsr + bic ip, ip, #F_BIT + msr cpsr, ip + mov pc, lr + +/* + * cpu_arm925_proc_init() + */ +ENTRY(cpu_arm925_proc_init) + mov pc, lr + +/* + * cpu_arm925_proc_fin() + */ +ENTRY(cpu_arm925_proc_fin) + stmfd sp!, {lr} + mov ip, #F_BIT | I_BIT | SVC_MODE + msr cpsr_c, ip + bl cpu_arm925_cache_clean_invalidate_all + mrc p15, 0, r0, c1, c0, 0 @ ctrl register + bic r0, r0, #0x1000 @ ...i............ + bic r0, r0, #0x000e @ ............wca. + mcr p15, 0, r0, c1, c0, 0 @ disable caches + ldmfd sp!, {pc} + +/* + * cpu_arm925_reset(loc) + * + * Perform a soft reset of the system. Put the CPU into the + * same state as it would be if it had been reset, and branch + * to what would be the reset vector. + * + * loc: location to jump to for soft reset + */ + .align 5 +ENTRY(cpu_arm925_reset) + mov ip, #0 + mcr p15, 0, ip, c7, c7, 0 @ invalidate I,D caches + mcr p15, 0, ip, c7, c10, 4 @ drain WB + mcr p15, 0, ip, c8, c7, 0 @ invalidate I & D TLBs + mrc p15, 0, ip, c1, c0, 0 @ ctrl register + bic ip, ip, #0x000f @ ............wcam + bic ip, ip, #0x1100 @ ...i...s........ + mcr p15, 0, ip, c1, c0, 0 @ ctrl register + mov pc, r0 + +/* + * cpu_arm925_do_idle() + */ + .align 5 +ENTRY(cpu_arm925_do_idle) +#if defined(CONFIG_CPU_ARM925_CPU_IDLE) + mcr p15, 0, r0, c7, c0, 4 @ Wait for interrupt +#endif + mov pc, lr + +/* ================================= CACHE ================================ */ + + +/* + * cpu_arm925_cache_clean_invalidate_all() + * + * clean and invalidate all cache lines + * + * Note: + * 1. we should preserve r0 at all times + */ + .align 5 +ENTRY(cpu_arm925_cache_clean_invalidate_all) + mov r2, #1 +cpu_arm925_cache_clean_invalidate_all_r2: + mov ip, #0 +#ifdef CONFIG_CPU_ARM925_WRITETHROUGH + mcr p15, 0, ip, c7, c6, 0 @ invalidate D cache +#else +/* + * 'Clean & Invalidate whole DCache' + * Re-written to use Index Ops. + * NOTE: Requires TI925T Configuration Register C bit <- 0 + * for clean and invalidate of both D-Cache sets. + */ + mov r3, #255 << 4 @ 256 entries/set + @ ((NSETS - 1) << (CIR[13-12] + 3)) +2: mcr p15, 0, r3, c7, c14, 2 @ clean & invalidate D index + subs r3, r3, #1 << 4 + bcs 2b @ entries 255 to 0 +#endif + teq r2, #0 + mcrne p15, 0, ip, c7, c5, 0 @ invalidate I cache + mcr p15, 0, ip, c7, c10, 4 @ drain WB + mov pc, lr + +/* + * cpu_arm925_cache_clean_invalidate_range(start, end, flags) + * + * clean and invalidate all cache lines associated with this area of memory + * + * This is a little misleading, it is not intended to clean out + * the i-cache but to make sure that any data written to the + * range is made consistant. This means that when we execute code + * in that region, everything works as we expect. + * + * This generally means writing back data in the Dcache and + * write buffer and flushing the Icache over that region + * start: Area start address + * end: Area end address + * flags: nonzero for I cache as well + */ + .align 5 +ENTRY(cpu_arm925_cache_clean_invalidate_range) + bic r0, r0, #DCACHELINESIZE - 1 @ && added by PGM + bic r1, r1, #DCACHELINESIZE - 1 @ && added by DHM + sub r3, r1, r0 + cmp r3, #MAX_AREA_SIZE + bgt cpu_arm925_cache_clean_invalidate_all_r2 +1: teq r2, #0 +#ifdef CONFIG_CPU_ARM925_WRITETHROUGH + mcr p15, 0, r0, c7, c6, 1 @ invalidate D entry + mcrne p15, 0, r0, c7, c5, 1 @ invalidate I entry + add r0, r0, #DCACHELINESIZE + mcr p15, 0, r0, c7, c6, 1 @ invalidate D entry + mcrne p15, 0, r0, c7, c5, 1 @ invalidate I entry + add r0, r0, #DCACHELINESIZE +#else + mcr p15, 0, r0, c7, c14, 1 @ clean and invalidate D entry + mcrne p15, 0, r0, c7, c5, 1 @ invalidate I entry + add r0, r0, #DCACHELINESIZE + mcr p15, 0, r0, c7, c14, 1 @ clean and invalidate D entry + mcrne p15, 0, r0, c7, c5, 1 @ invalidate I entry + add r0, r0, #DCACHELINESIZE +#endif + cmp r0, r1 + blt 1b + + mcr p15, 0, r1, c7, c10, 4 @ drain WB + mov pc, lr + +/* + * cpu_arm925_flush_ram_page(page) + * + * clean and invalidate all cache lines associated with this area of memory + * + * page: page to clean and invalidate + */ + .align 5 +ENTRY(cpu_arm925_flush_ram_page) + mov r1, #PAGESIZE +#ifdef CONFIG_CPU_ARM925_WRITETHROUGH +1: mcr p15, 0, r0, c7, c6, 1 @ invalidate D entry + add r0, r0, #DCACHELINESIZE + mcr p15, 0, r0, c7, c6, 1 @ invalidate D entry + add r0, r0, #DCACHELINESIZE +#else +1: mcr p15, 0, r0, c7, c14, 1 @ clean and invalidate D entry + add r0, r0, #DCACHELINESIZE + mcr p15, 0, r0, c7, c14, 1 @ clean and invalidate D entry + add r0, r0, #DCACHELINESIZE +#endif + subs r1, r1, #2 * DCACHELINESIZE + bne 1b + mcr p15, 0, r1, c7, c10, 4 @ drain WB + mov pc, lr + +/* ================================ D-CACHE =============================== */ + +/* + * cpu_arm925_dcache_invalidate_range(start, end) + * + * throw away all D-cached data in specified region without an obligation + * to write them back. Note however that we must clean the D-cached entries + * around the boundaries if the start and/or end address are not cache + * aligned. + * + * start: virtual start address + * end: virtual end address + */ + .align 5 +ENTRY(cpu_arm925_dcache_invalidate_range) +#ifndef CONFIG_CPU_ARM925_WRITETHROUGH + tst r0, #DCACHELINESIZE - 1 + mcrne p15, 0, r0, c7, c10, 1 @ clean D entry + tst r1, #DCACHELINESIZE - 1 + mcrne p15, 0, r1, c7, c10, 1 @ clean D entry +#endif + bic r0, r0, #DCACHELINESIZE - 1 + bic r1, r1, #DCACHELINESIZE - 1 +1: mcr p15, 0, r0, c7, c6, 1 @ invalidate D entry + add r0, r0, #DCACHELINESIZE + cmp r0, r1 + blt 1b + mov pc, lr + +/* + * cpu_arm925_dcache_clean_range(start, end) + * + * For the specified virtual address range, ensure that all caches contain + * clean data, such that peripheral accesses to the physical RAM fetch + * correct data. + * + * start: virtual start address + * end: virtual end address + */ + .align 5 +ENTRY(cpu_arm925_dcache_clean_range) +#ifndef CONFIG_CPU_ARM925_WRITETHROUGH + bic r0, r0, #DCACHELINESIZE - 1 + sub r1, r1, r0 + cmp r1, #MAX_AREA_SIZE + mov r2, #0 + bgt cpu_arm925_cache_clean_invalidate_all_r2 + + bic r1, r1, #DCACHELINESIZE -1 + add r1, r1, #DCACHELINESIZE + +1: mcr p15, 0, r0, c7, c10, 1 @ clean D entry + add r0, r0, #DCACHELINESIZE + subs r1, r1, #DCACHELINESIZE + bpl 1b +#endif + mcr p15, 0, r2, c7, c10, 4 @ drain WB + mov pc, lr + +/* + * cpu_arm925_dcache_clean_page(page) + * + * Cleans a single page of dcache so that if we have any future aliased + * mappings, they will be consistent at the time that they are created. + * + * page: virtual address of page to clean from dcache + * + * Note: + * 1. we don't need to flush the write buffer in this case. + * 2. we don't invalidate the entries since when we write the page + * out to disk, the entries may get reloaded into the cache. + */ + .align 5 +ENTRY(cpu_arm925_dcache_clean_page) +#ifndef CONFIG_CPU_ARM925_WRITETHROUGH + mov r1, #PAGESIZE +1: mcr p15, 0, r0, c7, c10, 1 @ clean D entry + add r0, r0, #DCACHELINESIZE + mcr p15, 0, r0, c7, c10, 1 @ clean D entry + add r0, r0, #DCACHELINESIZE + subs r1, r1, #2 * DCACHELINESIZE + bne 1b +#endif + mov pc, lr + +/* + * cpu_arm925_dcache_clean_entry(addr) + * + * Clean the specified entry of any caches such that the MMU + * translation fetches will obtain correct data. + * + * addr: cache-unaligned virtual address + */ + .align 5 +ENTRY(cpu_arm925_dcache_clean_entry) +#ifndef CONFIG_CPU_ARM925_WRITETHROUGH + mcr p15, 0, r0, c7, c10, 1 @ clean D entry +#endif + mcr p15, 0, r0, c7, c10, 4 @ drain WB + mov pc, lr + +/* ================================ I-CACHE =============================== */ + +/* + * cpu_arm925_icache_invalidate_range(start, end) + * + * invalidate a range of virtual addresses from the Icache + * + * This is a little misleading, it is not intended to clean out + * the i-cache but to make sure that any data written to the + * range is made consistant. This means that when we execute code + * in that region, everything works as we expect. + * + * This generally means writing back data in the Dcache and + * write buffer and flushing the Icache over that region + * + * start: virtual start address + * end: virtual end address + * + * NOTE: ICACHELINESIZE == DCACHELINESIZE (so we don't need to + * loop twice, once for i-cache, once for d-cache) + */ + .align 5 +ENTRY(cpu_arm925_icache_invalidate_range) + bic r0, r0, #ICACHELINESIZE - 1 @ Safety check + sub r1, r1, r0 + cmp r1, #MAX_AREA_SIZE + bgt cpu_arm925_cache_clean_invalidate_all_r2 + + bic r1, r1, #ICACHELINESIZE - 1 + add r1, r1, #ICACHELINESIZE + +1: mcr p15, 0, r0, c7, c5, 1 @ Clean I entry + mcr p15, 0, r0, c7, c10, 1 @ Clean D entry + add r0, r0, #ICACHELINESIZE + subs r1, r1, #ICACHELINESIZE + bne 1b + + mov r0, #0 + mcr p15, 0, r0, c7, c10, 4 @ drain WB + mov pc, lr + + +ENTRY(cpu_arm925_icache_invalidate_page) + mcr p15, 0, r0, c7, c5, 0 @ invalidate I cache + nop + nop + nop + nop + mov pc, lr + +/* ================================== TLB ================================= */ + +/* + * cpu_arm925_tlb_invalidate_all() + * + * Invalidate all TLB entries + */ + .align 5 +ENTRY(cpu_arm925_tlb_invalidate_all) + mov r0, #0 + mcr p15, 0, r0, c7, c10, 4 @ drain WB + mcr p15, 0, r0, c8, c7, 0 @ invalidate I & D TLBs + mov pc, lr + +/* + * cpu_arm925_tlb_invalidate_range(start, end) + * + * invalidate TLB entries covering the specified range + * + * start: range start address + * end: range end address + */ + .align 5 +ENTRY(cpu_arm925_tlb_invalidate_range) + mov r3, #0 + mcr p15, 0, r3, c7, c10, 4 @ drain WB + + mov r3, #PAGESIZE + sub r3, r3, #1 + bic r0, r0, r3 + bic r1, r1, r3 + +1: mcr p15, 0, r0, c8, c6, 1 @ invalidate D TLB entry + mcr p15, 0, r0, c8, c5, 1 @ invalidate I TLB entry + add r0, r0, #PAGESIZE + cmp r0, r1 + blt 1b + mov pc, lr + +/* + * cpu_arm925_tlb_invalidate_page(page, flags) + * + * invalidate the TLB entries for the specified page. + * + * page: page to invalidate + * flags: non-zero if we include the I TLB + */ + .align 5 +ENTRY(cpu_arm925_tlb_invalidate_page) + mov r3, #0 + mcr p15, 0, r3, c7, c10, 4 @ drain WB + teq r1, #0 + mcr p15, 0, r0, c8, c6, 1 @ invalidate D TLB entry + mcrne p15, 0, r0, c8, c5, 1 @ invalidate I TLB entry + mov pc, lr + +/* =============================== PageTable ============================== */ + +/* + * cpu_arm925_set_pgd(pgd) + * + * Set the translation base pointer to be as described by pgd. + * + * pgd: new page tables + */ + .align 5 +ENTRY(cpu_arm925_set_pgd) + mov ip, #0 +#ifdef CONFIG_CPU_ARM925_WRITETHROUGH + mcr p15, 0, ip, c7, c7, 0 @ invalidate I & D cache +#else +/* + * 'Clean & Invalidate whole DCache' + * Re-written to use Index Ops. + * NOTE: Requires TI925T Configuration Register C bit <- 0 + * for clean and invalidate of both D-Cache sets. + */ + mov r3, #255 << 4 @ 256 entries/set + @ ((NSETS - 1) << (CIR[13-12] + 3)) +2: mcr p15, 0, r3, c7, c14, 2 @ clean & invalidate D index + subs r3, r3, #1 << 4 + bcs 2b @ entries 255 to 0 +#endif + mcr p15, 0, ip, c7, c5, 0 @ invalidate I cache + mcr p15, 0, ip, c7, c10, 4 @ drain WB + mcr p15, 0, r0, c2, c0, 0 @ load page table pointer + mcr p15, 0, ip, c8, c7, 0 @ invalidate I & D TLBs + mov pc, lr + +/* + * cpu_arm925_set_pmd(pmdp, pmd) + * + * Set a level 1 translation table entry, and clean it out of + * any caches such that the MMUs can load it correctly. + * + * pmdp: pointer to PMD entry + * pmd: PMD value to store + */ + .align 5 +ENTRY(cpu_arm925_set_pmd) +#ifdef CONFIG_CPU_ARM925_WRITETHROUGH + eor r2, r1, #0x0a @ C & Section + tst r2, #0x0b + biceq r1, r1, #4 @ clear bufferable bit +#endif + str r1, [r0] + mcr p15, 0, r0, c7, c10, 1 @ clean D entry + mcr p15, 0, r0, c7, c10, 4 @ drain WB + mov pc, lr + +/* + * cpu_arm925_set_pte(ptep, pte) + * + * Set a PTE and flush it out + */ + .align 5 +ENTRY(cpu_arm925_set_pte) + str r1, [r0], #-1024 @ linux version + + eor r1, r1, #LPTE_PRESENT | LPTE_YOUNG | LPTE_WRITE | LPTE_DIRTY + + bic r2, r1, #0xff0 + bic r2, r2, #3 + orr r2, r2, #HPTE_TYPE_SMALL + + tst r1, #LPTE_USER | LPTE_EXEC @ User or Exec? + orrne r2, r2, #HPTE_AP_READ + + tst r1, #LPTE_WRITE | LPTE_DIRTY @ Write and Dirty? + orreq r2, r2, #HPTE_AP_WRITE + + tst r1, #LPTE_PRESENT | LPTE_YOUNG @ Present and Young? + movne r2, #0 + +#ifdef CONFIG_CPU_ARM925_WRITETHROUGH + eor r3, r2, #0x0a @ C & small page? + tst r3, #0x0b + biceq r2, r2, #4 +#endif + str r2, [r0] @ hardware version + mov r0, r0 + mcr p15, 0, r0, c7, c10, 1 @ clean D entry + mcr p15, 0, r0, c7, c10, 4 @ drain WB + mov pc, lr + + +ENTRY(cpu_arm925_name) + .ascii "Arm925T" +#if defined(CONFIG_CPU_ARM925_CPU_IDLE) + .ascii "s" +#endif +#if defined(CONFIG_CPU_ARM925_I_CACHE_ON) + .ascii "i" +#endif +#if defined(CONFIG_CPU_ARM925_D_CACHE_ON) + .ascii "d" +#if defined(CONFIG_CPU_ARM925_WRITETHROUGH) + .ascii "(wt)" +#else + .ascii "(wb)" +#endif +#endif + .ascii "\0" + +ENTRY(cpu_arm915_name) + .ascii "Arm915T" +#if defined(CONFIG_CPU_ARM925_CPU_IDLE) + .ascii "s" +#endif +#if defined(CONFIG_CPU_ARM925_I_CACHE_ON) + .ascii "i" +#endif +#if defined(CONFIG_CPU_ARM925_D_CACHE_ON) + .ascii "d" +#if defined(CONFIG_CPU_ARM925_WRITETHROUGH) + .ascii "(wt)" +#else + .ascii "(wb)" +#endif +#endif + .ascii "\0" + .align + + .section ".text.init", #alloc, #execinstr + +__arm925_setup: + mov r0, #0 +#if defined(CONFIG_CPU_ARM925_NON_STREAMING_ON) + orr r0,r0,#0x80 +#endif +#if defined(CONFIG_CPU_ARM925_TRANSPARENT_ON) + orr r0,r0,#0x2 +#endif + mcr p15, 0, r0, c15, c1, 0 @ write TI config register + mov r0, #0 + mcr p15, 0, r0, c7, c7 @ invalidate I,D caches on v4 + mcr p15, 0, r0, c7, c10, 4 @ drain write buffer on v4 + mcr p15, 0, r0, c8, c7 @ invalidate I,D TLBs on v4 + mcr p15, 0, r4, c2, c0 @ load page table pointer + mov r0, #0x1f @ Domains 0, 1 = client + mcr p15, 0, r0, c3, c0 @ load domain access register + mrc p15, 0, r0, c1, c0 @ get control register v4 +/* + * Clear out 'unwanted' bits (then put them in if we need them) + */ + @ VI ZFRS BLDP WCAM + bic r0, r0, #0x0e00 + bic r0, r0, #0x0002 + bic r0, r0, #0x000c + bic r0, r0, #0x1000 @ ...0 000. .... 000. +/* + * Turn on what we want + */ + orr r0, r0, #0x0031 + orr r0, r0, #0x2100 @ ..1. ...1 ..11 ...1 + +#ifdef CONFIG_CPU_ARM925_WRITEBUFFER_ON + orr r0, r0, #0x0008 @ .... .... .... 1... +#endif +#ifdef CONFIG_CPU_ARM925_D_CACHE_ON + orr r0, r0, #0x0004 @ .... .... .... .1.. +#endif +#ifdef CONFIG_CPU_ARM925_I_CACHE_ON + orr r0, r0, #0x1000 @ ...1 .... .... .... +#endif + mov pc, lr + + .text + +/* + * Purpose : Function pointers used to access above functions - all calls + * come through these + */ + .type arm925_processor_functions, #object +arm925_processor_functions: + .word cpu_arm925_data_abort + .word cpu_arm925_check_bugs + .word cpu_arm925_proc_init + .word cpu_arm925_proc_fin + .word cpu_arm925_reset + .word cpu_arm925_do_idle + + /* cache */ + .word cpu_arm925_cache_clean_invalidate_all + .word cpu_arm925_cache_clean_invalidate_range + .word cpu_arm925_flush_ram_page + + /* dcache */ + .word cpu_arm925_dcache_invalidate_range + .word cpu_arm925_dcache_clean_range + .word cpu_arm925_dcache_clean_page + .word cpu_arm925_dcache_clean_entry + + /* icache */ + .word cpu_arm925_icache_invalidate_range + .word cpu_arm925_icache_invalidate_page + + /* tlb */ + .word cpu_arm925_tlb_invalidate_all + .word cpu_arm925_tlb_invalidate_range + .word cpu_arm925_tlb_invalidate_page + + /* pgtable */ + .word cpu_arm925_set_pgd + .word cpu_arm925_set_pmd + .word cpu_arm925_set_pte + .size arm925_processor_functions, . - arm925_processor_functions + + .type cpu_arm925_info, #object +cpu_arm925_info: + .long 0 + .long cpu_arm925_name + .size cpu_arm925_info, . - cpu_arm925_info + + .type cpu_arm915_info, #object +cpu_arm915_info: + .long cpu_manu_name + .long cpu_arm915_name + .size cpu_arm915_info, . - cpu_arm915_info + + .type cpu_arch_name, #object +cpu_arch_name: + .asciz "armv4" + .size cpu_arch_name, . - cpu_arch_name + + .type cpu_elf_name, #object +cpu_elf_name: + .asciz "v4" + .size cpu_elf_name, . - cpu_elf_name + .align + +#if defined(CONFIG_CPU_ARM925_WRITETHROUGH) +# define MMU_FLAGS 0x00000c1a +#else +# define MMU_FLAGS 0x00000c1e +#endif + .section ".proc.info", #alloc, #execinstr + + .type __arm925_proc_info,#object +__arm925_proc_info: + .long 0x54029252 + .long 0xffffffff + .long MMU_FLAGS + b __arm925_setup + .long cpu_arch_name + .long cpu_elf_name + .long HWCAP_SWP | HWCAP_HALF | HWCAP_THUMB + .long cpu_arm925_info + .long arm925_processor_functions + .size __arm925_proc_info, . - __arm925_proc_info + + .type __arm915_proc_info,#object +__arm915_proc_info: + .long 0x54029152 + .long 0xffffffff + .long MMU_FLAGS + b __arm925_setup + .long cpu_arch_name + .long cpu_elf_name + .long HWCAP_SWP | HWCAP_HALF | HWCAP_THUMB + .long cpu_arm915_info + .long arm925_processor_functions + .size __arm925_proc_info, . - __arm925_proc_info diff -urN orig/arch/arm/mm/proc-arm926.S linux/arch/arm/mm/proc-arm926.S --- orig/arch/arm/mm/proc-arm926.S Mon Aug 5 13:29:47 2002 +++ linux/arch/arm/mm/proc-arm926.S Thu Jan 23 20:40:32 2003 @@ -81,9 +81,8 @@ tst r3, #1<<5 @ Check for Thumb-bit (NE -> found) ldrneh r1, [r2] @ Read aborted Thumb instruction - tstne r1, r1, lsr #12 @ C = bit 11 - ldreq r1, [r2] @ Read aborted ARM instruction + movne r1, r1, lsl #(20-12) @ shift thumb bit 10 to ARM bit 20 tsteq r1, r1, lsr #21 @ C = bit 20 sbc r1, r1, r1 @ r1 = C - 1 @@ -146,9 +145,7 @@ */ .align 5 ENTRY(cpu_arm926_do_idle) -#if defined(CONFIG_CPU_ARM926_CPU_IDLE) mcr p15, 0, r0, c7, c0, 4 @ Wait for interrupt -#endif mov pc, lr /* ================================= CACHE ================================ */ @@ -167,7 +164,7 @@ mov r2, #1 cpu_arm926_cache_clean_invalidate_all_r2: mov ip, #0 -#ifdef CONFIG_CPU_ARM926_WRITETHROUGH +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH mcr p15, 0, ip, c7, c6, 0 @ invalidate D cache #else 1: mrc p15, 0, r15, c7, c14, 3 @ test,clean,invalidate @@ -202,7 +199,7 @@ bhi cpu_arm926_cache_clean_invalidate_all_r2 1: teq r2, #0 -#ifdef CONFIG_CPU_ARM926_WRITETHROUGH +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH mcr p15, 0, r0, c7, c6, 1 @ invalidate D entry mcrne p15, 0, r0, c7, c5, 1 @ invalidate I entry add r0, r0, #DCACHELINESIZE @@ -235,7 +232,7 @@ .align 5 ENTRY(cpu_arm926_flush_ram_page) mov r1, #PAGESIZE -#ifdef CONFIG_CPU_ARM926_WRITETHROUGH +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH 1: mcr p15, 0, r0, c7, c6, 1 @ invalidate D entry add r0, r0, #DCACHELINESIZE mcr p15, 0, r0, c7, c6, 1 @ invalidate D entry @@ -408,6 +405,10 @@ */ .align 5 ENTRY(cpu_arm926_tlb_invalidate_range) + sub r3, r1, r0 + cmp r3, #256 * PAGESIZE @ arbitary, should be tuned + bhi cpu_arm926_tlb_invalidate_all + mov r3, #0 mcr p15, 0, r3, c7, c10, 4 @ drain WB @@ -450,7 +451,7 @@ .align 5 ENTRY(cpu_arm926_set_pgd) mov ip, #0 -#ifdef CONFIG_CPU_ARM926_WRITETHROUGH +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH /* Any reason why we don't use mcr p15, 0, r0, c7, c7, 0 here? --rmk */ mcr p15, 0, ip, c7, c6, 0 @ invalidate D cache #else @@ -475,7 +476,7 @@ */ .align 5 ENTRY(cpu_arm926_set_pmd) -#ifdef CONFIG_CPU_ARM926_WRITETHROUGH +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH eor r2, r1, #0x0a @ C & Section tst r2, #0x0b biceq r1, r1, #4 @ clear bufferable bit @@ -511,38 +512,33 @@ tst r1, #LPTE_PRESENT | LPTE_YOUNG @ Present and Young? movne r2, #0 -#ifdef CONFIG_CPU_ARM926_WRITETHROUGH +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH eor r3, r2, #0x0a @ C & small page? tst r3, #0x0b biceq r2, r2, #4 #endif str r2, [r0] @ hardware version mov r0, r0 -#ifndef CONFIG_CPU_ARM926_WRITETHROUGH +#ifndef CONFIG_CPU_DCACHE_WRITETHROUGH mcr p15, 0, r0, c7, c10, 1 @ clean D entry #endif mcr p15, 0, r0, c7, c10, 4 @ drain WB mov pc, lr -cpu_manu_name: - .asciz "ARM" ENTRY(cpu_arm926_name) .ascii "ARM926EJ-S" -#if defined(CONFIG_CPU_ARM926_CPU_IDLE) - .ascii "s" -#endif -#if defined(CONFIG_CPU_ARM926_I_CACHE_ON) +#ifndef CONFIG_CPU_ICACHE_DISABLE .ascii "i" #endif -#if defined(CONFIG_CPU_ARM926_D_CACHE_ON) +#ifndef CONFIG_CPU_DCACHE_DISABLE .ascii "d" -#if defined(CONFIG_CPU_ARM926_WRITETHROUGH) +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH .ascii "(wt)" #else .ascii "(wb)" #endif -#ifdef CONFIG_CPU_ARM926_ROUND_ROBIN +#ifdef CONFIG_CPU_CACHE_ROUND_ROBIN .ascii "RR" #endif #endif @@ -559,7 +555,7 @@ mcr p15, 0, r4, c2, c0 @ load page table pointer -#if defined(CONFIG_CPU_ARM926_WRITETHROUGH) +#ifdef CONFIG_CPU_DCACHE_WRITETHROUGH mov r0, #4 @ disable write-back on caches explicitly mcr p15, 7, r0, c15, c0, 0 #endif @@ -581,13 +577,13 @@ orr r0, r0, #0x0031 orr r0, r0, #0x2100 @ ..1. ...1 ..11 ...1 -#ifdef CONFIG_CPU_ARM926_ROUND_ROBIN +#ifdef CONFIG_CPU_CACHE_ROUND_ROBIN orr r0, r0, #0x4000 @ .1.. .... .... .... #endif -#ifdef CONFIG_CPU_ARM926_D_CACHE_ON +#ifndef CONFIG_CPU_DCACHE_DISABLE orr r0, r0, #0x0004 @ .... .... .... .1.. #endif -#ifdef CONFIG_CPU_ARM926_I_CACHE_ON +#ifndef CONFIG_CPU_ICACHE_DISABLE orr r0, r0, #0x1000 @ ...1 .... .... .... #endif mov pc, lr @@ -635,7 +631,7 @@ .type cpu_arm926_info, #object cpu_arm926_info: - .long cpu_manu_name + .long 0 .long cpu_arm926_name .size cpu_arm926_info, . - cpu_arm926_info @@ -660,7 +656,8 @@ b __arm926_setup .long cpu_arch_name .long cpu_elf_name - .long HWCAP_SWP | HWCAP_HALF | HWCAP_26BIT + .long HWCAP_SWP | HWCAP_HALF | HWCAP_THUMB | \ + HWCAP_FAST_MULT .long cpu_arm926_info .long arm926_processor_functions .size __arm926_proc_info, . - __arm926_proc_info diff -urN orig/arch/arm/mm/proc-sa110.S linux/arch/arm/mm/proc-sa110.S --- orig/arch/arm/mm/proc-sa110.S Mon Aug 5 13:29:47 2002 +++ linux/arch/arm/mm/proc-sa110.S Thu Jan 23 20:40:32 2003 @@ -159,43 +159,12 @@ mov pc, r0 /* - * cpu_sa110_do_idle(type) + * cpu_sa110_do_idle() * * Cause the processor to idle - * - * type: call type: - * 0 = slow idle - * 1 = fast idle - * 2 = switch to slow processor clock - * 3 = switch to fast processor clock */ .align 5 -idle: mcr p15, 0, r0, c15, c8, 2 @ Wait for interrupt, cache aligned - mov r0, r0 @ safety - mov pc, lr - ENTRY(cpu_sa110_do_idle) - mov ip, #0 - cmp r0, #4 - addcc pc, pc, r0, lsl #2 - mov pc, lr - - b idle - b idle - b slow_clock - b fast_clock - -fast_clock: - mcr p15, 0, ip, c15, c1, 2 @ enable clock switching - mov pc, lr - -slow_clock: - mcr p15, 0, ip, c15, c2, 2 @ disable clock switching - ldr r1, =UNCACHEABLE_ADDR @ load from uncacheable loc - ldr r1, [r1, #0] @ force switch to MCLK - mov pc, lr - - .align 5 ENTRY(cpu_sa1100_do_idle) mov r0, r0 @ 4 nop padding mov r0, r0 @@ -468,6 +437,9 @@ ENTRY(cpu_sa1100_tlb_invalidate_range) bic r0, r0, #0x0ff bic r0, r0, #0xf00 + sub r3, r1, r0 + cmp r3, #256 * PAGESIZE @ arbitary, should be tuned + bhi cpu_sa110_tlb_invalidate_all mov r3, #0 mcr p15, 0, r3, c7, c10, 4 @ drain WB 1: mcr p15, 0, r0, c8, c6, 1 @ invalidate D TLB entry @@ -595,8 +567,6 @@ mov pc, lr -cpu_manu_name: - .asciz "Intel" cpu_sa110_name: .asciz "StrongARM-110" cpu_sa1100_name: @@ -676,7 +646,7 @@ .type cpu_sa110_info, #object cpu_sa110_info: - .long cpu_manu_name + .long 0 .long cpu_sa110_name .size cpu_sa110_info, . - cpu_sa110_info @@ -720,12 +690,12 @@ .size sa1100_processor_functions, . - sa1100_processor_functions cpu_sa1100_info: - .long cpu_manu_name + .long 0 .long cpu_sa1100_name .size cpu_sa1100_info, . - cpu_sa1100_info cpu_sa1110_info: - .long cpu_manu_name + .long 0 .long cpu_sa1110_name .size cpu_sa1110_info, . - cpu_sa1110_info diff -urN orig/arch/arm/mm/small_page.c linux/arch/arm/mm/small_page.c --- orig/arch/arm/mm/small_page.c Sat Apr 28 11:24:54 2001 +++ linux/arch/arm/mm/small_page.c Fri Sep 6 11:33:02 2002 @@ -150,8 +150,8 @@ unsigned long flags; struct page *page; - page = virt_to_page(spage); - if (VALID_PAGE(page)) { + if (virt_addr_valid(spage)) { + page = virt_to_page(spage); /* * The container-page must be marked Reserved diff -urN orig/arch/arm/nwfpe/entry.S linux/arch/arm/nwfpe/entry.S --- orig/arch/arm/nwfpe/entry.S Mon Aug 5 13:29:47 2002 +++ linux/arch/arm/nwfpe/entry.S Thu Aug 8 10:42:24 2002 @@ -75,14 +75,14 @@ ldr r5, [sp, #60] @ get contents of PC; sub r8, r5, #4 -.Lx2: ldrt r0, [r8] @ get actual instruction into r0 +.Lx1: ldrt r0, [r8] @ get actual instruction into r0 emulate: bl EmulateAll @ emulate the instruction cmp r0, #0 @ was emulation successful moveq pc, r4 @ no, return failure next: -.Lx1: ldrt r6, [r5], #4 @ get the next instruction and +.Lx2: ldrt r6, [r5], #4 @ get the next instruction and @ increment PC and r2, r6, #0x0F000000 @ test for FP insns diff -urN orig/arch/arm/nwfpe/fpopcode.h linux/arch/arm/nwfpe/fpopcode.h --- orig/arch/arm/nwfpe/fpopcode.h Sat Apr 28 11:24:54 2001 +++ linux/arch/arm/nwfpe/fpopcode.h Wed Sep 4 17:37:19 2002 @@ -26,7 +26,7 @@ ARM Floating Point Instruction Classes | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | |c o n d|1 1 0 P|U|u|W|L| Rn |v| Fd |0|0|0|1| o f f s e t | CPDT -|c o n d|1 1 0 P|U|w|W|L| Rn |x| Fd |0|0|0|1| o f f s e t | CPDT +|c o n d|1 1 0 P|U|w|W|L| Rn |x| Fd |0|0|1|0| o f f s e t | CPDT (copro 2) | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | |c o n d|1 1 1 0|a|b|c|d|e| Fn |j| Fd |0|0|0|1|f|g|h|0|i| Fm | CPDO |c o n d|1 1 1 0|a|b|c|L|e| Fn | Rd |0|0|0|1|f|g|h|1|i| Fm | CPRT @@ -34,7 +34,7 @@ | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | CPDT data transfer instructions - LDF, STF, LFM, SFM + LDF, STF, LFM (copro 2), SFM (copro 2) CPDO dyadic arithmetic instructions ADF, MUF, SUF, RSF, DVF, RDF, diff -urN orig/arch/arm/nwfpe/softfloat.c linux/arch/arm/nwfpe/softfloat.c --- orig/arch/arm/nwfpe/softfloat.c Mon Aug 5 13:29:47 2002 +++ linux/arch/arm/nwfpe/softfloat.c Sun Sep 15 13:15:48 2002 @@ -184,9 +184,9 @@ { #if 0 float32 f; - __asm__("@ packFloat32; - mov %0, %1, asl #31; - orr %0, %2, asl #23; + __asm__("@ packFloat32 \n\ + mov %0, %1, asl #31 \n\ + orr %0, %2, asl #23 \n\ orr %0, %3" : /* no outputs */ : "g" (f), "g" (zSign), "g" (zExp), "g" (zSig) diff -urN orig/arch/arm/tools/mach-types linux/arch/arm/tools/mach-types --- orig/arch/arm/tools/mach-types Mon Aug 5 13:29:47 2002 +++ linux/arch/arm/tools/mach-types Wed Mar 19 16:01:37 2003 @@ -6,7 +6,7 @@ # To add an entry into this database, please see Documentation/arm/README, # or contact rmk@arm.linux.org.uk # -# Last update: Sat Mar 16 15:56:27 2002 +# Last update: Wed Mar 19 16:01:36 2003 # # machine_is_xxx CONFIG_xxxx MACH_TYPE_xxx number # @@ -71,7 +71,7 @@ jupiter SA1100_JUPITER JUPITER 59 psionw ARCH_PSIONW PSIONW 60 aln SA1100_ALN ALN 61 -epxa10db ARCH_CAMELOT CAMELOT 62 +epxa ARCH_CAMELOT CAMELOT 62 gds2200 SA1100_GDS2200 GDS2200 63 psion_series7 SA1100_PSION_SERIES7 PSION_SERIES7 64 xfile SA1100_XFILE XFILE 65 @@ -149,12 +149,12 @@ blue_v1 ARCH_BLUE_V1 BLUE_V1 138 pxa_cerf ARCH_PXA_CERF PXA_CERF 139 arm7tevb ARCH_ARM7TEVB ARM7TEVB 140 -d7400 ARCH_D7400 D7400 141 +d7400 SA1100_D7400 D7400 141 piranha ARCH_PIRANHA PIRANHA 142 sbcamelot SA1100_SBCAMELOT SBCAMELOT 143 kings SA1100_KINGS KINGS 144 smdk2400 ARCH_SMDK2400 SMDK2400 145 -collie ARCH_COLLIE COLLIE 146 +collie SA1100_COLLIE COLLIE 146 idr ARCH_IDR IDR 147 badge4 SA1100_BADGE4 BADGE4 148 webnet ARCH_WEBNET WEBNET 149 @@ -175,6 +175,139 @@ iam SA1100_IAM IAM 164 tt530 SA1100_TT530 TT530 165 sam2400 ARCH_SAM2400 SAM2400 166 -jornada56x ARCH_JORNADA56X JORNADA56X 167 +jornada56x SA1100_JORNADA56X JORNADA56X 167 active SA1100_ACTIVE ACTIVE 168 iq80321 ARCH_IQ80321 IQ80321 169 +wid SA1100_WID WID 170 +sabinal ARCH_SABINAL SABINAL 171 +ixp425_matacumbe ARCH_IXP425_MATACUMBE IXP425_MATACUMBE 172 +miniprint SA1100_MINIPRINT MINIPRINT 173 +adm510x ARCH_ADM510X ADM510X 174 +svs200 SA1100_SVS200 SVS200 175 +atg_tcu ARCH_ATG_TCU ATG_TCU 176 +jornada820 SA1100_JORNADA820 JORNADA820 177 +s3c44b0 ARCH_S3C44B0 S3C44B0 178 +margis2 ARCH_MARGIS2 MARGIS2 179 +ks8695 ARCH_KS8695 KS8695 180 +brh ARCH_BRH BRH 181 +s3c2410 ARCH_S3C2410 S3C2410 182 +possio_px30 ARCH_POSSIO_PX30 POSSIO_PX30 183 +s3c2800 ARCH_S3C2800 S3C2800 184 +fleetwood SA1100_FLEETWOOD FLEETWOOD 185 +omaha ARCH_OMAHA OMAHA 186 +ta7 ARCH_TA7 TA7 187 +nova SA1100_NOVA NOVA 188 +hmk ARCH_HMK HMK 189 +karo ARCH_KARO KARO 190 +fester SA1100_FESTER FESTER 191 +gpi ARCH_GPI GPI 192 +smdk2410 ARCH_SMDK2410 SMDK2410 193 +premium ARCH_PREMIUM PREMIUM 194 +nexio SA1100_NEXIO NEXIO 195 +bitbox SA1100_BITBOX BITBOX 196 +g200 SA1100_G200 G200 197 +gill SA1100_GILL GILL 198 +pxa_mercury ARCH_PXA_MERCURY PXA_MERCURY 199 +ceiva ARCH_CEIVA CEIVA 200 +fret SA1100_FRET FRET 201 +emailphone SA1100_EMAILPHONE EMAILPHONE 202 +h3900 ARCH_H3900 H3900 203 +pxa1 ARCH_PXA1 PXA1 204 +koan369 SA1100_KOAN369 KOAN369 205 +cogent ARCH_COGENT COGENT 206 +esl_simputer ARCH_ESL_SIMPUTER ESL_SIMPUTER 207 +esl_simputer_clr ARCH_ESL_SIMPUTER_CLR ESL_SIMPUTER_CLR 208 +esl_simputer_bw ARCH_ESL_SIMPUTER_BW ESL_SIMPUTER_BW 209 +hhp_cradle ARCH_HHP_CRADLE HHP_CRADLE 210 +he500 ARCH_HE500 HE500 211 +inhandelf2 SA1100_INHANDELF2 INHANDELF2 212 +inhandftip SA1100_INHANDFTIP INHANDFTIP 213 +dnp1110 SA1100_DNP1110 DNP1110 214 +pnp1110 SA1100_PNP1110 PNP1110 215 +csb226 ARCH_CSB226 CSB226 216 +arnold SA1100_ARNOLD ARNOLD 217 +psiboard SA1100_PSIBOARD PSIBOARD 218 +jz8028 ARCH_JZ8028 JZ8028 219 +h5400 ARCH_IPAQ3 IPAQ3 220 +forte SA1100_FORTE FORTE 221 +acam SA1100_ACAM ACAM 222 +abox SA1100_ABOX ABOX 223 +atmel ARCH_ATMEL ATMEL 224 +sitsang ARCH_SITSANG SITSANG 225 +cpu1110lcdnet SA1100_CPU1110LCDNET CPU1110LCDNET 226 +mpl_vcma9 ARCH_MPL_VCMA9 MPL_VCMA9 227 +opus_a1 ARCH_OPUS_A1 OPUS_A1 228 +daytona ARCH_DAYTONA DAYTONA 229 +killbear SA1100_KILLBEAR KILLBEAR 230 +yoho ARCH_YOHO YOHO 231 +jasper ARCH_JASPER JASPER 232 +dsc25 ARCH_DSC25 DSC25 233 +innovator ARCH_INNOVATOR INNOVATOR 234 +ramses ARCH_RAMSES RAMSES 235 +s28x ARCH_S28X S28X 236 +mport3 ARCH_MPORT3 MPORT3 237 +pxa_eagle250 ARCH_PXA_EAGLE250 PXA_EAGLE250 238 +pdb ARCH_PDB PDB 239 +blue_2g SA1100_BLUE_2G BLUE_2G 240 +bluearch SA1100_BLUEARCH BLUEARCH 241 +ixdp2400 ARCH_IXMB2400 IXMB2400 242 +ixdp2800 ARCH_IXMB2800 IXMB2800 243 +explorer SA1100_EXPLORER EXPLORER 244 +ixdp425 ARCH_IXDP425 IXDP425 245 +chimp ARCH_CHIMP CHIMP 246 +stork_nest ARCH_STORK_NEST STORK_NEST 247 +stork_egg ARCH_STORK_EGG STORK_EGG 248 +wismo SA1100_WISMO WISMO 249 +ezlinx ARCH_EZLINX EZLINX 250 +at91 ARCH_AT91 AT91 251 +orion ARCH_ORION ORION 252 +neptune ARCH_NEPTUNE NEPTUNE 253 +hackkit SA1100_HACKKIT HACKKIT 254 +pxa_wins30 ARCH_PXA_WINS30 PXA_WINS30 255 +lavinna SA1100_LAVINNA LAVINNA 256 +pxa_uengine ARCH_PXA_UENGINE PXA_UENGINE 257 +innokom ARCH_INNOKOM INNOKOM 258 +bms ARCH_BMS BMS 259 +ixcdp1100 ARCH_IXCDP1100 IXCDP1100 260 +prpmc1100 ARCH_PRPMC1100 PRPMC1100 261 +at91rm9200dk ARCH_AT91RM9200DK AT91RM9200DK 262 +armstick ARCH_ARMSTICK ARMSTICK 263 +armonie ARCH_ARMONIE ARMONIE 264 +mport1 ARCH_MPORT1 MPORT1 265 +s3c5410 ARCH_S3C5410 S3C5410 266 +zcp320a ARCH_ZCP320A ZCP320A 267 +i_box ARCH_I_BOX I_BOX 268 +stlc1502 ARCH_STLC1502 STLC1502 269 +siren ARCH_SIREN SIREN 270 +greenlake ARCH_GREENLAKE GREENLAKE 271 +argus ARCH_ARGUS ARGUS 272 +combadge SA1100_COMBADGE COMBADGE 273 +rokepxa ARCH_ROKEPXA ROKEPXA 274 +cintegrator ARCH_CINTEGRATOR CINTEGRATOR 275 +guidea07 ARCH_GUIDEA07 GUIDEA07 276 +tat257 ARCH_TAT257 TAT257 277 +igp2425 ARCH_IGP2425 IGP2425 278 +bluegrama ARCH_BLUEGRAMMA BLUEGRAMMA 279 +ipod ARCH_IPOD IPOD 280 +adsbitsyx ARCH_ADSBITSYX ADSBITSYX 281 +trizeps2 ARCH_TRIZEPS2 TRIZEPS2 282 +viper ARCH_VIPER VIPER 283 +adsbitsyplus SA1100_ADSBITSYPLUS ADSBITSYPLUS 284 +adsagc SA1100_ADSAGC ADSAGC 285 +stp7312 ARCH_STP7312 STP7312 286 +nx_phnx ARCH_PXA255 PXA255 287 +wep_ep250 ARCH_WEP_EP250 WEP_EP250 288 +inhandelf3 ARCH_INHANDELF3 INHANDELF3 289 +adi_coyote ARCH_ADI_COYOTE ADI_COYOTE 290 +iyonix ARCH_IYONIX IYONIX 291 +damicam_sa1110 ARCH_DAMICAM_SA1110 DAMICAM_SA1110 292 +meg03 ARCH_MEG03 MEG03 293 +pxa_whitechapel ARCH_PXA_WHITECHAPEL PXA_WHITECHAPEL 294 +nwsc ARCH_NWSC NWSC 295 +nwlarm ARCH_NWLARM NWLARM 296 +ixp425_mguard ARCH_IXP425_MGUARD IXP425_MGUARD 297 +pxa_netdcu4 ARCH_PXA_NETDCU4 PXA_NETDCU4 298 +ixdp2401 ARCH_IXDP2401 IXDP2401 299 +ixdp2801 ARCH_IXDP2801 IXDP2801 300 +zodiac ARCH_ZODIAC ZODIAC 301 +armmodul ARCH_ARMMODUL ARMMODUL 302 diff -urN orig/arch/i386/config.in linux/arch/i386/config.in --- orig/arch/i386/config.in Mon Aug 5 13:29:50 2002 +++ linux/arch/i386/config.in Mon Aug 5 13:41:29 2002 @@ -10,6 +10,7 @@ define_bool CONFIG_UID16 y +define_bool CONFIG_GENERIC_ISA_DMA y mainmenu_option next_comment comment 'Code maturity level options' bool 'Prompt for development and/or incomplete code/drivers' CONFIG_EXPERIMENTAL diff -urN orig/arch/i386/kernel/Makefile linux/arch/i386/kernel/Makefile --- orig/arch/i386/kernel/Makefile Fri Nov 23 10:12:07 2001 +++ linux/arch/i386/kernel/Makefile Tue Nov 13 12:34:50 2001 @@ -7,8 +7,8 @@ # # Note 2! The CFLAGS definitions are now in the main makefile... -.S.o: - $(CC) $(AFLAGS) -traditional -c $< -o $*.o +USE_STANDARD_AS_RULE := true +EXTRA_AFLAGS := -traditional all: kernel.o head.o init_task.o diff -urN orig/arch/i386/kernel/apm.c linux/arch/i386/kernel/apm.c --- orig/arch/i386/kernel/apm.c Mon Aug 5 13:29:50 2002 +++ linux/arch/i386/kernel/apm.c Mon Aug 5 13:41:31 2002 @@ -1216,6 +1216,7 @@ as->suspend_wait = 0; as->suspend_result = err; } + ignore_normal_resume = 1; wake_up_interruptible(&apm_suspend_waitqueue); return err; } @@ -1268,6 +1269,8 @@ if (ignore_bounce && ((jiffies - last_resume) > bounce_interval)) ignore_bounce = 0; + if (ignore_normal_resume && (event != APM_NORMAL_RESUME)) + ignore_normal_resume = 0; switch (event) { case APM_SYS_STANDBY: diff -urN orig/arch/i386/lib/Makefile linux/arch/i386/lib/Makefile --- orig/arch/i386/lib/Makefile Mon Oct 1 23:10:05 2001 +++ linux/arch/i386/lib/Makefile Sun Aug 12 17:52:40 2001 @@ -2,8 +2,7 @@ # Makefile for i386-specific library files.. # -.S.o: - $(CC) $(AFLAGS) -c $< -o $*.o +USE_STANDARD_AS_RULE := true L_TARGET = lib.a diff -urN orig/arch/i386/math-emu/Makefile linux/arch/i386/math-emu/Makefile --- orig/arch/i386/math-emu/Makefile Sun Dec 31 22:12:08 2000 +++ linux/arch/i386/math-emu/Makefile Sun Dec 17 22:31:38 2000 @@ -2,15 +2,15 @@ # Makefile for wm-FPU-emu # +USE_STANDARD_AS_RULE := true + O_TARGET := math.o #DEBUG = -DDEBUGGING DEBUG = PARANOID = -DPARANOID CFLAGS := $(CFLAGS) $(PARANOID) $(DEBUG) -fno-builtin $(MATH_EMULATION) - -.S.o: - $(CC) $(AFLAGS) $(PARANOID) -c $< +EXTRA_AFLAGS := $(PARANOID) # From 'C' language sources: C_OBJS =fpu_entry.o errors.o \ diff -urN orig/arch/ia64/config.in linux/arch/ia64/config.in --- orig/arch/ia64/config.in Mon Aug 5 13:29:51 2002 +++ linux/arch/ia64/config.in Mon Aug 5 13:41:34 2002 @@ -25,6 +25,7 @@ define_bool CONFIG_SBUS n define_bool CONFIG_RWSEM_GENERIC_SPINLOCK y define_bool CONFIG_RWSEM_XCHGADD_ALGORITHM n +define_bool CONFIG_GENERIC_ISA_DMA y if [ "$CONFIG_IA64_HP_SIM" = "n" ]; then define_bool CONFIG_ACPI y diff -urN orig/arch/m68k/config.in linux/arch/m68k/config.in --- orig/arch/m68k/config.in Mon Aug 5 13:30:02 2002 +++ linux/arch/m68k/config.in Mon Aug 5 13:41:57 2002 @@ -6,6 +6,7 @@ define_bool CONFIG_UID16 y define_bool CONFIG_RWSEM_GENERIC_SPINLOCK y define_bool CONFIG_RWSEM_XCHGADD_ALGORITHM n +define_bool CONFIG_GENERIC_ISA_DMA y mainmenu_name "Linux/68k Kernel Configuration" diff -urN orig/arch/mips/config.in linux/arch/mips/config.in --- orig/arch/mips/config.in Mon Aug 5 13:30:04 2002 +++ linux/arch/mips/config.in Mon Aug 5 14:21:14 2002 @@ -4,6 +4,7 @@ # define_bool CONFIG_MIPS y define_bool CONFIG_MIPS32 y +define_bool CONFIG_GENERIC_ISA_DMA y mainmenu_name "Linux/MIPS Kernel Configuration" diff -urN orig/arch/mips64/config.in linux/arch/mips64/config.in --- orig/arch/mips64/config.in Mon Aug 5 13:30:18 2002 +++ linux/arch/mips64/config.in Mon Aug 5 14:02:20 2002 @@ -55,6 +55,7 @@ define_bool CONFIG_RWSEM_GENERIC_SPINLOCK y define_bool CONFIG_RWSEM_XCHGADD_ALGORITHM n +define_bool CONFIG_GENERIC_ISA_DMA y # # Select some configuration options automatically based on user selections. diff -urN orig/arch/parisc/config.in linux/arch/parisc/config.in --- orig/arch/parisc/config.in Fri Nov 16 10:08:22 2001 +++ linux/arch/parisc/config.in Tue Nov 13 12:35:22 2001 @@ -9,6 +9,7 @@ define_bool CONFIG_UID16 n define_bool CONFIG_RWSEM_GENERIC_SPINLOCK y define_bool CONFIG_RWSEM_XCHGADD_ALGORITHM n +define_bool CONFIG_GENERIC_ISA_DMA y mainmenu_option next_comment comment 'Code maturity level options' diff -urN orig/arch/ppc/Makefile linux/arch/ppc/Makefile --- orig/arch/ppc/Makefile Fri Oct 26 16:45:59 2001 +++ linux/arch/ppc/Makefile Tue Oct 2 10:40:47 2001 @@ -21,7 +21,7 @@ CHECKS = checks endif -ASFLAGS = +AFLAGS = LINKFLAGS = -T arch/ppc/vmlinux.lds -Ttext $(KERNELLOAD) -Bstatic CPPFLAGS := $(CPPFLAGS) -D__powerpc__ CFLAGS := $(CFLAGS) -D__powerpc__ -fsigned-char \ diff -urN orig/arch/ppc/config.in linux/arch/ppc/config.in --- orig/arch/ppc/config.in Mon Aug 5 13:30:25 2002 +++ linux/arch/ppc/config.in Mon Aug 5 13:42:42 2002 @@ -7,6 +7,7 @@ define_bool CONFIG_RWSEM_GENERIC_SPINLOCK n define_bool CONFIG_RWSEM_XCHGADD_ALGORITHM y define_bool CONFIG_HAVE_DEC_LOCK y +define_bool CONFIG_GENERIC_ISA_DMA y mainmenu_name "Linux/PowerPC Kernel Configuration" diff -urN orig/arch/sh/config.in linux/arch/sh/config.in --- orig/arch/sh/config.in Wed Feb 27 14:24:53 2002 +++ linux/arch/sh/config.in Tue Feb 26 17:37:54 2002 @@ -9,6 +9,7 @@ define_bool CONFIG_UID16 y define_bool CONFIG_RWSEM_GENERIC_SPINLOCK y define_bool CONFIG_RWSEM_XCHGADD_ALGORITHM n +define_bool CONFIG_GENERIC_ISA_DMA y mainmenu_option next_comment comment 'Code maturity level options' diff -urN orig/arch/sparc/config.in linux/arch/sparc/config.in --- orig/arch/sparc/config.in Mon Aug 5 13:30:38 2002 +++ linux/arch/sparc/config.in Mon Aug 5 13:43:01 2002 @@ -6,6 +6,7 @@ define_bool CONFIG_UID16 y define_bool CONFIG_HIGHMEM y +define_bool CONFIG_GENERIC_ISA_DMA y mainmenu_option next_comment comment 'Code maturity level options' diff -urN orig/arch/sparc64/config.in linux/arch/sparc64/config.in --- orig/arch/sparc64/config.in Mon Aug 5 13:30:38 2002 +++ linux/arch/sparc64/config.in Mon Aug 5 13:43:04 2002 @@ -37,6 +37,7 @@ define_bool CONFIG_HAVE_DEC_LOCK y define_bool CONFIG_RWSEM_GENERIC_SPINLOCK n define_bool CONFIG_RWSEM_XCHGADD_ALGORITHM y +define_bool CONFIG_GENERIC_ISA_DMA y define_bool CONFIG_ISA n define_bool CONFIG_ISAPNP n define_bool CONFIG_EISA n diff -urN orig/drivers/Makefile linux/drivers/Makefile --- orig/drivers/Makefile Mon Aug 5 13:30:41 2002 +++ linux/drivers/Makefile Mon Aug 5 13:43:09 2002 @@ -8,9 +8,9 @@ mod-subdirs := dio mtd sbus video macintosh usb input telephony sgi ide \ message/i2o message/fusion scsi md ieee1394 pnp isdn atm \ - fc4 net/hamradio i2c acpi bluetooth + fc4 net/hamradio i2c l3 acpi bluetooth serial -subdir-y := parport char block net sound misc media cdrom hotplug +subdir-y := parport serial char block net sound misc media cdrom hotplug pld subdir-m := $(subdir-y) @@ -43,8 +43,10 @@ # CONFIG_HAMRADIO can be set without CONFIG_NETDEVICE being set -- ch subdir-$(CONFIG_HAMRADIO) += net/hamradio subdir-$(CONFIG_I2C) += i2c +subdir-$(CONFIG_L3) += l3 subdir-$(CONFIG_ACPI) += acpi subdir-$(CONFIG_BLUEZ) += bluetooth +subdir-$(CONFIG_SSI) += ssi include $(TOPDIR)/Rules.make diff -urN orig/drivers/acorn/block/fd1772.c linux/drivers/acorn/block/fd1772.c --- orig/drivers/acorn/block/fd1772.c Mon Aug 5 13:30:41 2002 +++ linux/drivers/acorn/block/fd1772.c Mon Dec 30 15:14:14 2002 @@ -1567,29 +1567,30 @@ int fd1772_init(void) { - int i; + int i, err; if (!machine_is_archimedes()) return 0; - if (register_blkdev(MAJOR_NR, "fd", &floppy_fops)) { + err = register_blkdev(MAJOR_NR, "fd", &floppy_fops); + if (err) { printk("Unable to get major %d for floppy\n", MAJOR_NR); - return 1; + goto err_out; } + err = -EBUSY; if (request_dma(FLOPPY_DMA, "fd1772")) { printk("Unable to grab DMA%d for the floppy (1772) driver\n", FLOPPY_DMA); - return 1; + goto err_blkdev; }; if (request_dma(FIQ_FD1772, "fd1772 end")) { printk("Unable to grab DMA%d for the floppy (1772) driver\n", FIQ_FD1772); - free_dma(FLOPPY_DMA); - return 1; + goto err_dma1; }; - enable_dma(FIQ_FD1772); /* This inserts a call to our command end routine */ /* initialize variables */ + err = -ENOMEM; SelectedDrive = -1; #ifdef TRACKBUFFER BufferDrive = BufferSide = BufferTrack = -1; @@ -1601,7 +1602,10 @@ out of some special memory... */ DMABuffer = (char *) kmalloc(2048); /* Copes with pretty large sectors */ #endif + if (DMABuffer == NULL) + goto err_dma2; + enable_dma(FIQ_FD1772); /* This inserts a call to our command end routine */ for (i = 0; i < FD_MAX_UNITS; i++) { unit[i].track = -1; } @@ -1619,4 +1623,13 @@ config_types(); return 0; + + err_dma2: + free_dma(FIQ_FD1772); + err_dma1: + free_dma(FLOPPY_DMA); + err_blkdev: + unregister_blkdev(MAJOR_NR, &floppy_fops); + err_out: + return err; } diff -urN orig/drivers/acorn/char/Makefile linux/drivers/acorn/char/Makefile --- orig/drivers/acorn/char/Makefile Mon Sep 3 14:14:43 2001 +++ linux/drivers/acorn/char/Makefile Fri Dec 20 17:03:01 2002 @@ -24,7 +24,9 @@ obj-arc := keyb_arc.o obj-rpc := keyb_ps2.o obj-clps7500 := keyb_ps2.o defkeymap-acorn.o +obj-riscstation := keyb_ps2.o defkeymap-acorn.o +obj-$(CONFIG_7K5MOUSE) += mouse_ps2.o obj-$(CONFIG_RPCMOUSE) += mouse_rpc.o obj-$(CONFIG_ATOMWIDE_SERIAL) += serial-atomwide.o obj-$(CONFIG_DUALSP_SERIAL) += serial-dualsp.o diff -urN orig/drivers/acorn/char/i2c.c linux/drivers/acorn/char/i2c.c --- orig/drivers/acorn/char/i2c.c Sat Mar 31 23:47:19 2001 +++ linux/drivers/acorn/char/i2c.c Thu Jan 3 10:29:48 2002 @@ -14,6 +14,9 @@ */ #include #include +#include +#include +#include #include #include @@ -21,15 +24,19 @@ #include #include #include +#include #include "pcf8583.h" -extern unsigned long -mktime(unsigned int year, unsigned int mon, unsigned int day, - unsigned int hour, unsigned int min, unsigned int sec); extern int (*set_rtc)(void); static struct i2c_client *rtc_client; +static const unsigned char days_in_mon[] = + { 0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 }; +static unsigned int rtc_epoch = 1900; + +#define CMOS_CHECKSUM (63) +#define CMOS_YEAR (64 + 128) static inline int rtc_command(int cmd, void *data) { @@ -44,12 +51,10 @@ /* * Read the current RTC time and date, and update xtime. */ -static void get_rtc_time(void) +static void get_rtc_time(struct rtc_tm *rtctm, unsigned int *year) { - unsigned char ctrl; - unsigned char year; - struct rtc_tm rtctm; - struct mem rtcmem = { 0xc0, 1, &year }; + unsigned char ctrl, yr[2]; + struct mem rtcmem = { CMOS_YEAR, sizeof(yr), yr }; /* * Ensure that the RTC is running. @@ -73,22 +78,53 @@ if (rtc_command(MEM_READ, &rtcmem)) return; - if (rtc_command(RTC_GETDATETIME, &rtctm)) + if (rtc_command(RTC_GETDATETIME, rtctm)) return; - if (year < 70) - year += 100; + *year = yr[1] * 100 + yr[0]; +} + +static int set_rtc_time(struct rtc_tm *rtctm, unsigned int year) +{ + unsigned char yr[2], leap, chk; + struct mem cmos_year = { CMOS_YEAR, sizeof(yr), yr }; + struct mem cmos_check = { CMOS_CHECKSUM, 1, &chk }; + int ret; + + leap = (!(year % 4) && (year % 100)) || !(year % 400); + + if (rtctm->mon > 12 || rtctm->mday == 0) + return -EINVAL; + + if (rtctm->mday > (days_in_mon[rtctm->mon] + (rtctm->mon == 2 && leap))) + return -EINVAL; + + if (rtctm->hours >= 24 || rtctm->mins >= 60 || rtctm->secs >= 60) + return -EINVAL; - xtime.tv_usec = rtctm.cs * 10000; - xtime.tv_sec = mktime(1900 + year, rtctm.mon, rtctm.mday, - rtctm.hours, rtctm.mins, rtctm.secs); + ret = rtc_command(RTC_SETDATETIME, rtctm); + if (ret == 0) { + rtc_command(MEM_READ, &cmos_check); + rtc_command(MEM_READ, &cmos_year); + + chk -= yr[1] + yr[0]; + + yr[1] = year / 100; + yr[0] = year % 100; + + chk += yr[1] + yr[0]; + + rtc_command(MEM_WRITE, &cmos_year); + rtc_command(MEM_WRITE, &cmos_check); + } + return ret; } /* * Set the RTC time only. Note that * we do not touch the date. */ -static int set_rtc_time(void) +static int k_set_rtc_time(void) { struct rtc_tm new_rtctm, old_rtctm; unsigned long nowtime = xtime.tv_sec; @@ -110,13 +146,70 @@ * [ rtc: 1/1/2000 23:58:00, real 2/1/2000 00:01:00, * rtc gets set to 1/1/2000 00:01:00 ] */ - if ((old_rtctm.hours == 23 && old_rtctm.mins == 59) || - (new_rtctm.hours == 23 && new_rtctm.mins == 59)) + if ((old_rtctm.hours == 23 && old_rtctm.mins == 59) || + (new_rtctm.hours == 23 && new_rtctm.mins == 59)) return 1; return rtc_command(RTC_SETTIME, &new_rtctm); } +static int rtc_ioctl(struct inode *inode, struct file *file, + unsigned int cmd, unsigned long arg) +{ + unsigned int year; + struct rtc_time rtctm; + struct rtc_tm rtc_raw; + + switch (cmd) { + case RTC_ALM_READ: + case RTC_ALM_SET: + break; + + case RTC_RD_TIME: + get_rtc_time(&rtc_raw, &year); + rtctm.tm_sec = rtc_raw.secs; + rtctm.tm_min = rtc_raw.mins; + rtctm.tm_hour = rtc_raw.hours; + rtctm.tm_mday = rtc_raw.mday; + rtctm.tm_mon = rtc_raw.mon - 1; /* month starts at 0 */ + rtctm.tm_year = year - 1900; /* starts at 1900 */ + return copy_to_user((void *)arg, &rtctm, sizeof(rtctm)) + ? -EFAULT : 0; + + case RTC_SET_TIME: + if (!capable(CAP_SYS_TIME)) + return -EACCES; + + if (copy_from_user(&rtctm, (void *)arg, sizeof(rtctm))) + return -EFAULT; + rtc_raw.secs = rtctm.tm_sec; + rtc_raw.mins = rtctm.tm_min; + rtc_raw.hours = rtctm.tm_hour; + rtc_raw.mday = rtctm.tm_mday; + rtc_raw.mon = rtctm.tm_mon + 1; + rtc_raw.year_off = 2; + year = rtctm.tm_year + 1900; + return set_rtc_time(&rtc_raw, year); + break; + + case RTC_EPOCH_READ: + return put_user(rtc_epoch, (unsigned long *)arg); + + } + return -EINVAL; +} + +static struct file_operations rtc_fops = { + ioctl: rtc_ioctl, +}; + +static struct miscdevice rtc_dev = { + minor: RTC_MINOR, + name: "rtc", + fops: &rtc_fops, +}; + +/* IOC / IOMD i2c driver */ #define FORCE_ONES 0xdc #define SCL 0x02 @@ -184,9 +277,16 @@ { if (client->id == I2C_DRIVERID_PCF8583 && client->addr == 0x50) { + struct rtc_tm rtctm; + unsigned int year; + rtc_client = client; - get_rtc_time(); - set_rtc = set_rtc_time; + get_rtc_time(&rtctm, &year); + + xtime.tv_usec = rtctm.cs * 10000; + xtime.tv_sec = mktime(year, rtctm.mon, rtctm.mday, + rtctm.hours, rtctm.mins, rtctm.secs); + set_rtc = k_set_rtc_time; } return 0; @@ -212,9 +312,16 @@ static int __init i2c_ioc_init(void) { + int ret; + force_ones = FORCE_ONES | SCL | SDA; - return i2c_bit_add_bus(&ioc_ops); + ret = i2c_bit_add_bus(&ioc_ops); + + if (ret >= 0) + misc_register(&rtc_dev); + + return ret; } __initcall(i2c_ioc_init); diff -urN orig/drivers/acorn/char/keyb_arc.c linux/drivers/acorn/char/keyb_arc.c --- orig/drivers/acorn/char/keyb_arc.c Mon Sep 3 14:14:43 2001 +++ linux/drivers/acorn/char/keyb_arc.c Mon Aug 27 18:06:17 2001 @@ -419,11 +419,50 @@ disable_irq(irq); } +static int a5kkbd_setkeycode(unsigned int scancode, unsigned int keycode) +{ + return -EINVAL; +} + +static int a5kkbd_getkeycode(unsigned int scancode) +{ + return -EINVAL; +} + +static int a5kkbd_translate(unsigned char scancode, unsigned char *keycode, char rawmode) +{ + *keycode = scancode; + return 1; +} + +static char a5kkbd_unexpected_up(unsigned char keycode) +{ + return 0200; +} + +static int a5kkbd_rate(struct kbd_repeat *rep) +{ + return -EINVAL; +} + #ifdef CONFIG_KBDMOUSE static struct busmouse a5kkbd_mouse = { 6, "kbdmouse", NULL, NULL, NULL, 7 }; #endif + +struct kbd_ops_struct a5k_kbd_ops = { + k_setkeycode: a5kkbd_setkeycode, + k_getkeycode: a5kkbd_getkeycode, + k_translate: a5kkbd_translate, + k_unexpected_up: a5kkbd_unexpected_up, + k_leds: a5kkbd_leds, + k_rate: a5kkbd_rate, +#ifdef CONFIG_MAGIC_SYSRQ + k_sysrq_xlate: a5kkbd_sysrq_xlate, + k_sysrq_key: 13, +#endif +}; void __init a5kkbd_init_hw (void) { diff -urN orig/drivers/acorn/char/mouse_ps2.c linux/drivers/acorn/char/mouse_ps2.c --- orig/drivers/acorn/char/mouse_ps2.c Mon Aug 5 13:30:41 2002 +++ linux/drivers/acorn/char/mouse_ps2.c Fri Dec 20 17:03:01 2002 @@ -1,8 +1,6 @@ /* * Driver for PS/2 mouse on IOMD interface */ - -#include #include #include #include @@ -288,3 +286,6 @@ return 0; } + +module_init(psaux_init); + diff -urN orig/drivers/acorn/char/pcf8583.c linux/drivers/acorn/char/pcf8583.c --- orig/drivers/acorn/char/pcf8583.c Sat Mar 31 23:47:19 2001 +++ linux/drivers/acorn/char/pcf8583.c Mon Dec 30 15:00:20 2002 @@ -73,6 +73,7 @@ pcf8583_detach(struct i2c_client *client) { i2c_detach_client(client); + kfree(client); return 0; } diff -urN orig/drivers/acorn/char/serial-atomwide.c linux/drivers/acorn/char/serial-atomwide.c --- orig/drivers/acorn/char/serial-atomwide.c Tue Oct 3 20:08:24 2000 +++ linux/drivers/acorn/char/serial-atomwide.c Thu Jan 3 20:57:09 2002 @@ -20,7 +20,4 @@ #define MY_PORT_ADDRESS(port,cardaddr) \ ((cardaddr) + 0x200 - (port) * 0x100) -#define INIT serial_card_atomwide_init -#define EXIT serial_card_atomwide_exit - #include "serial-card.c" diff -urN orig/drivers/acorn/char/serial-card.c linux/drivers/acorn/char/serial-card.c --- orig/drivers/acorn/char/serial-card.c Mon Oct 1 23:10:27 2001 +++ linux/drivers/acorn/char/serial-card.c Mon Mar 11 16:16:57 2002 @@ -29,6 +29,7 @@ #include #include #include +#include #include #include @@ -38,95 +39,84 @@ #define NUM_SERIALS MY_NUMPORTS * MAX_ECARDS #endif -#ifdef MODULE -static int __serial_ports[NUM_SERIALS]; -static int __serial_pcount; -static int __serial_addr[NUM_SERIALS]; +static int serial_ports[NUM_SERIALS]; +static int serial_pcount; +static int serial_addr[NUM_SERIALS]; static struct expansion_card *expcard[MAX_ECARDS]; -#define ADD_ECARD(ec,card) expcard[(card)] = (ec) -#define ADD_PORT(port,addr) \ - do { \ - __serial_ports[__serial_pcount] = (port); \ - __serial_addr[__serial_pcount] = (addr); \ - __serial_pcount += 1; \ - } while (0) -#else -#define ADD_ECARD(ec,card) -#define ADD_PORT(port,addr) -#endif static const card_ids serial_cids[] = { MY_CARD_LIST, { 0xffff, 0xffff } }; static inline int serial_register_onedev (unsigned long port, int irq) { - struct serial_struct req; + struct serial_struct req; - memset(&req, 0, sizeof(req)); - req.baud_base = MY_BAUD_BASE; - req.irq = irq; - req.port = port; - req.flags = 0; + memset(&req, 0, sizeof(req)); + req.baud_base = MY_BAUD_BASE; + req.irq = irq; + req.port = port; + req.flags = 0; - return register_serial(&req); + return register_serial(&req); } -static int __init INIT (void) +static int __init serial_card_init(void) { - int card = 0; - - ecard_startfind (); - - do { - struct expansion_card *ec; - unsigned long cardaddr; - int port; - - ec = ecard_find (0, serial_cids); - if (!ec) - break; - - cardaddr = MY_BASE_ADDRESS(ec); - - for (port = 0; port < MY_NUMPORTS; port ++) { - unsigned long address; - int line; + int card = 0; - address = MY_PORT_ADDRESS(port, cardaddr); + ecard_startfind (); - line = serial_register_onedev (address, ec->irq); - if (line < 0) - break; - ADD_PORT(line, address); - } - - if (port) { - ecard_claim (ec); - ADD_ECARD(ec, card); - } else - break; - } while (++card < MAX_ECARDS); - return card ? 0 : -ENODEV; + do { + struct expansion_card *ec; + unsigned long cardaddr; + int port; + + ec = ecard_find (0, serial_cids); + if (!ec) + break; + + cardaddr = MY_BASE_ADDRESS(ec); + + for (port = 0; port < MY_NUMPORTS; port ++) { + unsigned long address; + int line; + + address = MY_PORT_ADDRESS(port, cardaddr); + + line = serial_register_onedev (address, ec->irq); + if (line < 0) + break; + serial_ports[serial_pcount] = line; + serial_addr[serial_pcount] = address; + serial_pcount += 1; + } + + if (port) { + ecard_claim (ec); + expcard[card] = ec; + } else + break; + } while (++card < MAX_ECARDS); + return card ? 0 : -ENODEV; } -static void __exit EXIT (void) +static void __exit serial_card_exit(void) { -#ifdef MODULE - int i; + int i; - for (i = 0; i < __serial_pcount; i++) { - unregister_serial(__serial_ports[i]); - release_region(__serial_addr[i], 8); - } - - for (i = 0; i < MAX_ECARDS; i++) - if (expcard[i]) - ecard_release (expcard[i]); -#endif + for (i = 0; i < serial_pcount; i++) { + unregister_serial(serial_ports[i]); + release_region(serial_addr[i], 8); + } + + for (i = 0; i < MAX_ECARDS; i++) + if (expcard[i]) + ecard_release (expcard[i]); } EXPORT_NO_SYMBOLS; +MODULE_AUTHOR("Russell King"); MODULE_LICENSE("GPL"); -module_init(INIT); -module_exit(EXIT); +module_init(serial_card_init); +module_exit(serial_card_exit); diff -urN orig/drivers/acorn/char/serial-dualsp.c linux/drivers/acorn/char/serial-dualsp.c --- orig/drivers/acorn/char/serial-dualsp.c Tue Oct 3 20:08:24 2000 +++ linux/drivers/acorn/char/serial-dualsp.c Thu Jan 3 20:57:09 2002 @@ -18,7 +18,4 @@ #define MY_PORT_ADDRESS(port,cardaddress) \ ((cardaddress) + (port) * 8) -#define INIT serial_card_dualsp_init -#define EXIT serial_card_dualsp_exit - #include "serial-card.c" diff -urN orig/drivers/acorn/net/ether1.c linux/drivers/acorn/net/ether1.c --- orig/drivers/acorn/net/ether1.c Mon Oct 1 23:10:27 2001 +++ linux/drivers/acorn/net/ether1.c Sun Sep 15 20:43:59 2002 @@ -647,6 +647,11 @@ { struct ether1_priv *priv = (struct ether1_priv *)dev->priv; + if (!is_valid_ether_addr(dev->dev_addr)) { + printk("%s: invalid ethernet MAC address\n", dev->name); + return -EINVAL; + } + if (request_irq(dev->irq, ether1_interrupt, 0, "ether1", dev)) return -EAGAIN; @@ -971,6 +976,23 @@ return &priv->stats; } +static int +ether1_set_mac_address(struct net_device *dev, void *p) +{ + struct sockaddr *addr = p; + + if (netif_running(dev)) + return -EBUSY; + + memcpy(dev->dev_addr, addr->sa_data, dev->addr_len); + + /* + * We'll set the MAC address on the chip when we open it. + */ + + return 0; +} + /* * Set or clear the multicast filter for this adaptor. * num_addrs == -1 Promiscuous mode, receive all packets. @@ -1038,6 +1060,7 @@ dev->hard_start_xmit = ether1_sendpacket; dev->get_stats = ether1_getstats; dev->set_multicast_list = ether1_setmulticastlist; + dev->set_mac_address = ether1_set_mac_address; dev->tx_timeout = ether1_timeout; dev->watchdog_timeo = 5 * HZ / 100; return 0; diff -urN orig/drivers/acorn/net/ether3.c linux/drivers/acorn/net/ether3.c --- orig/drivers/acorn/net/ether3.c Mon Oct 1 23:10:27 2001 +++ linux/drivers/acorn/net/ether3.c Sun Sep 15 20:45:30 2002 @@ -417,6 +417,9 @@ static int ether3_open(struct net_device *dev) { + if (!is_valid_ether_addr(dev->dev_addr)) + return -EINVAL; + if (request_irq(dev->irq, ether3_interrupt, 0, "ether3", dev)) return -EAGAIN; @@ -460,6 +463,23 @@ return &priv->stats; } +static int +ether3_set_mac_address(struct net_device *dev, void *p) +{ + struct sockaddr *addr = p; + + if (netif_running(dev)) + return -EBUSY; + + memcpy(dev->dev_addr, addr->sa_data, dev->addr_len); + + /* + * We'll set the MAC address on the chip when we open it. + */ + + return 0; +} + /* * Set or clear promiscuous/multicast mode filter for this adaptor. * @@ -718,7 +738,7 @@ /* * Don't print this message too many times... */ - if (jiffies - last_warned > 30 * HZ) { + if (time_after(jiffies, last_warned + 10 * HZ)) { last_warned = jiffies; printk("%s: memory squeeze, dropping packet.\n", dev->name); } @@ -875,6 +895,7 @@ dev->hard_start_xmit = ether3_sendpacket; dev->get_stats = ether3_getstats; dev->set_multicast_list = ether3_setmulticastlist; + dev->set_mac_address = ether3_set_mac_address; dev->tx_timeout = ether3_timeout; dev->watchdog_timeo = 5 * HZ / 100; return 0; diff -urN orig/drivers/acorn/net/etherh.c linux/drivers/acorn/net/etherh.c --- orig/drivers/acorn/net/etherh.c Mon Aug 5 13:30:41 2002 +++ linux/drivers/acorn/net/etherh.c Sun Sep 15 20:46:58 2002 @@ -1,7 +1,7 @@ /* * linux/drivers/acorn/net/etherh.c * - * Copyright (C) 2000 Russell King + * Copyright (C) 2000-2002 Russell King * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as @@ -23,6 +23,7 @@ * 12-10-1999 CK/TEW EtherM driver first release * 21-12-2000 TTC EtherH/EtherM integration * 25-12-2000 RMK 1.08 Clean integration of EtherM into this driver. + * 03-01-2002 RMK 1.09 Always enable IRQs if we're in the nic slot. */ #include @@ -64,13 +65,18 @@ { 0xffff, 0xffff } }; +struct etherh_priv { + unsigned int id; + unsigned int ctrl_port; + unsigned int ctrl; +}; MODULE_AUTHOR("Russell King"); MODULE_DESCRIPTION("EtherH/EtherM driver"); MODULE_LICENSE("GPL"); static char version[] __initdata = - "EtherH/EtherM Driver (c) 2000 Russell King v1.08\n"; + "EtherH/EtherM Driver (c) 2002 Russell King v1.09\n"; #define ETHERH500_DATAPORT 0x200 /* MEMC */ #define ETHERH500_NS8390 0x000 /* MEMC */ @@ -97,18 +103,61 @@ #define ETHERM_TX_START_PAGE 64 #define ETHERM_STOP_PAGE 127 -/* --------------------------------------------------------------------------- */ +/* ------------------------------------------------------------------------ */ + +static inline void etherh_set_ctrl(struct etherh_priv *eh, unsigned int mask) +{ + eh->ctrl |= mask; + outb(eh->ctrl, eh->ctrl_port); +} + +static inline void etherh_clr_ctrl(struct etherh_priv *eh, unsigned int mask) +{ + eh->ctrl &= ~mask; + outb(eh->ctrl, eh->ctrl_port); +} + +static inline unsigned int etherh_get_stat(struct etherh_priv *eh) +{ + return inb(eh->ctrl_port); +} + + + + +static void etherh_irq_enable(ecard_t *ec, int irqnr) +{ + struct etherh_priv *eh = ec->irq_data; + + etherh_set_ctrl(eh, ETHERH_CP_IE); +} + +static void etherh_irq_disable(ecard_t *ec, int irqnr) +{ + struct etherh_priv *eh = ec->irq_data; + + etherh_clr_ctrl(eh, ETHERH_CP_IE); +} + +static expansioncard_ops_t etherh_ops = { + irqenable: etherh_irq_enable, + irqdisable: etherh_irq_disable, +}; + + + static void etherh_setif(struct net_device *dev) { struct ei_device *ei_local = (struct ei_device *) dev->priv; + struct etherh_priv *eh = (struct etherh_priv *)dev->rmem_start; unsigned long addr, flags; save_flags_cli(flags); /* set the interface type */ - switch (dev->mem_end) { + switch (eh->id) { case PROD_I3_ETHERLAN600: case PROD_I3_ETHERLAN600A: addr = dev->base_addr + EN0_RCNTHI; @@ -124,14 +173,13 @@ break; case PROD_I3_ETHERLAN500: - addr = dev->rmem_start; - switch (dev->if_port) { case IF_PORT_10BASE2: - outb(inb(addr) & ~ETHERH_CP_IF, addr); + etherh_clr_ctrl(eh, ETHERH_CP_IF); break; + case IF_PORT_10BASET: - outb(inb(addr) | ETHERH_CP_IF, addr); + etherh_set_ctrl(eh, ETHERH_CP_IF); break; } break; @@ -147,9 +195,10 @@ etherh_getifstat(struct net_device *dev) { struct ei_device *ei_local = (struct ei_device *) dev->priv; + struct etherh_priv *eh = (struct etherh_priv *)dev->rmem_start; int stat = 0; - switch (dev->mem_end) { + switch (eh->id) { case PROD_I3_ETHERLAN600: case PROD_I3_ETHERLAN600A: switch (dev->if_port) { @@ -168,7 +217,7 @@ stat = 1; break; case IF_PORT_10BASET: - stat = inb(dev->rmem_start) & ETHERH_CP_HEARTBEAT; + stat = etherh_get_stat(eh) & ETHERH_CP_HEARTBEAT; break; } break; @@ -251,7 +300,13 @@ return; } - ei_local->dmaing |= 1; + /* + * Make sure we have a round number of bytes if we're in word mode. + */ + if (count & 1 && ei_local->word16) + count++; + + ei_local->dmaing = 1; addr = dev->base_addr; dma_addr = dev->mem_start; @@ -291,7 +346,7 @@ } outb (ENISR_RDC, addr + EN0_ISR); - ei_local->dmaing &= ~1; + ei_local->dmaing = 0; } /* @@ -311,7 +366,7 @@ return; } - ei_local->dmaing |= 1; + ei_local->dmaing = 1; addr = dev->base_addr; dma_addr = dev->mem_start; @@ -332,7 +387,7 @@ insb (dma_addr, buf, count); outb (ENISR_RDC, addr + EN0_ISR); - ei_local->dmaing &= ~1; + ei_local->dmaing = 0; } /* @@ -351,7 +406,7 @@ return; } - ei_local->dmaing |= 1; + ei_local->dmaing = 1; addr = dev->base_addr; dma_addr = dev->mem_start; @@ -369,7 +424,7 @@ insb (dma_addr, hdr, sizeof (*hdr)); outb (ENISR_RDC, addr + EN0_ISR); - ei_local->dmaing &= ~1; + ei_local->dmaing = 0; } /* @@ -385,6 +440,12 @@ { struct ei_device *ei_local = (struct ei_device *) dev->priv; + if (!is_valid_ether_addr(dev->dev_addr)) { + printk(KERN_WARNING "%s: invalid ethernet address\n", + dev->name); + return -EINVAL; + } + if (request_irq(dev->irq, ei_interrupt, 0, dev->name, dev)) return -EAGAIN; @@ -427,22 +488,22 @@ return 0; } -static void etherh_irq_enable(ecard_t *ec, int irqnr) +static int +etherh_set_mac_address(struct net_device *dev, void *p) { - unsigned int ctrl_addr = (unsigned int)ec->irq_data; - outb(inb(ctrl_addr) | ETHERH_CP_IE, ctrl_addr); -} + struct sockaddr *addr = p; -static void etherh_irq_disable(ecard_t *ec, int irqnr) -{ - unsigned int ctrl_addr = (unsigned int)ec->irq_data; - outb(inb(ctrl_addr) & ~ETHERH_CP_IE, ctrl_addr); -} + if (netif_running(dev)) + return -EBUSY; -static expansioncard_ops_t etherh_ops = { - irqenable: etherh_irq_enable, - irqdisable: etherh_irq_disable, -}; + memcpy(dev->dev_addr, addr->sa_data, dev->addr_len); + + /* + * We'll set the MAC address on the chip when we open it. + */ + + return 0; +} /* * Initialisation @@ -460,11 +521,13 @@ * Read the ethernet address string from the on board rom. * This is an ascii string... */ -static int __init etherh_addr(char *addr, struct expansion_card *ec) +static void __init etherh_addr(char *addr, struct expansion_card *ec) { struct in_chunk_dir cd; char *s; + memset(addr, 0, 6); + if (ecard_readchunk(&cd, ec, 0xf5, 0) && (s = strchr(cd.d.string, '('))) { int i; for (i = 0; i < 6; i++) { @@ -472,31 +535,25 @@ if (*s != (i == 5? ')' : ':')) break; } - if (i == 6) - return 0; } - return -ENODEV; } /* * Create an ethernet address from the system serial number. */ -static int __init etherm_addr(char *addr) +static void __init etherm_addr(char *addr) { unsigned int serial; - if (system_serial_low == 0 && system_serial_high == 0) - return -ENODEV; - serial = system_serial_low | system_serial_high; - - addr[0] = 0; - addr[1] = 0; - addr[2] = 0xa4; - addr[3] = 0x10 + (serial >> 24); - addr[4] = serial >> 16; - addr[5] = serial >> 8; - return 0; + if (serial != 0) { + addr[0] = 0; + addr[1] = 0; + addr[2] = 0xa4; + addr[3] = 0x10 + (serial >> 24); + addr[4] = serial >> 16; + addr[5] = serial >> 8; + } } static u32 etherh_regoffsets[16]; @@ -506,6 +563,7 @@ { struct ei_device *ei_local; struct net_device *dev; + struct etherh_priv *eh; const char *dev_type; int i, size; @@ -517,42 +575,50 @@ if (!dev) goto out; + eh = kmalloc(sizeof(struct etherh_priv), GFP_KERNEL); + if (!eh) + goto out_nopriv; + SET_MODULE_OWNER(dev); - dev->open = etherh_open; - dev->stop = etherh_close; - dev->set_config = etherh_set_config; - dev->irq = ec->irq; - dev->base_addr = ecard_address(ec, ECARD_MEMC, 0); - dev->mem_end = ec->cid.product; + dev->open = etherh_open; + dev->stop = etherh_close; + dev->set_mac_address = etherh_set_mac_address; + dev->set_config = etherh_set_config; + dev->irq = ec->irq; + dev->base_addr = ecard_address(ec, ECARD_MEMC, 0); + dev->rmem_start = (unsigned long)eh; + + /* + * IRQ and control port handling + */ ec->ops = ðerh_ops; + ec->irq_data = eh; + eh->ctrl = 0; + eh->id = ec->cid.product; switch (ec->cid.product) { case PROD_ANT_ETHERM: - if (etherm_addr(dev->dev_addr)) - goto free; + etherm_addr(dev->dev_addr); dev->base_addr += ETHERM_NS8390; dev->mem_start = dev->base_addr + ETHERM_DATAPORT; - ec->irq_data = (void *)(dev->base_addr + ETHERM_CTRLPORT); + eh->ctrl_port = dev->base_addr + ETHERM_CTRLPORT; break; case PROD_I3_ETHERLAN500: - if (etherh_addr(dev->dev_addr, ec)) - goto free; + etherh_addr(dev->dev_addr, ec); dev->base_addr += ETHERH500_NS8390; dev->mem_start = dev->base_addr + ETHERH500_DATAPORT; - dev->rmem_start = (unsigned long) - ec->irq_data = (void *)ecard_address (ec, ECARD_IOC, ECARD_FAST) + eh->ctrl_port = ecard_address (ec, ECARD_IOC, ECARD_FAST) + ETHERH500_CTRLPORT; break; case PROD_I3_ETHERLAN600: case PROD_I3_ETHERLAN600A: - if (etherh_addr(dev->dev_addr, ec)) - goto free; + etherh_addr(dev->dev_addr, ec); dev->base_addr += ETHERH600_NS8390; - dev->mem_start = dev->base_addr + ETHERH600_DATAPORT; - ec->irq_data = (void *)(dev->base_addr + ETHERH600_CTRLPORT); + dev->mem_start = dev->base_addr + ETHERH600_DATAPORT; + eh->ctrl_port = dev->base_addr + ETHERH600_CTRLPORT; break; default: @@ -572,6 +638,12 @@ goto release; /* + * If we're in the NIC slot, make sure the IRQ is enabled + */ + if (dev->irq == 11) + etherh_set_ctrl(eh, ETHERH_CP_IE); + + /* * Unfortunately, ethdev_init eventually calls * ether_setup, which re-writes dev->flags. */ @@ -636,6 +708,8 @@ release: release_region(dev->base_addr, 16); free: + kfree(eh); +out_nopriv: unregister_netdev(dev); kfree(dev); out: @@ -696,6 +770,7 @@ } if (e_card[i]) { e_card[i]->ops = NULL; + kfree(e_card[i]->irq_data); ecard_release(e_card[i]); e_card[i] = NULL; } diff -urN orig/drivers/acorn/scsi/arxescsi.c linux/drivers/acorn/scsi/arxescsi.c --- orig/drivers/acorn/scsi/arxescsi.c Fri Oct 26 16:46:02 2001 +++ linux/drivers/acorn/scsi/arxescsi.c Thu Jan 23 12:57:02 2003 @@ -37,16 +37,22 @@ #include "../../scsi/sd.h" #include "../../scsi/hosts.h" -#include "arxescsi.h" +#include #include "fas216.h" +#include "scsi.h" -/* Hmm - this should go somewhere else */ -#define BUS_ADDR(x) ((((unsigned long)(x)) << 2) + IO_BASE) +struct arxescsi_info { + FAS216_Info info; + struct expansion_card *ec; + unsigned int dmaarea; /* Pseudo DMA area */ +}; -/* Configuration */ -#define ARXESCSI_XTALFREQ 24 -#define ARXESCSI_ASYNC_PERIOD 200 -#define ARXESCSI_SYNC_DEPTH 0 +#define DMADATA_OFFSET (0x200/4) + +#define DMASTAT_OFFSET (0x600/4) +#define DMASTAT_DRQ (1 << 0) + +#define CSTATUS_IRQ (1 << 0) /* * List of devices that the driver will recognise @@ -56,11 +62,7 @@ /* * Version */ -#define VER_MAJOR 0 -#define VER_MINOR 1 -#define VER_PATCH 1 - -static struct expansion_card *ecs[MAX_ECARDS]; +#define VERSION "1.10 (22/01/2003 2.4.19-rmk5)" /* * Function: int arxescsi_dma_setup(host, SCpnt, direction, min_type) @@ -157,7 +159,7 @@ void arxescsi_dma_pseudo(struct Scsi_Host *host, Scsi_Pointer *SCp, fasdmadir_t direction, int transfer) { - ARXEScsi_Info *info = (ARXEScsi_Info *)host->hostdata; + struct arxescsi_info *info = (struct arxescsi_info *)host->hostdata; unsigned int length, io, error=0; unsigned char *addr; @@ -182,7 +184,7 @@ if (getb(io, 4) & STAT_INT) break; - if (!(getb(io, 48) & CSTATUS_IRQ)) + if (!(getb(io, 48) & DMASTAT_DRQ)) continue; word = *addr | *(addr + 1) << 8; @@ -205,7 +207,7 @@ break; } - if (!(getb(io, 48) & CSTATUS_IRQ)) + if (!(getb(io, 48) & DMASTAT_DRQ)) continue; insw(info->dmaarea, addr, 256 >> 1); @@ -221,7 +223,7 @@ if (getb(io, 4) & STAT_INT) break; - if (!(getb(io, 48) & CSTATUS_IRQ)) + if (!(getb(io, 48) & DMASTAT_DRQ)) continue; word = getw(io, 16); @@ -248,96 +250,6 @@ } /* - * Function: int arxescsi_detect(Scsi_Host_Template * tpnt) - * Purpose : initialises ARXE SCSI driver - * Params : tpnt - template for this SCSI adapter - * Returns : >0 if host found, 0 otherwise. - */ -int arxescsi_detect(Scsi_Host_Template *tpnt) -{ - static const card_ids arxescsi_cids[] = { ARXESCSI_LIST, { 0xffff, 0xffff} }; - int count = 0; - struct Scsi_Host *host; - - tpnt->proc_name = "arxescsi"; - memset(ecs, 0, sizeof (ecs)); - - ecard_startfind(); - - while (1) { - ARXEScsi_Info *info; - - ecs[count] = ecard_find(0, arxescsi_cids); - if (!ecs[count]) - break; - - ecard_claim(ecs[count]); - - host = scsi_register(tpnt, sizeof (ARXEScsi_Info)); - if (!host) { - ecard_release(ecs[count]); - break; - } - - host->io_port = ecard_address(ecs[count], ECARD_MEMC, 0) + 0x0800; - host->irq = NO_IRQ; - host->dma_channel = NO_DMA; - host->can_queue = 0; /* no command queueing */ - info = (ARXEScsi_Info *)host->hostdata; - - info->info.scsi.io_port = host->io_port; - info->info.scsi.irq = host->irq; - info->info.scsi.io_shift = 3; - info->info.ifcfg.clockrate = ARXESCSI_XTALFREQ; - info->info.ifcfg.select_timeout = 255; - info->info.ifcfg.asyncperiod = ARXESCSI_ASYNC_PERIOD; - info->info.ifcfg.sync_max_depth = ARXESCSI_SYNC_DEPTH; - info->info.ifcfg.cntl3 = CNTL3_FASTSCSI | CNTL3_FASTCLK; - info->info.ifcfg.disconnect_ok = 0; - info->info.ifcfg.wide_max_size = 0; - info->info.dma.setup = arxescsi_dma_setup; - info->info.dma.pseudo = arxescsi_dma_pseudo; - info->info.dma.stop = arxescsi_dma_stop; - info->dmaarea = host->io_port + 128; - info->cstatus = host->io_port + 384; - - ecs[count]->irqaddr = (unsigned char *)BUS_ADDR(host->io_port); - ecs[count]->irqmask = CSTATUS_IRQ; - - request_region(host->io_port , 120, "arxescsi-fas"); - request_region(host->io_port + 128, 384, "arxescsi-dma"); - - printk("scsi%d: Has no interrupts - using polling mode\n", - host->host_no); - - fas216_init(host); - ++count; - } - return count; -} - -/* - * Function: int arxescsi_release(struct Scsi_Host * host) - * Purpose : releases all resources used by this adapter - * Params : host - driver host structure to return info for. - * Returns : nothing - */ -int arxescsi_release(struct Scsi_Host *host) -{ - int i; - - fas216_release(host); - - release_region(host->io_port, 120); - release_region(host->io_port + 128, 384); - - for (i = 0; i < MAX_ECARDS; i++) - if (ecs[i] && host->io_port == (ecard_address(ecs[i], ECARD_MEMC, 0) + 0x0800)) - ecard_release(ecs[i]); - return 0; -} - -/* * Function: const char *arxescsi_info(struct Scsi_Host * host) * Purpose : returns a descriptive string about this interface, * Params : host - driver host structure to return info for. @@ -345,13 +257,12 @@ */ const char *arxescsi_info(struct Scsi_Host *host) { - ARXEScsi_Info *info = (ARXEScsi_Info *)host->hostdata; - static char string[100], *p; + struct arxescsi_info *info = (struct arxescsi_info *)host->hostdata; + static char string[150]; - p = string; - p += sprintf(p, "%s ", host->hostt->name); - p += fas216_info(&info->info, p); - p += sprintf(p, "v%d.%d.%d", VER_MAJOR, VER_MINOR, VER_PATCH); + sprintf(string, "%s (%s) in slot %d v%s", + host->hostt->name, info->info.scsi.type, info->ec->slot_no, + VERSION); return string; } @@ -374,26 +285,20 @@ int length, int host_no, int inout) { int pos, begin; - struct Scsi_Host *host = scsi_hostlist; - ARXEScsi_Info *info; + struct Scsi_Host *host; + struct arxescsi_info *info; Scsi_Device *scd; - while (host) { - if (host->host_no == host_no) - break; - host = host->next; - } + host = scsi_host_hn_get(host_no); if (!host) return 0; - info = (ARXEScsi_Info *)host->hostdata; + info = (struct arxescsi_info *)host->hostdata; if (inout == 1) return -EINVAL; begin = 0; - pos = sprintf(buffer, - "ARXE 16-bit SCSI driver version %d.%d.%d\n", - VER_MAJOR, VER_MINOR, VER_PATCH); + pos = sprintf(buffer, "ARXE 16-bit SCSI driver v%s\n", VERSION); pos += fas216_print_host(&info->info, buffer + pos); pos += fas216_print_stats(&info->info, buffer + pos); @@ -418,11 +323,156 @@ return pos; } -static Scsi_Host_Template arxescsi_template = ARXEScsi; +static int arxescsi_probe(struct expansion_card *ec); + +/* + * Function: int arxescsi_detect(Scsi_Host_Template * tpnt) + * Purpose : initialises ARXE SCSI driver + * Params : tpnt - template for this SCSI adapter + * Returns : >0 if host found, 0 otherwise. + */ +static int arxescsi_detect(Scsi_Host_Template *tpnt) +{ + static const card_ids arxescsi_cids[] = { ARXESCSI_LIST, { 0xffff, 0xffff} }; + int count = 0, ret; + + ecard_startfind(); + + while (1) { + struct expansion_card *ec; + ec = ecard_find(0, arxescsi_cids); + if (!ec) + break; + + ecard_claim(ec); + ret = arxescsi_probe(ec); + if (ret) { + ecard_release(ec); + break; + } + ++count; + } + return count; +} + +static void arxescsi_remove(struct Scsi_Host *host); + +/* + * Function: int arxescsi_release(struct Scsi_Host * host) + * Purpose : releases all resources used by this adapter + * Params : host - driver host structure to return info for. + * Returns : nothing + */ +static int arxescsi_release(struct Scsi_Host *host) +{ + arxescsi_remove(host); + return 0; +} + +static Scsi_Host_Template arxescsi_template = { + .module = THIS_MODULE, + .proc_info = arxescsi_proc_info, + .name = "ARXE SCSI card", + .detect = arxescsi_detect, + .release = arxescsi_release, + .info = arxescsi_info, + .bios_param = scsicam_bios_param, + .command = fas216_command, + .queuecommand = fas216_queue_command, + .eh_host_reset_handler = fas216_eh_host_reset, + .eh_bus_reset_handler = fas216_eh_bus_reset, + .eh_device_reset_handler = fas216_eh_device_reset, + .eh_abort_handler = fas216_eh_abort, + .use_new_eh_code = 1, + + .can_queue = 0, + .this_id = 7, + .sg_tablesize = SG_ALL, + .cmd_per_lun = 1, + .use_clustering = DISABLE_CLUSTERING, + .proc_name = "arxescsi", +}; + +static int +arxescsi_probe(struct expansion_card *ec) +{ + struct Scsi_Host *host; + struct arxescsi_info *info; + unsigned long base; + int ret; + + base = ecard_address(ec, ECARD_MEMC, 0) + 0x0800; + + if (!request_region(base, 512, "arxescsi")) { + ret = -EBUSY; + goto out; + } + + host = scsi_register(&arxescsi_template, sizeof(struct arxescsi_info)); + if (!host) { + ret = -ENOMEM; + goto out_region; + } + + host->io_port = base; + host->irq = NO_IRQ; + host->dma_channel = NO_DMA; + + info = (struct arxescsi_info *)host->hostdata; + info->ec = ec; + info->dmaarea = base + DMADATA_OFFSET; + + info->info.scsi.io_port = host->io_port; + info->info.scsi.irq = NO_IRQ; + info->info.scsi.io_shift = 3; + info->info.ifcfg.clockrate = 24; /* MHz */ + info->info.ifcfg.select_timeout = 255; + info->info.ifcfg.asyncperiod = 200; /* ns */ + info->info.ifcfg.sync_max_depth = 0; + info->info.ifcfg.cntl3 = CNTL3_FASTSCSI | CNTL3_FASTCLK; + info->info.ifcfg.disconnect_ok = 0; + info->info.ifcfg.wide_max_size = 0; + info->info.ifcfg.capabilities = FASCAP_PSEUDODMA; + info->info.dma.setup = arxescsi_dma_setup; + info->info.dma.pseudo = arxescsi_dma_pseudo; + info->info.dma.stop = arxescsi_dma_stop; + + ec->irqaddr = (unsigned char *)ioaddr(base); + ec->irqmask = CSTATUS_IRQ; + + ret = fas216_init(host); + if (ret) + goto out_unregister; + + ret = fas216_add(host); + if (ret == 0) + goto out; + + fas216_release(host); + out_unregister: + scsi_unregister(host); + out_region: + release_region(base, 512); + out: + return ret; +} + +static void arxescsi_remove(struct Scsi_Host *host) +{ + struct arxescsi_info *info; + + info = (struct arxescsi_info *)host->hostdata; + + fas216_remove(host); + + release_region(host->io_port, 512); + + ecard_release(info->ec); + fas216_release(host); +} static int __init init_arxe_scsi_driver(void) { - arxescsi_template.module = THIS_MODULE; scsi_register_module(MODULE_SCSI_HA, &arxescsi_template); if (arxescsi_template.present) return 0; diff -urN orig/drivers/acorn/scsi/arxescsi.h linux/drivers/acorn/scsi/arxescsi.h --- orig/drivers/acorn/scsi/arxescsi.h Sat May 13 01:01:39 2000 +++ linux/drivers/acorn/scsi/arxescsi.h Thu Jan 1 01:00:00 1970 @@ -1,82 +0,0 @@ -/* - * ARXE SCSI card driver - * - * Copyright (C) 1997-2000 Russell King - * - * Changes to support ARXE 16-bit SCSI card by Stefan Hanske - */ -#ifndef ARXE_SCSI_H -#define ARXE_SCSI_H - -#define MANU_ARXE 0x0041 -#define PROD_ARXE_SCSI 0x00be - -extern int arxescsi_detect (Scsi_Host_Template *); -extern int arxescsi_release (struct Scsi_Host *); -extern const char *arxescsi_info (struct Scsi_Host *); -extern int arxescsi_proc_info (char *buffer, char **start, off_t offset, - int length, int hostno, int inout); - -#ifndef NULL -#define NULL ((void *)0) -#endif - -#ifndef CAN_QUEUE -/* - * Default queue size - */ -#define CAN_QUEUE 1 -#endif - -#ifndef CMD_PER_LUN -#define CMD_PER_LUN 1 -#endif - -#ifndef SCSI_ID -/* - * Default SCSI host ID - */ -#define SCSI_ID 7 -#endif - -#include - -#include "fas216.h" - -#define ARXEScsi { \ -proc_info: arxescsi_proc_info, \ -name: "ARXE SCSI card", \ -detect: arxescsi_detect, \ -release: arxescsi_release, \ -info: arxescsi_info, \ -bios_param: scsicam_bios_param, \ -can_queue: CAN_QUEUE, \ -this_id: SCSI_ID, \ -sg_tablesize: SG_ALL, \ -cmd_per_lun: CMD_PER_LUN, \ -use_clustering: DISABLE_CLUSTERING, \ -command: fas216_command, \ -queuecommand: fas216_queue_command, \ -eh_host_reset_handler: fas216_eh_host_reset, \ -eh_bus_reset_handler: fas216_eh_bus_reset, \ -eh_device_reset_handler: fas216_eh_device_reset, \ -eh_abort_handler: fas216_eh_abort, \ -use_new_eh_code: 1 \ - } - -#ifndef HOSTS_C - -typedef struct { - FAS216_Info info; - - /* other info... */ - unsigned int cstatus; /* card status register */ - unsigned int dmaarea; /* Pseudo DMA area */ -} ARXEScsi_Info; - -#define CSTATUS_IRQ (1 << 0) -#define CSTATUS_DRQ (1 << 0) - -#endif /* HOSTS_C */ - -#endif /* ARXE_SCSI_H */ diff -urN orig/drivers/acorn/scsi/cumana_2.c linux/drivers/acorn/scsi/cumana_2.c --- orig/drivers/acorn/scsi/cumana_2.c Fri Nov 16 10:09:50 2001 +++ linux/drivers/acorn/scsi/cumana_2.c Fri Feb 21 15:44:55 2003 @@ -1,7 +1,7 @@ /* * linux/drivers/acorn/scsi/cumana_2.c * - * Copyright (C) 1997-2000 Russell King + * Copyright (C) 1997-2002 Russell King * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as @@ -23,8 +23,6 @@ #include #include #include -#include -#include #include #include #include @@ -38,14 +36,10 @@ #include "../../scsi/sd.h" #include "../../scsi/hosts.h" #include "fas216.h" +#include "scsi.h" #include -/* Configuration */ -#define CUMANASCSI2_XTALFREQ 40 -#define CUMANASCSI2_ASYNC_PERIOD 200 -#define CUMANASCSI2_SYNC_DEPTH 7 - /* * List of devices that the driver will recognise */ @@ -76,11 +70,7 @@ /* * Version */ -#define VER_MAJOR 0 -#define VER_MINOR 0 -#define VER_PATCH 4 - -static struct expansion_card *ecs[MAX_ECARDS]; +#define VERSION "1.00 (13/11/2002 2.4.19-rmk5)" /* * Use term=0,1,0,0,0 to turn terminators on/off @@ -89,16 +79,16 @@ #define NR_SG 256 -typedef struct { - FAS216_Info info; - - /* other info... */ - unsigned int status; /* card status register */ - unsigned int alatch; /* Control register */ - unsigned int terms; /* Terminator state */ - unsigned int dmaarea; /* Pseudo DMA area */ - struct scatterlist sg[NR_SG]; /* Scatter DMA list */ -} CumanaScsi2_Info; +struct cumanascsi2_info { + FAS216_Info info; + struct expansion_card *ec; + + unsigned int status; /* card status register */ + unsigned int alatch; /* Control register */ + unsigned int terms; /* Terminator state */ + unsigned int dmaarea; /* Pseudo DMA area */ + struct scatterlist sg[NR_SG]; /* Scatter DMA list */ +}; #define CSTATUS_IRQ (1 << 0) #define CSTATUS_DRQ (1 << 1) @@ -128,12 +118,8 @@ } static const expansioncard_ops_t cumanascsi_2_ops = { - cumanascsi_2_irqenable, - cumanascsi_2_irqdisable, - NULL, - NULL, - NULL, - NULL + .irqenable = cumanascsi_2_irqenable, + .irqdisable = cumanascsi_2_irqdisable, }; /* Prototype: void cumanascsi_2_terminator_ctl(host, on_off) @@ -144,7 +130,7 @@ static void cumanascsi_2_terminator_ctl(struct Scsi_Host *host, int on_off) { - CumanaScsi2_Info *info = (CumanaScsi2_Info *)host->hostdata; + struct cumanascsi2_info *info = (struct cumanascsi2_info *)host->hostdata; if (on_off) { info->terms = 1; @@ -181,36 +167,30 @@ cumanascsi_2_dma_setup(struct Scsi_Host *host, Scsi_Pointer *SCp, fasdmadir_t direction, fasdmatype_t min_type) { - CumanaScsi2_Info *info = (CumanaScsi2_Info *)host->hostdata; + struct cumanascsi2_info *info = (struct cumanascsi2_info *)host->hostdata; int dmach = host->dma_channel; outb(ALATCH_DIS_DMA, info->alatch); if (dmach != NO_DMA && (min_type == fasdma_real_all || SCp->this_residual >= 512)) { - int bufs = SCp->buffers_residual; - int pci_dir, dma_dir, alatch_dir; + int bufs, map_dir, dma_dir, alatch_dir; - if (bufs) - memcpy(info->sg + 1, SCp->buffer + 1, - sizeof(struct scatterlist) * bufs); - info->sg[0].address = SCp->ptr; - info->sg[0].page = NULL; - info->sg[0].length = SCp->this_residual; + bufs = copy_SCp_to_sg(&info->sg[0], SCp, NR_SG); if (direction == DMA_OUT) - pci_dir = PCI_DMA_TODEVICE, + map_dir = PCI_DMA_TODEVICE, dma_dir = DMA_MODE_WRITE, alatch_dir = ALATCH_DMA_OUT; else - pci_dir = PCI_DMA_FROMDEVICE, + map_dir = PCI_DMA_FROMDEVICE, dma_dir = DMA_MODE_READ, alatch_dir = ALATCH_DMA_IN; - pci_map_sg(NULL, info->sg, bufs + 1, pci_dir); + pci_map_sg(NULL, info->sg, bufs, map_dir); disable_dma(dmach); - set_dma_sg(dmach, info->sg, bufs + 1); + set_dma_sg(dmach, info->sg, bufs); outb(alatch_dir, info->alatch); set_dma_mode(dmach, dma_dir); enable_dma(dmach); @@ -238,7 +218,7 @@ cumanascsi_2_dma_pseudo(struct Scsi_Host *host, Scsi_Pointer *SCp, fasdmadir_t direction, int transfer) { - CumanaScsi2_Info *info = (CumanaScsi2_Info *)host->hostdata; + struct cumanascsi2_info *info = (struct cumanascsi2_info *)host->hostdata; unsigned int length; unsigned char *addr; @@ -271,7 +251,7 @@ unsigned int status = inb(info->status); if (status & STATUS_INT) - goto end; + return; if (!(status & STATUS_DRQ)) continue; @@ -287,7 +267,7 @@ unsigned int status = inb(info->status); if (status & STATUS_INT) - goto end; + return; if (!(status & STATUS_DRQ)) continue; @@ -300,8 +280,6 @@ } } } - -end: } /* Prototype: int cumanascsi_2_dma_stop(host, SCpnt) @@ -312,124 +290,13 @@ static void cumanascsi_2_dma_stop(struct Scsi_Host *host, Scsi_Pointer *SCp) { - CumanaScsi2_Info *info = (CumanaScsi2_Info *)host->hostdata; + struct cumanascsi2_info *info = (struct cumanascsi2_info *)host->hostdata; if (host->dma_channel != NO_DMA) { outb(ALATCH_DIS_DMA, info->alatch); disable_dma(host->dma_channel); } } -/* Prototype: int cumanascsi_2_detect(Scsi_Host_Template * tpnt) - * Purpose : initialises Cumana SCSI 2 driver - * Params : tpnt - template for this SCSI adapter - * Returns : >0 if host found, 0 otherwise. - */ -int -cumanascsi_2_detect(Scsi_Host_Template *tpnt) -{ - static const card_ids cumanascsi_2_cids[] = - { CUMANASCSI2_LIST, { 0xffff, 0xffff} }; - int count = 0; - struct Scsi_Host *host; - - tpnt->proc_name = "cumanascs2"; - memset(ecs, 0, sizeof (ecs)); - - ecard_startfind(); - - while (1) { - CumanaScsi2_Info *info; - - ecs[count] = ecard_find(0, cumanascsi_2_cids); - if (!ecs[count]) - break; - - ecard_claim(ecs[count]); - - host = scsi_register(tpnt, sizeof (CumanaScsi2_Info)); - if (!host) { - ecard_release(ecs[count]); - break; - } - - host->io_port = ecard_address(ecs[count], ECARD_MEMC, 0); - host->irq = ecs[count]->irq; - host->dma_channel = ecs[count]->dma; - info = (CumanaScsi2_Info *)host->hostdata; - - info->terms = term[count] ? 1 : 0; - cumanascsi_2_terminator_ctl(host, info->terms); - - info->info.scsi.io_port = host->io_port + CUMANASCSI2_FAS216_OFFSET; - info->info.scsi.io_shift = CUMANASCSI2_FAS216_SHIFT; - info->info.scsi.irq = host->irq; - info->info.ifcfg.clockrate = CUMANASCSI2_XTALFREQ; - info->info.ifcfg.select_timeout = 255; - info->info.ifcfg.asyncperiod = CUMANASCSI2_ASYNC_PERIOD; - info->info.ifcfg.sync_max_depth = CUMANASCSI2_SYNC_DEPTH; - info->info.ifcfg.cntl3 = CNTL3_BS8 | CNTL3_FASTSCSI | CNTL3_FASTCLK; - info->info.ifcfg.disconnect_ok = 1; - info->info.ifcfg.wide_max_size = 0; - info->info.dma.setup = cumanascsi_2_dma_setup; - info->info.dma.pseudo = cumanascsi_2_dma_pseudo; - info->info.dma.stop = cumanascsi_2_dma_stop; - info->dmaarea = host->io_port + CUMANASCSI2_PSEUDODMA; - info->status = host->io_port + CUMANASCSI2_STATUS; - info->alatch = host->io_port + CUMANASCSI2_ALATCH; - - ecs[count]->irqaddr = (unsigned char *)ioaddr(info->status); - ecs[count]->irqmask = STATUS_INT; - ecs[count]->irq_data = (void *)info->alatch; - ecs[count]->ops = (expansioncard_ops_t *)&cumanascsi_2_ops; - - request_region(host->io_port + CUMANASCSI2_FAS216_OFFSET, - 16 << CUMANASCSI2_FAS216_SHIFT, "cumanascsi2-fas"); - - if (host->irq != NO_IRQ && - request_irq(host->irq, cumanascsi_2_intr, - SA_INTERRUPT, "cumanascsi2", host)) { - printk("scsi%d: IRQ%d not free, interrupts disabled\n", - host->host_no, host->irq); - host->irq = NO_IRQ; - info->info.scsi.irq = NO_IRQ; - } - - if (host->dma_channel != NO_DMA && - request_dma(host->dma_channel, "cumanascsi2")) { - printk("scsi%d: DMA%d not free, DMA disabled\n", - host->host_no, host->dma_channel); - host->dma_channel = NO_DMA; - } - - fas216_init(host); - ++count; - } - return count; -} - -/* Prototype: int cumanascsi_2_release(struct Scsi_Host * host) - * Purpose : releases all resources used by this adapter - * Params : host - driver host structure to return info for. - */ -int cumanascsi_2_release(struct Scsi_Host *host) -{ - int i; - - fas216_release(host); - - if (host->irq != NO_IRQ) - free_irq(host->irq, host); - if (host->dma_channel != NO_DMA) - free_dma(host->dma_channel); - release_region(host->io_port + CUMANASCSI2_FAS216_OFFSET, - 16 << CUMANASCSI2_FAS216_SHIFT); - - for (i = 0; i < MAX_ECARDS; i++) - if (ecs[i] && host->io_port == ecard_address (ecs[i], ECARD_MEMC, 0)) - ecard_release (ecs[i]); - return 0; -} - /* Prototype: const char *cumanascsi_2_info(struct Scsi_Host * host) * Purpose : returns a descriptive string about this interface, * Params : host - driver host structure to return info for. @@ -437,15 +304,12 @@ */ const char *cumanascsi_2_info(struct Scsi_Host *host) { - CumanaScsi2_Info *info = (CumanaScsi2_Info *)host->hostdata; - static char string[100], *p; + struct cumanascsi2_info *info = (struct cumanascsi2_info *)host->hostdata; + static char string[150]; - p = string; - p += sprintf(p, "%s ", host->hostt->name); - p += fas216_info(&info->info, p); - p += sprintf(p, "v%d.%d.%d terminators o%s", - VER_MAJOR, VER_MINOR, VER_PATCH, - info->terms ? "n" : "ff"); + sprintf(string, "%s (%s) in slot %d v%s terminators o%s", + host->hostt->name, info->info.scsi.type, info->ec->slot_no, + VERSION, info->terms ? "n" : "ff"); return string; } @@ -498,28 +362,21 @@ int length, int host_no, int inout) { int pos, begin; - struct Scsi_Host *host = scsi_hostlist; - CumanaScsi2_Info *info; + struct Scsi_Host *host; + struct cumanascsi2_info *info; Scsi_Device *scd; - while (host) { - if (host->host_no == host_no) - break; - host = host->next; - } + host = scsi_host_hn_get(host_no); if (!host) return 0; if (inout == 1) return cumanascsi_2_set_proc_info(host, buffer, length); - info = (CumanaScsi2_Info *)host->hostdata; + info = (struct cumanascsi2_info *)host->hostdata; begin = 0; - pos = sprintf(buffer, - "Cumana SCSI II driver version %d.%d.%d\n", - VER_MAJOR, VER_MINOR, VER_PATCH); - + pos = sprintf(buffer, "Cumana SCSI II driver v%s\n", VERSION); pos += fas216_print_host(&info->info, buffer + pos); pos += sprintf(buffer + pos, "Term : o%s\n", info->terms ? "n" : "ff"); @@ -556,27 +413,193 @@ return pos; } +static int cumanascsi2_probe(struct expansion_card *ec); + +/* Prototype: int cumanascsi_2_detect(Scsi_Host_Template * tpnt) + * Purpose : initialises Cumana SCSI 2 driver + * Params : tpnt - template for this SCSI adapter + * Returns : >0 if host found, 0 otherwise. + */ +static int +cumanascsi_2_detect(Scsi_Host_Template *tpnt) +{ + static const card_ids cumanascsi_2_cids[] = + { CUMANASCSI2_LIST, { 0xffff, 0xffff} }; + int count = 0, ret; + + ecard_startfind(); + + while (1) { + struct expansion_card *ec; + + ec = ecard_find(0, cumanascsi_2_cids); + if (!ec) + break; + + ecard_claim(ec); + ret = cumanascsi2_probe(ec); + if (ret) { + ecard_release(ec); + break; + } + + ++count; + } + return count; +} + +static void cumanascsi2_remove(struct Scsi_Host *host); + +/* Prototype: int cumanascsi_2_release(struct Scsi_Host * host) + * Purpose : releases all resources used by this adapter + * Params : host - driver host structure to return info for. + */ +static int cumanascsi_2_release(struct Scsi_Host *host) +{ + cumanascsi2_remove(host); + return 0; +} + static Scsi_Host_Template cumanascsi2_template = { - module: THIS_MODULE, - proc_info: cumanascsi_2_proc_info, - name: "Cumana SCSI II", - detect: cumanascsi_2_detect, - release: cumanascsi_2_release, - info: cumanascsi_2_info, - bios_param: scsicam_bios_param, - can_queue: 1, - this_id: 7, - sg_tablesize: SG_ALL, - cmd_per_lun: 1, - use_clustering: DISABLE_CLUSTERING, - command: fas216_command, - queuecommand: fas216_queue_command, - eh_host_reset_handler: fas216_eh_host_reset, - eh_bus_reset_handler: fas216_eh_bus_reset, - eh_device_reset_handler: fas216_eh_device_reset, - eh_abort_handler: fas216_eh_abort, - use_new_eh_code: 1 + .module = THIS_MODULE, + .proc_info = cumanascsi_2_proc_info, + .name = "Cumana SCSI II", + .detect = cumanascsi_2_detect, + .release = cumanascsi_2_release, + .info = cumanascsi_2_info, + .bios_param = scsicam_bios_param, + .command = fas216_command, + .queuecommand = fas216_queue_command, + .eh_host_reset_handler = fas216_eh_host_reset, + .eh_bus_reset_handler = fas216_eh_bus_reset, + .eh_device_reset_handler = fas216_eh_device_reset, + .eh_abort_handler = fas216_eh_abort, + .use_new_eh_code = 1, + + .can_queue = 1, + .this_id = 7, + .sg_tablesize = SG_ALL, + .cmd_per_lun = 1, + .use_clustering = DISABLE_CLUSTERING, + .proc_name = "cumanascsi2", }; + +static int +cumanascsi2_probe(struct expansion_card *ec) +{ + struct Scsi_Host *host; + struct cumanascsi2_info *info; + unsigned long base; + int ret; + + base = ecard_address(ec, ECARD_MEMC, 0); + + if (request_region(base + CUMANASCSI2_FAS216_OFFSET, + 16 << CUMANASCSI2_FAS216_SHIFT, "cumanascsi2-fas")) { + ret = -EBUSY; + goto out; + } + + host = scsi_register(&cumanascsi2_template, + sizeof(struct cumanascsi2_info)); + if (!host) { + ret = -ENOMEM; + goto out_region; + } + + host->io_port = base; + host->irq = ec->irq; + host->dma_channel = ec->dma; + + info = (struct cumanascsi2_info *)host->hostdata; + info->ec = ec; + info->dmaarea = base + CUMANASCSI2_PSEUDODMA; + info->status = base + CUMANASCSI2_STATUS; + info->alatch = base + CUMANASCSI2_ALATCH; + + ec->irqaddr = (unsigned char *)ioaddr(info->status); + ec->irqmask = STATUS_INT; + ec->irq_data = (void *)base + CUMANASCSI2_ALATCH; + ec->ops = (expansioncard_ops_t *)&cumanascsi_2_ops; + + cumanascsi_2_terminator_ctl(host, term[ec->slot_no]); + + info->info.scsi.io_port = base + CUMANASCSI2_FAS216_OFFSET; + info->info.scsi.io_shift = CUMANASCSI2_FAS216_SHIFT; + info->info.scsi.irq = host->irq; + info->info.ifcfg.clockrate = 40; /* MHz */ + info->info.ifcfg.select_timeout = 255; + info->info.ifcfg.asyncperiod = 200; /* ns */ + info->info.ifcfg.sync_max_depth = 7; + info->info.ifcfg.cntl3 = CNTL3_BS8 | CNTL3_FASTSCSI | CNTL3_FASTCLK; + info->info.ifcfg.disconnect_ok = 1; + info->info.ifcfg.wide_max_size = 0; + info->info.ifcfg.capabilities = 0; + info->info.dma.setup = cumanascsi_2_dma_setup; + info->info.dma.pseudo = cumanascsi_2_dma_pseudo; + info->info.dma.stop = cumanascsi_2_dma_stop; + + ret = fas216_init(host); + if (ret) + goto out_free; + + ret = request_irq(host->irq, cumanascsi_2_intr, + SA_INTERRUPT, "cumanascsi2", host); + if (ret) { + printk("scsi%d: IRQ%d not free: %d\n", + host->host_no, host->irq, ret); + goto out_release; + } + + if (host->dma_channel != NO_DMA) { + if (request_dma(host->dma_channel, "cumanascsi2")) { + printk("scsi%d: DMA%d not free, using PIO\n", + host->host_no, host->dma_channel); + host->dma_channel = NO_DMA; + } else { + set_dma_speed(host->dma_channel, 180); + info->info.ifcfg.capabilities |= FASCAP_DMA; + } + } + + ret = fas216_add(host); + if (ret == 0) + goto out; + + if (host->dma_channel != NO_DMA) + free_dma(host->dma_channel); + free_irq(host->irq, host); + + out_release: + fas216_release(host); + + out_free: + scsi_unregister(host); + + out_region: + release_region(base + CUMANASCSI2_FAS216_OFFSET, + 16 << CUMANASCSI2_FAS216_SHIFT); + + out: + return ret; +} + +static void cumanascsi2_remove(struct Scsi_Host *host) +{ + struct cumanascsi2_info *info = (struct cumanascsi2_info *)host->hostdata; + + fas216_remove(host); + + if (host->dma_channel != NO_DMA) + free_dma(host->dma_channel); + free_irq(host->irq, host); + + release_region(host->io_port + CUMANASCSI2_FAS216_OFFSET, + 16 << CUMANASCSI2_FAS216_SHIFT); + + fas216_release(host); + ecard_release(info->ec); +} static int __init cumanascsi2_init(void) { diff -urN orig/drivers/acorn/scsi/eesox.c linux/drivers/acorn/scsi/eesox.c --- orig/drivers/acorn/scsi/eesox.c Fri Nov 16 10:09:50 2001 +++ linux/drivers/acorn/scsi/eesox.c Sun Feb 16 23:31:45 2003 @@ -1,7 +1,7 @@ /* * linux/drivers/acorn/scsi/eesox.c * - * Copyright (C) 1997-2000 Russell King + * Copyright (C) 1997-2003 Russell King * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as @@ -29,8 +29,6 @@ #include #include #include -#include -#include #include #include #include @@ -44,14 +42,10 @@ #include "../../scsi/sd.h" #include "../../scsi/hosts.h" #include "fas216.h" +#include "scsi.h" #include -/* Configuration */ -#define EESOX_XTALFREQ 40 -#define EESOX_ASYNC_PERIOD 200 -#define EESOX_SYNC_DEPTH 7 - /* * List of devices that the driver will recognise */ @@ -60,7 +54,7 @@ #define EESOX_FAS216_OFFSET 0xc00 #define EESOX_FAS216_SHIFT 3 -#define EESOX_STATUS 0xa00 +#define EESOX_DMASTAT 0xa00 #define EESOX_STAT_INTR 0x01 #define EESOX_STAT_DMA 0x02 @@ -69,16 +63,12 @@ #define EESOX_TERM_ENABLE 0x02 #define EESOX_RESET 0x01 -#define EESOX_DMA_OFFSET 0xe00 +#define EESOX_DMADATA 0xe00 /* * Version */ -#define VER_MAJOR 0 -#define VER_MINOR 0 -#define VER_PATCH 3 - -static struct expansion_card *ecs[MAX_ECARDS]; +#define VERSION "1.10 (22/01/2003 2.4.19-rmk5)" /* * Use term=0,1,0,0,0 to turn terminators on/off @@ -87,20 +77,15 @@ #define NR_SG 256 -struct control { - unsigned int io_port; - unsigned int control; +struct eesoxscsi_info { + FAS216_Info info; + struct expansion_card *ec; + + unsigned int ctl_port; + unsigned int control; + struct scatterlist sg[NR_SG]; /* Scatter DMA list */ }; -typedef struct { - FAS216_Info info; - - struct control control; - - unsigned int dmaarea; /* Pseudo DMA area */ - struct scatterlist sg[NR_SG]; /* Scatter DMA list */ -} EESOXScsi_Info; - /* Prototype: void eesoxscsi_irqenable(ec, irqnr) * Purpose : Enable interrupts on EESOX SCSI card * Params : ec - expansion card structure @@ -109,11 +94,11 @@ static void eesoxscsi_irqenable(struct expansion_card *ec, int irqnr) { - struct control *control = (struct control *)ec->irq_data; + struct eesoxscsi_info *info = (struct eesoxscsi_info *)ec->irq_data; - control->control |= EESOX_INTR_ENABLE; + info->control |= EESOX_INTR_ENABLE; - outb(control->control, control->io_port); + outb(info->control, info->ctl_port); } /* Prototype: void eesoxscsi_irqdisable(ec, irqnr) @@ -124,20 +109,16 @@ static void eesoxscsi_irqdisable(struct expansion_card *ec, int irqnr) { - struct control *control = (struct control *)ec->irq_data; + struct eesoxscsi_info *info = (struct eesoxscsi_info *)ec->irq_data; - control->control &= ~EESOX_INTR_ENABLE; + info->control &= ~EESOX_INTR_ENABLE; - outb(control->control, control->io_port); + outb(info->control, info->ctl_port); } static const expansioncard_ops_t eesoxscsi_ops = { - eesoxscsi_irqenable, - eesoxscsi_irqdisable, - NULL, - NULL, - NULL, - NULL + .irqenable = eesoxscsi_irqenable, + .irqdisable = eesoxscsi_irqdisable, }; /* Prototype: void eesoxscsi_terminator_ctl(*host, on_off) @@ -148,17 +129,17 @@ static void eesoxscsi_terminator_ctl(struct Scsi_Host *host, int on_off) { - EESOXScsi_Info *info = (EESOXScsi_Info *)host->hostdata; + struct eesoxscsi_info *info = (struct eesoxscsi_info *)host->hostdata; unsigned long flags; save_flags_cli(flags); if (on_off) - info->control.control |= EESOX_TERM_ENABLE; + info->control |= EESOX_TERM_ENABLE; else - info->control.control &= ~EESOX_TERM_ENABLE; - restore_flags(flags); + info->control &= ~EESOX_TERM_ENABLE; - outb(info->control.control, info->control.io_port); + outb(info->control, info->ctl_port); + restore_flags(flags); } /* Prototype: void eesoxscsi_intr(irq, *dev_id, *regs) @@ -187,32 +168,26 @@ eesoxscsi_dma_setup(struct Scsi_Host *host, Scsi_Pointer *SCp, fasdmadir_t direction, fasdmatype_t min_type) { - EESOXScsi_Info *info = (EESOXScsi_Info *)host->hostdata; + struct eesoxscsi_info *info = (struct eesoxscsi_info *)host->hostdata; int dmach = host->dma_channel; if (dmach != NO_DMA && (min_type == fasdma_real_all || SCp->this_residual >= 512)) { - int bufs = SCp->buffers_residual; - int pci_dir, dma_dir; + int bufs, map_dir, dma_dir; - if (bufs) - memcpy(info->sg + 1, SCp->buffer + 1, - sizeof(struct scatterlist) * bufs); - info->sg[0].address = SCp->ptr; - info->sg[0].page = NULL; - info->sg[0].length = SCp->this_residual; + bufs = copy_SCp_to_sg(&info->sg[0], SCp, NR_SG); if (direction == DMA_OUT) - pci_dir = PCI_DMA_TODEVICE, + map_dir = PCI_DMA_TODEVICE, dma_dir = DMA_MODE_WRITE; else - pci_dir = PCI_DMA_FROMDEVICE, + map_dir = PCI_DMA_FROMDEVICE, dma_dir = DMA_MODE_READ; - pci_map_sg(NULL, info->sg, bufs + 1, pci_dir); + pci_map_sg(NULL, info->sg, bufs, map_dir); disable_dma(dmach); - set_dma_sg(dmach, info->sg, bufs + 1); + set_dma_sg(dmach, info->sg, bufs); set_dma_mode(dmach, dma_dir); enable_dma(dmach); return fasdma_real_all; @@ -225,215 +200,174 @@ return fasdma_pseudo; } -static void -eesoxscsi_dma_pseudo(struct Scsi_Host *host, Scsi_Pointer *SCp, - fasdmadir_t dir, int transfer_size) +static void eesoxscsi_buffer_in(void *buf, int length, unsigned long base) { - EESOXScsi_Info *info = (EESOXScsi_Info *)host->hostdata; - unsigned int status; - unsigned int length = SCp->this_residual; - union { - unsigned char *c; - unsigned short *s; - unsigned long *l; - } buffer; + const unsigned long reg_fas = base + EESOX_FAS216_OFFSET; + const unsigned long reg_dmastat = base + EESOX_DMASTAT; + const unsigned long reg_dmadata = base + EESOX_DMADATA; + + do { + unsigned int status; + + /* + * Interrupt request? + */ + status = inb(reg_fas + (REG_STAT << EESOX_FAS216_SHIFT)); + if (status & STAT_INT) + break; - buffer.c = SCp->ptr; + /* + * DMA request active? + */ + status = inb(reg_dmastat); + if (!(status & EESOX_STAT_DMA)) + continue; + + /* + * Get number of bytes in FIFO + */ + status = inb(reg_fas + (REG_CFIS << EESOX_FAS216_SHIFT)) & CFIS_CF; + if (status > 16) + status = 16; + if (status > length) + status = length; + + /* + * Align buffer. + */ + if (((u32)buf) & 2 && status >= 2) { + *((u16 *)buf)++ = inw(reg_dmadata); + status -= 2; + length -= 2; + } - status = inb(host->io_port + EESOX_STATUS); - if (dir == DMA_IN) { - while (length > 8) { - if (status & EESOX_STAT_DMA) { - unsigned long l1, l2; - - l1 = inw(info->dmaarea); - l1 |= inw(info->dmaarea) << 16; - l2 = inw(info->dmaarea); - l2 |= inw(info->dmaarea) << 16; - *buffer.l++ = l1; - *buffer.l++ = l2; - length -= 8; - } else if (status & EESOX_STAT_INTR) - goto end; - status = inb(host->io_port + EESOX_STATUS); - } - - while (length > 1) { - if (status & EESOX_STAT_DMA) { - *buffer.s++ = inw(info->dmaarea); - length -= 2; - } else if (status & EESOX_STAT_INTR) - goto end; - status = inb(host->io_port + EESOX_STATUS); - } - - while (length > 0) { - if (status & EESOX_STAT_DMA) { - *buffer.c++ = inw(info->dmaarea); - length -= 1; - } else if (status & EESOX_STAT_INTR) - goto end; - status = inb(host->io_port + EESOX_STATUS); + if (status >= 8) { + unsigned long l1, l2; + + l1 = inw(reg_dmadata); + l1 |= inw(reg_dmadata) << 16; + l2 = inw(reg_dmadata); + l2 |= inw(reg_dmadata) << 16; + *((u32 *)buf)++ = l1; + *((u32 *)buf)++ = l2; + length -= 8; + continue; } - } else { - while (length > 8) { - if (status & EESOX_STAT_DMA) { - unsigned long l1, l2; - - l1 = *buffer.l++; - l2 = *buffer.l++; - - outw(l1, info->dmaarea); - outw(l1 >> 16, info->dmaarea); - outw(l2, info->dmaarea); - outw(l2 >> 16, info->dmaarea); - length -= 8; - } else if (status & EESOX_STAT_INTR) - goto end; - status = inb(host->io_port + EESOX_STATUS); - } - - while (length > 1) { - if (status & EESOX_STAT_DMA) { - outw(*buffer.s++, info->dmaarea); - length -= 2; - } else if (status & EESOX_STAT_INTR) - goto end; - status = inb(host->io_port + EESOX_STATUS); - } - - while (length > 0) { - if (status & EESOX_STAT_DMA) { - outw(*buffer.c++, info->dmaarea); - length -= 1; - } else if (status & EESOX_STAT_INTR) - goto end; - status = inb(host->io_port + EESOX_STATUS); + + if (status >= 4) { + unsigned long l1; + + l1 = inw(reg_dmadata); + l1 |= inw(reg_dmadata) << 16; + *((u32 *)buf)++ = l1; + length -= 4; + continue; } - } -end: -} -/* Prototype: int eesoxscsi_dma_stop(host, SCpnt) - * Purpose : stops DMA/PIO - * Params : host - host - * SCpnt - command - */ -static void -eesoxscsi_dma_stop(struct Scsi_Host *host, Scsi_Pointer *SCp) -{ - if (host->dma_channel != NO_DMA) - disable_dma(host->dma_channel); + if (status >= 2) { + *((u16 *)buf)++ = inw(reg_dmadata); + length -= 2; + } + } while (length); } -/* Prototype: int eesoxscsi_detect(Scsi_Host_Template * tpnt) - * Purpose : initialises EESOX SCSI driver - * Params : tpnt - template for this SCSI adapter - * Returns : >0 if host found, 0 otherwise. - */ -int -eesoxscsi_detect(Scsi_Host_Template *tpnt) +static void eesoxscsi_buffer_out(void *buf, int length, unsigned long base) { - static const card_ids eesoxscsi_cids[] = - { EESOXSCSI_LIST, { 0xffff, 0xffff} }; - int count = 0; - struct Scsi_Host *host; - - tpnt->proc_name = "eesox"; - memset(ecs, 0, sizeof (ecs)); + const unsigned long reg_fas = base + EESOX_FAS216_OFFSET; + const unsigned long reg_dmastat = base + EESOX_DMASTAT; + const unsigned long reg_dmadata = base + EESOX_DMADATA; - ecard_startfind(); - - while(1) { - EESOXScsi_Info *info; + do { + unsigned int status; - ecs[count] = ecard_find(0, eesoxscsi_cids); - if (!ecs[count]) + /* + * Interrupt request? + */ + status = inb(reg_fas + (REG_STAT << EESOX_FAS216_SHIFT)); + if (status & STAT_INT) break; - ecard_claim(ecs[count]); + /* + * DMA request active? + */ + status = inb(reg_dmastat); + if (!(status & EESOX_STAT_DMA)) + continue; + + /* + * Get number of bytes in FIFO + */ + status = inb(reg_fas + (REG_CFIS << EESOX_FAS216_SHIFT)) & CFIS_CF; + if (status > 16) + status = 16; + status = 16 - status; + if (status > length) + status = length; + + /* + * Align buffer. + */ + if (((u32)buf) & 2 && status >= 2) { + outw(*((u16 *)buf)++, reg_dmadata); + status -= 2; + length -= 2; + } - host = scsi_register(tpnt, sizeof (EESOXScsi_Info)); - if (!host) { - ecard_release(ecs[count]); - break; + if (status >= 8) { + unsigned long l1, l2; + + l1 = *((u32 *)buf)++; + l2 = *((u32 *)buf)++; + + outw(l1, reg_dmadata); + outw(l1 >> 16, reg_dmadata); + outw(l2, reg_dmadata); + outw(l2 >> 16, reg_dmadata); + status -= 8; + length -= 8; + continue; } - host->io_port = ecard_address(ecs[count], ECARD_IOC, ECARD_FAST); - host->irq = ecs[count]->irq; - host->dma_channel = ecs[count]->dma; - info = (EESOXScsi_Info *)host->hostdata; - - info->control.io_port = host->io_port + EESOX_CONTROL; - info->control.control = term[count] ? EESOX_TERM_ENABLE : 0; - outb(info->control.control, info->control.io_port); - - ecs[count]->irqaddr = (unsigned char *) - ioaddr(host->io_port + EESOX_STATUS); - ecs[count]->irqmask = EESOX_STAT_INTR; - ecs[count]->irq_data = &info->control; - ecs[count]->ops = (expansioncard_ops_t *)&eesoxscsi_ops; - - info->info.scsi.io_port = host->io_port + EESOX_FAS216_OFFSET; - info->info.scsi.io_shift = EESOX_FAS216_SHIFT; - info->info.scsi.irq = host->irq; - info->info.ifcfg.clockrate = EESOX_XTALFREQ; - info->info.ifcfg.select_timeout = 255; - info->info.ifcfg.asyncperiod = EESOX_ASYNC_PERIOD; - info->info.ifcfg.sync_max_depth = EESOX_SYNC_DEPTH; - info->info.ifcfg.cntl3 = CNTL3_BS8 | CNTL3_FASTSCSI | CNTL3_FASTCLK; - info->info.ifcfg.disconnect_ok = 1; - info->info.ifcfg.wide_max_size = 0; - info->info.dma.setup = eesoxscsi_dma_setup; - info->info.dma.pseudo = eesoxscsi_dma_pseudo; - info->info.dma.stop = eesoxscsi_dma_stop; - info->dmaarea = host->io_port + EESOX_DMA_OFFSET; - - request_region(host->io_port + EESOX_FAS216_OFFSET, - 16 << EESOX_FAS216_SHIFT, "eesox2-fas"); - - if (host->irq != NO_IRQ && - request_irq(host->irq, eesoxscsi_intr, - SA_INTERRUPT, "eesox", host)) { - printk("scsi%d: IRQ%d not free, interrupts disabled\n", - host->host_no, host->irq); - host->irq = NO_IRQ; + if (status >= 4) { + unsigned long l1; + + l1 = *((u32 *)buf)++; + outw(l1, reg_dmadata); + outw(l1 >> 16, reg_dmadata); + status -= 4; + length -= 4; + continue; } - if (host->dma_channel != NO_DMA && - request_dma(host->dma_channel, "eesox")) { - printk("scsi%d: DMA%d not free, DMA disabled\n", - host->host_no, host->dma_channel); - host->dma_channel = NO_DMA; + if (status >= 2) { + outw(*((u16 *)buf)++, reg_dmadata); + length -= 2; } + } while (length); +} - fas216_init(host); - ++count; +static void +eesoxscsi_dma_pseudo(struct Scsi_Host *host, Scsi_Pointer *SCp, + fasdmadir_t dir, int transfer_size) +{ + unsigned int base = host->io_port; + if (dir == DMA_IN) { + eesoxscsi_buffer_in(SCp->ptr, SCp->this_residual, base); + } else { + eesoxscsi_buffer_out(SCp->ptr, SCp->this_residual, base); } - return count; } -/* Prototype: int eesoxscsi_release(struct Scsi_Host * host) - * Purpose : releases all resources used by this adapter - * Params : host - driver host structure to return info for. +/* Prototype: int eesoxscsi_dma_stop(host, SCpnt) + * Purpose : stops DMA/PIO + * Params : host - host + * SCpnt - command */ -int eesoxscsi_release(struct Scsi_Host *host) +static void +eesoxscsi_dma_stop(struct Scsi_Host *host, Scsi_Pointer *SCp) { - int i; - - fas216_release(host); - - if (host->irq != NO_IRQ) - free_irq(host->irq, host); if (host->dma_channel != NO_DMA) - free_dma(host->dma_channel); - release_region(host->io_port + EESOX_FAS216_OFFSET, 16 << EESOX_FAS216_SHIFT); - - for (i = 0; i < MAX_ECARDS; i++) - if (ecs[i] && - host->io_port == ecard_address(ecs[i], ECARD_IOC, ECARD_FAST)) - ecard_release(ecs[i]); - return 0; + disable_dma(host->dma_channel); } /* Prototype: const char *eesoxscsi_info(struct Scsi_Host * host) @@ -443,15 +377,12 @@ */ const char *eesoxscsi_info(struct Scsi_Host *host) { - EESOXScsi_Info *info = (EESOXScsi_Info *)host->hostdata; - static char string[100], *p; + struct eesoxscsi_info *info = (struct eesoxscsi_info *)host->hostdata; + static char string[150]; - p = string; - p += sprintf(p, "%s ", host->hostt->name); - p += fas216_info(&info->info, p); - p += sprintf(p, "v%d.%d.%d terminators o%s", - VER_MAJOR, VER_MINOR, VER_PATCH, - info->control.control & EESOX_TERM_ENABLE ? "n" : "ff"); + sprintf(string, "%s (%s) in slot %d v%s terminators o%s", + host->hostt->name, info->info.scsi.type, info->ec->slot_no, + VERSION, info->control & EESOX_TERM_ENABLE ? "n" : "ff"); return string; } @@ -504,43 +435,37 @@ int length, int host_no, int inout) { int pos, begin; - struct Scsi_Host *host = scsi_hostlist; - EESOXScsi_Info *info; + struct Scsi_Host *host; + struct eesoxscsi_info *info; Scsi_Device *scd; - while (host) { - if (host->host_no == host_no) - break; - host = host->next; - } + host = scsi_host_hn_get(host_no); if (!host) return 0; if (inout == 1) return eesoxscsi_set_proc_info(host, buffer, length); - info = (EESOXScsi_Info *)host->hostdata; + info = (struct eesoxscsi_info *)host->hostdata; begin = 0; - pos = sprintf(buffer, - "EESOX SCSI driver version %d.%d.%d\n", - VER_MAJOR, VER_MINOR, VER_PATCH); + pos = sprintf(buffer, "EESOX SCSI driver v%s\n", VERSION); pos += fas216_print_host(&info->info, buffer + pos); pos += sprintf(buffer + pos, "Term : o%s\n", - info->control.control & EESOX_TERM_ENABLE ? "n" : "ff"); + info->control & EESOX_TERM_ENABLE ? "n" : "ff"); pos += fas216_print_stats(&info->info, buffer + pos); - pos += sprintf (buffer+pos, "\nAttached devices:\n"); + pos += sprintf(buffer+pos, "\nAttached devices:\n"); for (scd = host->host_queue; scd; scd = scd->next) { int len; - proc_print_scsidevice (scd, buffer, &len, pos); + proc_print_scsidevice(scd, buffer, &len, pos); pos += len; - pos += sprintf (buffer+pos, "Extensions: "); + pos += sprintf(buffer+pos, "Extensions: "); if (scd->tagged_supported) - pos += sprintf (buffer+pos, "TAG %sabled [%d] ", + pos += sprintf(buffer+pos, "TAG %sabled [%d] ", scd->tagged_queue ? "en" : "dis", scd->current_tag); pos += sprintf (buffer+pos, "\n"); @@ -558,27 +483,187 @@ return pos; } +static int eesoxscsi_probe(struct expansion_card *ec); + +/* Prototype: int eesoxscsi_detect(Scsi_Host_Template * tpnt) + * Purpose : initialises EESOX SCSI driver + * Params : tpnt - template for this SCSI adapter + * Returns : >0 if host found, 0 otherwise. + */ +static int eesoxscsi_detect(Scsi_Host_Template *tpnt) +{ + static const card_ids eesoxscsi_cids[] = + { EESOXSCSI_LIST, { 0xffff, 0xffff} }; + struct expansion_card *ec; + int count = 0, ret; + + ecard_startfind(); + + while(1) { + ec = ecard_find(0, eesoxscsi_cids); + if (!ec) + break; + + ecard_claim(ec); + + ret = eesoxscsi_probe(ec); + if (ret) { + ecard_release(ec); + break; + } + ++count; + } + return count; +} + +static void eesoxscsi_remove(struct Scsi_Host *host); + +/* Prototype: int eesoxscsi_release(struct Scsi_Host * host) + * Purpose : releases all resources used by this adapter + * Params : host - driver host structure to return info for. + */ +static int eesoxscsi_release(struct Scsi_Host *host) +{ + eesoxscsi_remove(host); + return 0; +} + static Scsi_Host_Template eesox_template = { - module: THIS_MODULE, - proc_info: eesoxscsi_proc_info, - name: "EESOX SCSI", - detect: eesoxscsi_detect, - release: eesoxscsi_release, - info: eesoxscsi_info, - bios_param: scsicam_bios_param, - can_queue: 1, - this_id: 7, - sg_tablesize: SG_ALL, - cmd_per_lun: 1, - use_clustering: DISABLE_CLUSTERING, - command: fas216_command, - queuecommand: fas216_queue_command, - eh_host_reset_handler: fas216_eh_host_reset, - eh_bus_reset_handler: fas216_eh_bus_reset, - eh_device_reset_handler: fas216_eh_device_reset, - eh_abort_handler: fas216_eh_abort, - use_new_eh_code: 1 + .module = THIS_MODULE, + .proc_info = eesoxscsi_proc_info, + .name = "EESOX SCSI", + .detect = eesoxscsi_detect, + .release = eesoxscsi_release, + .info = eesoxscsi_info, + .bios_param = scsicam_bios_param, + .command = fas216_command, + .queuecommand = fas216_queue_command, + .eh_host_reset_handler = fas216_eh_host_reset, + .eh_bus_reset_handler = fas216_eh_bus_reset, + .eh_device_reset_handler = fas216_eh_device_reset, + .eh_abort_handler = fas216_eh_abort, + .use_new_eh_code = 1, + + .can_queue = 1, + .this_id = 7, + .sg_tablesize = SG_ALL, + .cmd_per_lun = 1, + .use_clustering = DISABLE_CLUSTERING, + .proc_name = "eesox", }; + +static int +eesoxscsi_probe(struct expansion_card *ec) +{ + struct Scsi_Host *host; + struct eesoxscsi_info *info; + unsigned long base; + int ret; + + base = ecard_address(ec, ECARD_IOC, ECARD_FAST); + + if (!request_region(base + EESOX_FAS216_OFFSET, + 16 << EESOX_FAS216_SHIFT, "eesox2-fas")) { + ret = -EBUSY; + goto out; + } + + host = scsi_register(&eesox_template, + sizeof(struct eesoxscsi_info)); + if (!host) { + ret = -ENOMEM; + goto out_region; + } + + host->io_port = base; + host->irq = ec->irq; + host->dma_channel = ec->dma; + + info = (struct eesoxscsi_info *)host->hostdata; + info->ec = ec; + info->ctl_port = base + EESOX_CONTROL; + info->control = term[ec->slot_no] ? EESOX_TERM_ENABLE : 0; + outb(info->control, info->ctl_port); + + ec->irqaddr = (unsigned char *)ioaddr(base + EESOX_DMASTAT); + ec->irqmask = EESOX_STAT_INTR; + ec->irq_data = info; + ec->ops = (expansioncard_ops_t *)&eesoxscsi_ops; + + info->info.scsi.io_port = base + EESOX_FAS216_OFFSET; + info->info.scsi.io_shift = EESOX_FAS216_SHIFT; + info->info.scsi.irq = host->irq; + info->info.ifcfg.clockrate = 40; /* MHz */ + info->info.ifcfg.select_timeout = 255; + info->info.ifcfg.asyncperiod = 200; /* ns */ + info->info.ifcfg.sync_max_depth = 7; + info->info.ifcfg.cntl3 = CNTL3_FASTSCSI | CNTL3_FASTCLK; + info->info.ifcfg.disconnect_ok = 1; + info->info.ifcfg.wide_max_size = 0; + info->info.ifcfg.capabilities = FASCAP_PSEUDODMA; + info->info.dma.setup = eesoxscsi_dma_setup; + info->info.dma.pseudo = eesoxscsi_dma_pseudo; + info->info.dma.stop = eesoxscsi_dma_stop; + + ret = fas216_init(host); + if (ret) + goto out_free; + + ret = request_irq(host->irq, eesoxscsi_intr, 0, "eesox", host); + if (ret) { + printk("scsi%d: IRQ%d not free: %d\n", + host->host_no, host->irq, ret); + goto out_remove; + } + + if (host->dma_channel != NO_DMA) { + if (request_dma(host->dma_channel, "eesox")) { + printk("scsi%d: DMA%d not free, DMA disabled\n", + host->host_no, host->dma_channel); + host->dma_channel = NO_DMA; + } else { + set_dma_speed(host->dma_channel, 180); + info->info.ifcfg.capabilities |= FASCAP_DMA; + info->info.ifcfg.cntl3 |= CNTL3_BS8; + } + } + + ret = fas216_add(host); + if (ret == 0) + goto out; + + if (host->dma_channel != NO_DMA) + free_dma(host->dma_channel); + free_irq(host->irq, host); + + out_remove: + fas216_remove(host); + + out_free: + scsi_unregister(host); + + out_region: + release_region(base + EESOX_FAS216_OFFSET, 16 << EESOX_FAS216_SHIFT); + + out: + return ret; +} + +static void eesoxscsi_remove(struct Scsi_Host *host) +{ + struct eesoxscsi_info *info = (struct eesoxscsi_info *)host->hostdata; + + fas216_remove(host); + + if (host->dma_channel != NO_DMA) + free_dma(host->dma_channel); + free_irq(host->irq, host); + + release_region(host->io_port + EESOX_FAS216_OFFSET, 16 << EESOX_FAS216_SHIFT); + + fas216_release(host); + ecard_release(info->ec); +} static int __init eesox_init(void) { diff -urN orig/drivers/acorn/scsi/fas216.c linux/drivers/acorn/scsi/fas216.c --- orig/drivers/acorn/scsi/fas216.c Fri Oct 26 16:46:02 2001 +++ linux/drivers/acorn/scsi/fas216.c Thu Jan 23 13:44:14 2003 @@ -1,7 +1,7 @@ /* - * linux/arch/arm/drivers/scsi/fas216.c + * linux/drivers/acorn/scsi/fas216.c * - * Copyright (C) 1997-2000 Russell King + * Copyright (C) 1997-2003 Russell King * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as @@ -10,7 +10,7 @@ * Based on information in qlogicfas.c by Tom Zerucha, Michael Griffith, and * other sources, including: * the AMD Am53CF94 data sheet - * the AMD Am53C94 data sheet + * the AMD Am53C94 data sheet * * This is a generic driver. To use it, have a look at cumana_2.c. You * should define your own structure that overlays FAS216_Info, eg: @@ -33,9 +33,6 @@ * 02-04-2000 RMK Converted to use the new error handling, and * automatically request sense data upon check * condition status from targets. - * - * Todo: - * - allow individual devices to enable sync xfers. */ #include #include @@ -44,9 +41,8 @@ #include #include #include -#include -#include #include +#include #include #include @@ -54,15 +50,10 @@ #include #include -#define FAS216_C - #include "../../scsi/scsi.h" #include "../../scsi/hosts.h" #include "fas216.h" - -#define VER_MAJOR 0 -#define VER_MINOR 0 -#define VER_PATCH 5 +#include "scsi.h" /* NOTE: SCSI2 Synchronous transfers *require* DMA according to * the data sheet. This restriction is crazy, especially when @@ -92,31 +83,61 @@ #undef SCSI2_TAG #undef DEBUG_CONNECT -#undef DEBUG_BUSSERVICE -#undef DEBUG_FUNCTIONDONE #undef DEBUG_MESSAGES #undef CHECK_STRUCTURE -static struct { int stat, ssr, isr, ph; } list[8]; -static int ptr; +#define LOG_CONNECT (1 << 0) +#define LOG_BUSSERVICE (1 << 1) +#define LOG_FUNCTIONDONE (1 << 2) +#define LOG_MESSAGES (1 << 3) +#define LOG_BUFFER (1 << 4) +#define LOG_ERROR (1 << 8) + +static int level_mask = LOG_ERROR; + +MODULE_PARM(level_mask, "i"); + +static inline unsigned char fas216_readb(FAS216_Info *info, unsigned int reg) +{ + unsigned int off = reg << info->scsi.io_shift; + + return inb(info->scsi.io_port + off); +} + +static inline void fas216_writeb(FAS216_Info *info, unsigned int reg, unsigned int val) +{ + unsigned int off = reg << info->scsi.io_shift; + + outb(val, info->scsi.io_port + off); +} static void fas216_dumpstate(FAS216_Info *info) { unsigned char is, stat, inst; - is = inb(REG_IS(info)); - stat = inb(REG_STAT(info)); - inst = inb(REG_INST(info)); + is = fas216_readb(info, REG_IS); + stat = fas216_readb(info, REG_STAT); + inst = fas216_readb(info, REG_INST); printk("FAS216: CTCL=%02X CTCM=%02X CMD=%02X STAT=%02X" " INST=%02X IS=%02X CFIS=%02X", - inb(REG_CTCL(info)), inb(REG_CTCM(info)), - inb(REG_CMD(info)), stat, inst, is, - inb(REG_CFIS(info))); + fas216_readb(info, REG_CTCL), + fas216_readb(info, REG_CTCM), + fas216_readb(info, REG_CMD), stat, inst, is, + fas216_readb(info, REG_CFIS)); printk(" CNTL1=%02X CNTL2=%02X CNTL3=%02X CTCH=%02X\n", - inb(REG_CNTL1(info)), inb(REG_CNTL2(info)), - inb(REG_CNTL3(info)), inb(REG_CTCH(info))); + fas216_readb(info, REG_CNTL1), + fas216_readb(info, REG_CNTL2), + fas216_readb(info, REG_CNTL3), + fas216_readb(info, REG_CTCH)); +} + +static void print_SCp(Scsi_Pointer *SCp, const char *prefix, const char *suffix) +{ + printk("%sptr %p this_residual 0x%x buffer %p buffers_residual 0x%x%s", + prefix, SCp->ptr, SCp->this_residual, SCp->buffer, + SCp->buffers_residual, suffix); } static void fas216_dumpinfo(FAS216_Info *info) @@ -139,9 +160,7 @@ info->scsi.type, info->scsi.phase, info->scsi.reconnected.target, info->scsi.reconnected.lun, info->scsi.reconnected.tag); - printk(" SCp={ ptr=%p this_residual=%X buffer=%p buffers_residual=%X }\n", - info->scsi.SCp.ptr, info->scsi.SCp.this_residual, - info->scsi.SCp.buffer, info->scsi.SCp.buffers_residual); + print_SCp(&info->scsi.SCp, " SCp={ ", " }\n"); printk(" msgs async_stp=%X disconnectable=%d aborting=%d }\n", info->scsi.async_stp, info->scsi.disconnectable, info->scsi.aborting); @@ -155,7 +174,7 @@ info->ifcfg.clockrate, info->ifcfg.select_timeout, info->ifcfg.asyncperiod, info->ifcfg.sync_max_depth); for (i = 0; i < 8; i++) { - printk(" busyluns[%d]=%X dev[%d]={ disconnect_ok=%d stp=%X sof=%X sync_state=%X }\n", + printk(" busyluns[%d]=%02x dev[%d]={ disconnect_ok=%d stp=%X sof=%X sync_state=%X }\n", i, info->busyluns[i], i, info->device[i].disconnect_ok, info->device[i].stp, info->device[i].sof, info->device[i].sync_state); @@ -228,39 +247,127 @@ return 'H'; } +static void +fas216_do_log(FAS216_Info *info, char target, char *fmt, va_list ap) +{ + static char buf[1024]; + + vsnprintf(buf, sizeof(buf), fmt, ap); + printk("scsi%d.%c: %s", info->host->host_no, target, buf); +} + +static void +fas216_log_command(FAS216_Info *info, int level, Scsi_Cmnd *SCpnt, char *fmt, ...) +{ + va_list args; + + if (level != 0 && !(level & level_mask)) + return; + + va_start(args, fmt); + fas216_do_log(info, '0' + SCpnt->target, fmt, args); + va_end(args); + + print_command(SCpnt->cmnd); +} + +static void +fas216_log_target(FAS216_Info *info, int level, int target, char *fmt, ...) +{ + va_list args; + + if (level != 0 && !(level & level_mask)) + return; + + if (target < 0) + target = 'H'; + else + target += '0'; + + va_start(args, fmt); + fas216_do_log(info, target, fmt, args); + va_end(args); + + printk("\n"); +} + +static void fas216_log(FAS216_Info *info, int level, char *fmt, ...) +{ + va_list args; + + if (level != 0 && !(level & level_mask)) + return; + + va_start(args, fmt); + fas216_do_log(info, fas216_target(info), fmt, args); + va_end(args); + + printk("\n"); +} + +#define PH_SIZE 32 + +static struct { int stat, ssr, isr, ph; } ph_list[PH_SIZE]; +static int ph_ptr; + static void add_debug_list(int stat, int ssr, int isr, int ph) { - list[ptr].stat = stat; - list[ptr].ssr = ssr; - list[ptr].isr = isr; - list[ptr].ph = ph; + ph_list[ph_ptr].stat = stat; + ph_list[ph_ptr].ssr = ssr; + ph_list[ph_ptr].isr = isr; + ph_list[ph_ptr].ph = ph; + + ph_ptr = (ph_ptr + 1) & (PH_SIZE-1); +} + +static struct { int command; void *from; } cmd_list[8]; +static int cmd_ptr; + +static void fas216_cmd(FAS216_Info *info, unsigned int command) +{ + cmd_list[cmd_ptr].command = command; + cmd_list[cmd_ptr].from = __builtin_return_address(0); - ptr = (ptr + 1) & 7; + cmd_ptr = (cmd_ptr + 1) & 7; + + fas216_writeb(info, REG_CMD, command); } static void print_debug_list(void) { int i; - i = ptr; + i = ph_ptr; + + printk(KERN_ERR "SCSI IRQ trail\n"); + do { + printk(" %02x:%02x:%02x:%1x", + ph_list[i].stat, ph_list[i].ssr, + ph_list[i].isr, ph_list[i].ph); + i = (i + 1) & (PH_SIZE - 1); + if (((i ^ ph_ptr) & 7) == 0) + printk("\n"); + } while (i != ph_ptr); + if ((i ^ ph_ptr) & 7) + printk("\n"); - printk(KERN_ERR "SCSI IRQ trail: "); + i = cmd_ptr; + printk(KERN_ERR "FAS216 commands: "); do { - printk("%02X:%02X:%02X:%1X ", - list[i].stat, list[i].ssr, - list[i].isr, list[i].ph); + printk("%02x:%p ", cmd_list[i].command, cmd_list[i].from); i = (i + 1) & 7; - } while (i != ptr); + } while (i != cmd_ptr); printk("\n"); } static void fas216_done(FAS216_Info *info, unsigned int result); -/* Function: int fas216_clockrate(unsigned int clock) - * Purpose : calculate correct value to be written into clock conversion - * factor register. - * Params : clock - clock speed in MHz - * Returns : CLKF_ value +/** + * fas216_clockrate - calculate clock conversion factor + * @clock: clock speed in MHz + * + * Calculate correct value to be written into clock conversion factor + * register. Returns CLKF_ value. */ static int fas216_clockrate(int clock) { @@ -274,10 +381,12 @@ return clock; } -/* Function: unsigned short fas216_get_last_msg(FAS216_Info *info, int pos) - * Purpose : retrieve a last message from the list, using position in fifo - * Params : info - interface to search - * : pos - current fifo position +/** + * fas216_get_last_msg - retrive last message from the list + * @info: interface to search + * @pos: current fifo position + * + * Retrieve a last message from the list, using position in fifo. */ static inline unsigned short fas216_get_last_msg(FAS216_Info *info, int pos) @@ -298,22 +407,21 @@ packed_msg = msg->msg[0]; } -#ifdef DEBUG_MESSAGES - printk("Message: %04X found at position %02X\n", - packed_msg, pos); -#endif + fas216_log(info, LOG_MESSAGES, + "Message: %04x found at position %02x\n", packed_msg, pos); + return packed_msg; } -/* Function: int fas216_syncperiod(FAS216_Info *info, int ns) - * Purpose : Calculate value to be loaded into the STP register - * for a given period in ns - * Params : info - state structure for interface connected to device - * : ns - period in ns (between subsequent bytes) - * Returns : Value suitable for REG_STP +/** + * fas216_syncperiod - calculate STP register value + * @info: state structure for interface connected to device + * @ns: period in ns (between subsequent bytes) + * + * Calculate value to be loaded into the STP register for a given period + * in ns. Returns a value suitable for REG_STP. */ -static int -fas216_syncperiod(FAS216_Info *info, int ns) +static int fas216_syncperiod(FAS216_Info *info, int ns) { int value = (info->ifcfg.clockrate * ns) / 1000; @@ -327,23 +435,28 @@ return value & 31; } -/* Function: void fas216_set_sync(FAS216_Info *info, int target) - * Purpose : Correctly setup FAS216 chip for specified transfer period. - * Params : info - state structure for interface - * : target - target +/** + * fas216_set_sync - setup FAS216 chip for specified transfer period. + * @info: state structure for interface connected to device + * @target: target + * + * Correctly setup FAS216 chip for specified transfer period. * Notes : we need to switch the chip out of FASTSCSI mode if we have * a transfer period >= 200ns - otherwise the chip will violate * the SCSI timings. */ -static void -fas216_set_sync(FAS216_Info *info, int target) +static void fas216_set_sync(FAS216_Info *info, int target) { - outb(info->device[target].sof, REG_SOF(info)); - outb(info->device[target].stp, REG_STP(info)); + unsigned int cntl3; + + fas216_writeb(info, REG_SOF, info->device[target].sof); + fas216_writeb(info, REG_STP, info->device[target].stp); + + cntl3 = info->scsi.cfg[2]; if (info->device[target].period >= (200 / 4)) - outb(info->scsi.cfg[2] & ~CNTL3_FASTSCSI, REG_CNTL3(info)); - else - outb(info->scsi.cfg[2], REG_CNTL3(info)); + cntl3 = cntl3 & ~CNTL3_FASTSCSI; + + fas216_writeb(info, REG_CNTL3, cntl3); } /* Synchronous transfer support @@ -364,19 +477,20 @@ * Using the correct method assures compatibility with wide data * transfers and future enhancements. * - * We will always initiate a synchronous transfer negociation request on + * We will always initiate a synchronous transfer negotiation request on * every INQUIRY or REQUEST SENSE message, unless the target itself has - * at some point performed a synchronous transfer negociation request, or + * at some point performed a synchronous transfer negotiation request, or * we have synchronous transfers disabled for this device. */ -/* Function: void fas216_handlesync(FAS216_Info *info, char *msg) - * Purpose : Handle a synchronous transfer message from the target - * Params : info - state structure for interface - * : msg - message from target +/** + * fas216_handlesync - Handle a synchronous transfer message + * @info: state structure for interface + * @msg: message from target + * + * Handle a synchronous transfer message from the target */ -static void -fas216_handlesync(FAS216_Info *info, char *msg) +static void fas216_handlesync(FAS216_Info *info, char *msg) { struct fas216_device *dev = &info->device[info->SCpnt->target]; enum { sync, async, none, reject } res = none; @@ -392,7 +506,7 @@ * message with a MESSAGE REJECT message. * * Hence, if we get this condition, we disable - * negociation for this device. + * negotiation for this device. */ if (dev->sync_state == neg_inprogress) { dev->sync_state = neg_invalid; @@ -410,14 +524,14 @@ res = reject; break; - /* We were not negociating a synchronous transfer, - * but the device sent us a negociation request. + /* We were not negotiating a synchronous transfer, + * but the device sent us a negotiation request. * Honour the request by sending back a SDTR * message containing our capability, limited by * the targets capability. */ default: - outb(CMD_SETATN, REG_CMD(info)); + fas216_cmd(info, CMD_SETATN); if (msg[4] > info->ifcfg.sync_max_depth) msg[4] = info->ifcfg.sync_max_depth; if (msg[3] < 1000 / info->ifcfg.clockrate) @@ -436,7 +550,7 @@ res = sync; break; - /* We initiated the synchronous transfer negociation, + /* We initiated the synchronous transfer negotiation, * and have successfully received a response from the * target. The synchronous transfer agreement has been * reached. Note: if the values returned are out of our @@ -465,7 +579,7 @@ break; case reject: - outb(CMD_SETATN, REG_CMD(info)); + fas216_cmd(info, CMD_SETATN); msgqueue_flush(&info->scsi.msgs); msgqueue_addmsg(&info->scsi.msgs, 1, MESSAGE_REJECT); info->scsi.phase = PHASE_MSGOUT_EXPECT; @@ -482,13 +596,14 @@ } } -/* Function: void fas216_handlewide(FAS216_Info *info, char *msg) - * Purpose : Handle a wide transfer message from the target - * Params : info - state structure for interface - * : msg - message from target +/** + * fas216_handlewide - Handle a wide transfer message + * @info: state structure for interface + * @msg: message from target + * + * Handle a wide transfer message from the target */ -static void -fas216_handlewide(FAS216_Info *info, char *msg) +static void fas216_handlewide(FAS216_Info *info, char *msg) { struct fas216_device *dev = &info->device[info->SCpnt->target]; enum { wide, bit8, none, reject } res = none; @@ -504,7 +619,7 @@ * WDTR message with a MESSAGE REJECT message. * * Hence, if we get this condition, we never - * reattempt negociation for this device. + * reattempt negotiation for this device. */ if (dev->wide_state == neg_inprogress) { dev->wide_state = neg_invalid; @@ -522,14 +637,14 @@ res = reject; break; - /* We were not negociating a wide data transfer, - * but the device sent is a negociation request. + /* We were not negotiating a wide data transfer, + * but the device sent is a negotiation request. * Honour the request by sending back a WDTR * message containing our capability, limited by * the targets capability. */ default: - outb(CMD_SETATN, REG_CMD(info)); + fas216_cmd(info, CMD_SETATN); if (msg[3] > info->ifcfg.wide_max_size) msg[3] = info->ifcfg.wide_max_size; @@ -541,7 +656,7 @@ res = wide; break; - /* We initiated the wide data transfer negociation, + /* We initiated the wide data transfer negotiation, * and have successfully received a response from the * target. The synchronous transfer agreement has been * reached. Note: if the values returned are out of our @@ -566,7 +681,7 @@ break; case reject: - outb(CMD_SETATN, REG_CMD(info)); + fas216_cmd(info, CMD_SETATN); msgqueue_flush(&info->scsi.msgs); msgqueue_addmsg(&info->scsi.msgs, 1, MESSAGE_REJECT); info->scsi.phase = PHASE_MSGOUT_EXPECT; @@ -580,236 +695,277 @@ } } -/* Function: void fas216_updateptrs(FAS216_Info *info, int bytes_transferred) - * Purpose : update data pointers after transfer suspended/paused - * Params : info - interface's local pointer to update - * bytes_transferred - number of bytes transferred +/** + * fas216_updateptrs - update data pointers after transfer suspended/paused + * @info: interface's local pointer to update + * @bytes_transferred: number of bytes transferred + * + * Update data pointers after transfer suspended/paused */ -static void -fas216_updateptrs(FAS216_Info *info, int bytes_transferred) +static void fas216_updateptrs(FAS216_Info *info, int bytes_transferred) { - unsigned char *ptr; - unsigned int residual; + Scsi_Pointer *SCp = &info->scsi.SCp; fas216_checkmagic(info); - ptr = info->scsi.SCp.ptr; - residual = info->scsi.SCp.this_residual; + BUG_ON(bytes_transferred < 0); info->SCpnt->request_bufflen -= bytes_transferred; - while (residual <= bytes_transferred && bytes_transferred) { - /* We have used up this buffer */ - bytes_transferred -= residual; - if (info->scsi.SCp.buffers_residual) { - info->scsi.SCp.buffer++; - info->scsi.SCp.buffers_residual--; - ptr = (unsigned char *)info->scsi.SCp.buffer->address; - residual = info->scsi.SCp.buffer->length; - } else { - ptr = NULL; - residual = 0; + while (bytes_transferred != 0) { + if (SCp->this_residual > bytes_transferred) + break; + /* + * We have used up this buffer. Move on to the + * next buffer. + */ + bytes_transferred -= SCp->this_residual; + if (!next_SCp(SCp) && bytes_transferred) { + printk(KERN_WARNING "scsi%d.%c: out of buffers\n", + info->host->host_no, '0' + info->SCpnt->target); + return; } } - residual -= bytes_transferred; - ptr += bytes_transferred; - - if (residual == 0) - ptr = NULL; - - info->scsi.SCp.ptr = ptr; - info->scsi.SCp.this_residual = residual; + SCp->this_residual -= bytes_transferred; + if (SCp->this_residual) + SCp->ptr += bytes_transferred; + else + SCp->ptr = NULL; } -/* Function: void fas216_pio(FAS216_Info *info, fasdmadir_t direction) - * Purpose : transfer data off of/on to card using programmed IO - * Params : info - interface to transfer data to/from - * direction - direction to transfer data (DMA_OUT/DMA_IN) - * Notes : this is incredibly slow +/** + * fas216_pio - transfer data off of/on to card using programmed IO + * @info: interface to transfer data to/from + * @direction: direction to transfer data (DMA_OUT/DMA_IN) + * + * Transfer data off of/on to card using programmed IO. + * Notes: this is incredibly slow. */ -static void -fas216_pio(FAS216_Info *info, fasdmadir_t direction) +static void fas216_pio(FAS216_Info *info, fasdmadir_t direction) { - unsigned int residual; - char *ptr; + Scsi_Pointer *SCp = &info->scsi.SCp; fas216_checkmagic(info); - residual = info->scsi.SCp.this_residual; - ptr = info->scsi.SCp.ptr; - if (direction == DMA_OUT) - outb(*ptr++, REG_FF(info)); + fas216_writeb(info, REG_FF, get_next_SCp_byte(SCp)); else - *ptr++ = inb(REG_FF(info)); + put_next_SCp_byte(SCp, fas216_readb(info, REG_FF)); - residual -= 1; + if (SCp->this_residual == 0) + next_SCp(SCp); +} - if (residual == 0) { - if (info->scsi.SCp.buffers_residual) { - info->scsi.SCp.buffer++; - info->scsi.SCp.buffers_residual--; - ptr = (unsigned char *)info->scsi.SCp.buffer->address; - residual = info->scsi.SCp.buffer->length; - } else { - ptr = NULL; - residual = 0; - } - } +static void fas216_set_stc(FAS216_Info *info, unsigned int length) +{ + fas216_writeb(info, REG_STCL, length); + fas216_writeb(info, REG_STCM, length >> 8); + fas216_writeb(info, REG_STCH, length >> 16); +} - info->scsi.SCp.ptr = ptr; - info->scsi.SCp.this_residual = residual; +static unsigned int fas216_get_ctc(FAS216_Info *info) +{ + return fas216_readb(info, REG_CTCL) + + (fas216_readb(info, REG_CTCM) << 8) + + (fas216_readb(info, REG_CTCH) << 16); } -/* Function: void fas216_starttransfer(FAS216_Info *info, - * fasdmadir_t direction) - * Purpose : Start a DMA/PIO transfer off of/on to card - * Params : info - interface from which device disconnected from - * direction - transfer direction (DMA_OUT/DMA_IN) +/** + * fas216_cleanuptransfer - clean up after a transfer has completed. + * @info: interface to clean up + * + * Update the data pointers according to the number of bytes transferred + * on the SCSI bus. */ -static void -fas216_starttransfer(FAS216_Info *info, fasdmadir_t direction, int flush_fifo) +static void fas216_cleanuptransfer(FAS216_Info *info) { - fasdmatype_t dmatype; + unsigned long total, residual, fifo; + fasdmatype_t dmatype = info->dma.transfer_type; - fas216_checkmagic(info); + info->dma.transfer_type = fasdma_none; + + /* + * PIO transfers do not need to be cleaned up. + */ + if (dmatype == fasdma_pio || dmatype == fasdma_none) + return; - info->scsi.phase = (direction == DMA_OUT) ? - PHASE_DATAOUT : PHASE_DATAIN; + if (dmatype == fasdma_real_all) + total = info->SCpnt->request_bufflen; + else + total = info->scsi.SCp.this_residual; - if (info->dma.transfer_type != fasdma_none && - info->dma.transfer_type != fasdma_pio) { - unsigned long total, residual; + residual = fas216_get_ctc(info); - if (info->dma.transfer_type == fasdma_real_all) - total = info->SCpnt->request_bufflen; - else - total = info->scsi.SCp.this_residual; + fifo = fas216_readb(info, REG_CFIS) & CFIS_CF; - residual = (inb(REG_CFIS(info)) & CFIS_CF) + - inb(REG_CTCL(info)) + - (inb(REG_CTCM(info)) << 8) + - (inb(REG_CTCH(info)) << 16); - fas216_updateptrs(info, total - residual); - } - info->dma.transfer_type = fasdma_none; + fas216_log(info, LOG_BUFFER, "cleaning up from previous " + "transfer: length 0x%06x, residual 0x%x, fifo %d", + total, residual, fifo); + + /* + * If we were performing Data-Out, the transfer counter + * counts down each time a byte is transferred by the + * host to the FIFO. This means we must include the + * bytes left in the FIFO from the transfer counter. + */ + if (info->scsi.phase == PHASE_DATAOUT) + residual += fifo; + + fas216_updateptrs(info, total - residual); +} + +/** + * fas216_transfer - Perform a DMA/PIO transfer off of/on to card + * @info: interface from which device disconnected from + * + * Start a DMA/PIO transfer off of/on to card + */ +static void fas216_transfer(FAS216_Info *info) +{ + fasdmadir_t direction; + fasdmatype_t dmatype; + + fas216_log(info, LOG_BUFFER, + "starttransfer: buffer %p length 0x%06x reqlen 0x%06x", + info->scsi.SCp.ptr, info->scsi.SCp.this_residual, + info->SCpnt->request_bufflen); if (!info->scsi.SCp.ptr) { - printk("scsi%d.%c: null buffer passed to " - "fas216_starttransfer\n", info->host->host_no, - fas216_target(info)); + fas216_log(info, LOG_ERROR, "null buffer passed to " + "fas216_starttransfer"); + print_SCp(&info->scsi.SCp, "SCp: ", "\n"); + print_SCp(&info->SCpnt->SCp, "Cmnd SCp: ", "\n"); return; } - /* flush FIFO */ - if (flush_fifo) - outb(CMD_FLUSHFIFO, REG_CMD(info)); - /* - * Default to PIO mode or DMA mode if we have a synchronous - * transfer agreement. + * If we have a synchronous transfer agreement in effect, we must + * use DMA mode. If we are using asynchronous transfers, we may + * use DMA mode or PIO mode. */ - if (info->device[info->SCpnt->target].sof && info->dma.setup) + if (info->device[info->SCpnt->target].sof) dmatype = fasdma_real_all; else dmatype = fasdma_pio; + if (info->scsi.phase == PHASE_DATAOUT) + direction = DMA_OUT; + else + direction = DMA_IN; + if (info->dma.setup) dmatype = info->dma.setup(info->host, &info->scsi.SCp, direction, dmatype); info->dma.transfer_type = dmatype; + if (dmatype == fasdma_real_all) + fas216_set_stc(info, info->SCpnt->request_bufflen); + else + fas216_set_stc(info, info->scsi.SCp.this_residual); + switch (dmatype) { case fasdma_pio: - outb(0, REG_SOF(info)); - outb(info->scsi.async_stp, REG_STP(info)); - outb(info->scsi.SCp.this_residual, REG_STCL(info)); - outb(info->scsi.SCp.this_residual >> 8, REG_STCM(info)); - outb(info->scsi.SCp.this_residual >> 16, REG_STCH(info)); - outb(CMD_TRANSFERINFO, REG_CMD(info)); + fas216_log(info, LOG_BUFFER, "PIO transfer"); + fas216_writeb(info, REG_SOF, 0); + fas216_writeb(info, REG_STP, info->scsi.async_stp); + fas216_cmd(info, CMD_TRANSFERINFO); fas216_pio(info, direction); break; case fasdma_pseudo: - outb(info->scsi.SCp.this_residual, REG_STCL(info)); - outb(info->scsi.SCp.this_residual >> 8, REG_STCM(info)); - outb(info->scsi.SCp.this_residual >> 16, REG_STCH(info)); - outb(CMD_TRANSFERINFO | CMD_WITHDMA, REG_CMD(info)); + fas216_log(info, LOG_BUFFER, "pseudo transfer"); + fas216_cmd(info, CMD_TRANSFERINFO | CMD_WITHDMA); info->dma.pseudo(info->host, &info->scsi.SCp, direction, info->SCpnt->transfersize); break; case fasdma_real_block: - outb(info->scsi.SCp.this_residual, REG_STCL(info)); - outb(info->scsi.SCp.this_residual >> 8, REG_STCM(info)); - outb(info->scsi.SCp.this_residual >> 16, REG_STCH(info)); - outb(CMD_TRANSFERINFO | CMD_WITHDMA, REG_CMD(info)); + fas216_log(info, LOG_BUFFER, "block dma transfer"); + fas216_cmd(info, CMD_TRANSFERINFO | CMD_WITHDMA); break; case fasdma_real_all: - outb(info->SCpnt->request_bufflen, REG_STCL(info)); - outb(info->SCpnt->request_bufflen >> 8, REG_STCM(info)); - outb(info->SCpnt->request_bufflen >> 16, REG_STCH(info)); - outb(CMD_TRANSFERINFO | CMD_WITHDMA, REG_CMD(info)); + fas216_log(info, LOG_BUFFER, "total dma transfer"); + fas216_cmd(info, CMD_TRANSFERINFO | CMD_WITHDMA); break; default: - printk(KERN_ERR "scsi%d.%d: invalid FAS216 DMA type\n", - info->host->host_no, fas216_target(info)); + fas216_log(info, LOG_BUFFER | LOG_ERROR, + "invalid FAS216 DMA type"); break; } } -/* Function: void fas216_stoptransfer(FAS216_Info *info) - * Purpose : Stop a DMA transfer onto / off of the card - * Params : info - interface from which device disconnected from +/** + * fas216_stoptransfer - Stop a DMA transfer onto / off of the card + * @info: interface from which device disconnected from + * + * Called when we switch away from DATA IN or DATA OUT phases. */ -static void -fas216_stoptransfer(FAS216_Info *info) +static void fas216_stoptransfer(FAS216_Info *info) { fas216_checkmagic(info); - if (info->dma.transfer_type != fasdma_none && - info->dma.transfer_type != fasdma_pio) { - unsigned long total, residual; + if (info->dma.transfer_type == fasdma_real_all || + info->dma.transfer_type == fasdma_real_block) + info->dma.stop(info->host, &info->scsi.SCp); - if ((info->dma.transfer_type == fasdma_real_all || - info->dma.transfer_type == fasdma_real_block) && - info->dma.stop) - info->dma.stop(info->host, &info->scsi.SCp); + fas216_cleanuptransfer(info); - if (info->dma.transfer_type == fasdma_real_all) - total = info->SCpnt->request_bufflen; - else - total = info->scsi.SCp.this_residual; + if (info->scsi.phase == PHASE_DATAIN) { + unsigned int fifo; - residual = (inb(REG_CFIS(info)) & CFIS_CF) + - inb(REG_CTCL(info)) + - (inb(REG_CTCM(info)) << 8) + - (inb(REG_CTCH(info)) << 16); - fas216_updateptrs(info, total - residual); - info->dma.transfer_type = fasdma_none; + /* + * If we were performing Data-In, then the FIFO counter + * contains the number of bytes not transferred via DMA + * from the on-board FIFO. Read them manually. + */ + fifo = fas216_readb(info, REG_CFIS) & CFIS_CF; + while (fifo && info->scsi.SCp.ptr) { + *info->scsi.SCp.ptr = fas216_readb(info, REG_FF); + fas216_updateptrs(info, 1); + fifo--; + } + } else { + /* + * After a Data-Out phase, there may be unsent bytes left + * in the FIFO. Flush them out. + */ + fas216_cmd(info, CMD_FLUSHFIFO); } - if (info->scsi.phase == PHASE_DATAOUT) - outb(CMD_FLUSHFIFO, REG_CMD(info)); } -/* Function: void fas216_disconnected_intr(FAS216_Info *info) - * Purpose : handle device disconnection - * Params : info - interface from which device disconnected from +static void fas216_aborttransfer(FAS216_Info *info) +{ + fas216_checkmagic(info); + + if (info->dma.transfer_type == fasdma_real_all || + info->dma.transfer_type == fasdma_real_block) + info->dma.stop(info->host, &info->scsi.SCp); + + info->dma.transfer_type = fasdma_none; + fas216_cmd(info, CMD_FLUSHFIFO); +} + +/** + * fas216_disconnected_intr - handle device disconnection + * @info: interface from which device disconnected from + * + * Handle device disconnection */ -static void -fas216_disconnect_intr(FAS216_Info *info) +static void fas216_disconnect_intr(FAS216_Info *info) { fas216_checkmagic(info); -#ifdef DEBUG_CONNECT - printk("scsi%d.%c: disconnect phase=%02X\n", info->host->host_no, - fas216_target(info), info->scsi.phase); -#endif + fas216_log(info, LOG_CONNECT, "disconnect phase=%02x", + info->scsi.phase); + msgqueue_flush(&info->scsi.msgs); + fas216_cmd(info, CMD_ENABLESEL); + switch (info->scsi.phase) { case PHASE_SELECTION: /* while selecting - no target */ case PHASE_SELSTEPS: @@ -817,7 +973,6 @@ break; case PHASE_MSGIN_DISCONNECT: /* message in - disconnecting */ - outb(CMD_ENABLESEL, REG_CMD(info)); info->scsi.disconnectable = 1; info->scsi.reconnected.tag = 0; info->scsi.phase = PHASE_IDLE; @@ -845,9 +1000,11 @@ } } -/* Function: void fas216_reselected_intr(FAS216_Info *info) - * Purpose : Start reconnection of a device - * Params : info - interface which was reselected +/** + * fas216_reselected_intr - start reconnection of a device + * @info: interface which was reselected + * + * Start reconnection of a device */ static void fas216_reselected_intr(FAS216_Info *info) @@ -869,24 +1026,16 @@ info->device[SCpnt->target].sync_state = neg_wait; } -#ifdef DEBUG_CONNECT - printk("scsi%d.%c: reconnect phase=%02X\n", info->host->host_no, - fas216_target(info), info->scsi.phase); -#endif + fas216_log(info, LOG_CONNECT, "reconnect phase=%02X", info->scsi.phase); - if ((inb(REG_CFIS(info)) & CFIS_CF) != 2) { + if ((fas216_readb(info, REG_CFIS) & CFIS_CF) != 2) { printk(KERN_ERR "scsi%d.H: incorrect number of bytes after reselect\n", info->host->host_no); - outb(CMD_SETATN, REG_CMD(info)); - outb(CMD_MSGACCEPTED, REG_CMD(info)); - msgqueue_flush(&info->scsi.msgs); - msgqueue_addmsg(&info->scsi.msgs, 1, INITIATOR_ERROR); - info->scsi.phase = PHASE_MSGOUT_EXPECT; - return; + goto initiator_error; } - target = inb(REG_FF(info)); - identify_msg = inb(REG_FF(info)); + target = fas216_readb(info, REG_FF); + identify_msg = fas216_readb(info, REG_FF); ok = 1; if (!(target & (1 << info->host->this_id))) { @@ -905,32 +1054,20 @@ * Something went wrong - send an initiator error to * the target. */ - outb(CMD_SETATN, REG_CMD(info)); - outb(CMD_MSGACCEPTED, REG_CMD(info)); - msgqueue_flush(&info->scsi.msgs); - msgqueue_addmsg(&info->scsi.msgs, 1, INITIATOR_ERROR); - info->scsi.phase = PHASE_MSGOUT_EXPECT; - return; + goto initiator_error; } target &= ~(1 << info->host->this_id); - switch (target) { - case 1: target = 0; break; - case 2: target = 1; break; - case 4: target = 2; break; - case 8: target = 3; break; - case 16: target = 4; break; - case 32: target = 5; break; - case 64: target = 6; break; - case 128: target = 7; break; - default: target = info->host->this_id; break; - } + target = ffs(target) - 1; identify_msg &= 7; info->scsi.reconnected.target = target; info->scsi.reconnected.lun = identify_msg; info->scsi.reconnected.tag = 0; + /* set up for synchronous transfers */ + fas216_set_sync(info, target); + ok = 0; if (info->scsi.disconnectable && info->SCpnt && info->SCpnt->target == target && info->SCpnt->lun == identify_msg) @@ -942,7 +1079,7 @@ msgqueue_flush(&info->scsi.msgs); if (ok) { info->scsi.phase = PHASE_RECONNECTED; - outb(target, REG_SDID(info)); + fas216_writeb(info, REG_SDID, target); } else { /* * Our command structure not found - abort the @@ -950,45 +1087,47 @@ * record of this command, we can't send * an INITIATOR DETECTED ERROR message. */ - outb(CMD_SETATN, REG_CMD(info)); + fas216_cmd(info, CMD_SETATN); msgqueue_addmsg(&info->scsi.msgs, 1, ABORT); info->scsi.phase = PHASE_MSGOUT_EXPECT; } - outb(CMD_MSGACCEPTED, REG_CMD(info)); + fas216_cmd(info, CMD_MSGACCEPTED); + return; + + initiator_error: + fas216_cmd(info, CMD_SETATN); + msgqueue_flush(&info->scsi.msgs); + msgqueue_addmsg(&info->scsi.msgs, 1, INITIATOR_ERROR); + info->scsi.phase = PHASE_MSGOUT_EXPECT; + fas216_cmd(info, CMD_MSGACCEPTED); } -/* Function: void fas216_finish_reconnect(FAS216_Info *info) - * Purpose : finish reconnection sequence for device - * Params : info - interface which caused function done interrupt +/** + * fas216_finish_reconnect - finish reconnection sequence for device + * @info: interface which caused function done interrupt + * + * Finish reconnection sequence for device */ static void fas216_finish_reconnect(FAS216_Info *info) { fas216_checkmagic(info); -#ifdef DEBUG_CONNECT - printk("Connected: %1X %1X %02X, reconnected: %1X %1X %02X\n", + fas216_log(info, LOG_CONNECT, "Connected: %1x %1x %02x, reconnected: %1x %1x %02x", info->SCpnt->target, info->SCpnt->lun, info->SCpnt->tag, info->scsi.reconnected.target, info->scsi.reconnected.lun, info->scsi.reconnected.tag); -#endif if (info->scsi.disconnectable && info->SCpnt) { info->scsi.disconnectable = 0; if (info->SCpnt->target == info->scsi.reconnected.target && info->SCpnt->lun == info->scsi.reconnected.lun && info->SCpnt->tag == info->scsi.reconnected.tag) { -#ifdef DEBUG_CONNECT - printk("scsi%d.%c: reconnected", - info->host->host_no, fas216_target(info)); -#endif + fas216_log(info, LOG_CONNECT, "reconnected"); } else { queue_add_cmd_tail(&info->queues.disconnected, info->SCpnt); -#ifdef DEBUG_CONNECT - printk("scsi%d.%c: had to move command to disconnected queue\n", - info->host->host_no, fas216_target(info)); -#endif + fas216_log(info, LOG_CONNECT, "had to move command to disconnected queue"); info->SCpnt = NULL; } } @@ -997,20 +1136,19 @@ info->scsi.reconnected.target, info->scsi.reconnected.lun, info->scsi.reconnected.tag); -#ifdef DEBUG_CONNECT - printk("scsi%d.%c: had to get command", - info->host->host_no, fas216_target(info)); -#endif + fas216_log(info, LOG_CONNECT, "had to get command"); } if (!info->SCpnt) { - outb(CMD_SETATN, REG_CMD(info)); + fas216_cmd(info, CMD_SETATN); msgqueue_flush(&info->scsi.msgs); + #if 0 if (info->scsi.reconnected.tag) msgqueue_addmsg(&info->scsi.msgs, 2, ABORT_TAG, info->scsi.reconnected.tag); else #endif msgqueue_addmsg(&info->scsi.msgs, 1, ABORT); + info->scsi.phase = PHASE_MSGOUT_EXPECT; info->scsi.aborting = 1; } else { @@ -1018,84 +1156,210 @@ * Restore data pointer from SAVED data pointer */ info->scsi.SCp = info->SCpnt->SCp; -#ifdef DEBUG_CONNECT - printk(", data pointers: [%p, %X]", + fas216_log(info, LOG_CONNECT, "data pointers: [%p, %X]", info->scsi.SCp.ptr, info->scsi.SCp.this_residual); -#endif } -#ifdef DEBUG_CONNECT - printk("\n"); -#endif } -static int fas216_wait_cmd(FAS216_Info *info, int cmd) +static void fas216_parse_message(FAS216_Info *info, unsigned char *message, int msglen) { - int tout; - int stat; + int i; - outb(cmd, REG_CMD(info)); + switch (message[0]) { + case COMMAND_COMPLETE: + if (msglen != 1) + goto unrecognised; - for (tout = 1000; tout; tout -= 1) { - stat = inb(REG_STAT(info)); - if (stat & STAT_INT) - break; - udelay(1); - } + printk(KERN_ERR "scsi%d.%c: command complete with no " + "status in MESSAGE_IN?\n", + info->host->host_no, fas216_target(info)); + break; - return stat; -} + case SAVE_POINTERS: + if (msglen != 1) + goto unrecognised; -static int fas216_get_msg_byte(FAS216_Info *info) -{ - int stat; + /* + * Save current data pointer to SAVED data pointer + * SCSI II standard says that we must not acknowledge + * this until we have really saved pointers. + * NOTE: we DO NOT save the command nor status pointers + * as required by the SCSI II standard. These always + * point to the start of their respective areas. + */ + info->SCpnt->SCp = info->scsi.SCp; + info->SCpnt->SCp.sent_command = 0; + fas216_log(info, LOG_CONNECT | LOG_MESSAGES | LOG_BUFFER, + "save data pointers: [%p, %X]", + info->scsi.SCp.ptr, info->scsi.SCp.this_residual); + break; - stat = fas216_wait_cmd(info, CMD_MSGACCEPTED); + case RESTORE_POINTERS: + if (msglen != 1) + goto unrecognised; - if ((stat & STAT_INT) == 0) - goto timedout; + /* + * Restore current data pointer from SAVED data pointer + */ + info->scsi.SCp = info->SCpnt->SCp; + fas216_log(info, LOG_CONNECT | LOG_MESSAGES | LOG_BUFFER, + "restore data pointers: [%p, 0x%x]", + info->scsi.SCp.ptr, info->scsi.SCp.this_residual); + break; - if ((stat & STAT_BUSMASK) != STAT_MESGIN) - goto unexpected_phase_change; + case DISCONNECT: + if (msglen != 1) + goto unrecognised; - inb(REG_INST(info)); + info->scsi.phase = PHASE_MSGIN_DISCONNECT; + break; - stat = fas216_wait_cmd(info, CMD_TRANSFERINFO); + case MESSAGE_REJECT: + if (msglen != 1) + goto unrecognised; - if ((stat & STAT_INT) == 0) - goto timedout; + switch (fas216_get_last_msg(info, info->scsi.msgin_fifo)) { + case EXTENDED_MESSAGE | EXTENDED_SDTR << 8: + fas216_handlesync(info, message); + break; - if ((stat & STAT_BUSMASK) != STAT_MESGIN) - goto unexpected_phase_change; + case EXTENDED_MESSAGE | EXTENDED_WDTR << 8: + fas216_handlewide(info, message); + break; - inb(REG_INST(info)); + default: + fas216_log(info, 0, "reject, last message 0x%04x", + fas216_get_last_msg(info, info->scsi.msgin_fifo)); + } + break; - return inb(REG_FF(info)); + case NOP: + break; -timedout: - printk("scsi%d.%c: timed out waiting for message byte\n", - info->host->host_no, fas216_target(info)); - return -1; + case SIMPLE_QUEUE_TAG: + if (msglen < 2) + goto unrecognised; -unexpected_phase_change: - printk("scsi%d.%c: unexpected phase change: status = %02X\n", - info->host->host_no, fas216_target(info), stat); + /* handled above - print a warning since this is untested */ + fas216_log(info, 0, "reconnect queue tag 0x%02x", message[1]); + break; + + case EXTENDED_MESSAGE: + if (msglen < 3) + goto unrecognised; + + switch (message[2]) { + case EXTENDED_SDTR: /* Sync transfer negotiation request/reply */ + fas216_handlesync(info, message); + break; + + case EXTENDED_WDTR: /* Wide transfer negotiation request/reply */ + fas216_handlewide(info, message); + break; + + default: + goto unrecognised; + } + break; + + default: + goto unrecognised; + } + return; + +unrecognised: + fas216_log(info, 0, "unrecognised message, rejecting"); + printk("scsi%d.%c: message was", info->host->host_no, fas216_target(info)); + for (i = 0; i < msglen; i++) + printk("%s%02X", i & 31 ? " " : "\n ", message[i]); + printk("\n"); + /* + * Something strange seems to be happening here - + * I can't use SETATN since the chip gives me an + * invalid command interrupt when I do. Weird. + */ +fas216_cmd(info, CMD_NOP); +fas216_dumpstate(info); + fas216_cmd(info, CMD_SETATN); + msgqueue_flush(&info->scsi.msgs); + msgqueue_addmsg(&info->scsi.msgs, 1, MESSAGE_REJECT); + info->scsi.phase = PHASE_MSGOUT_EXPECT; +fas216_dumpstate(info); +} + +static int fas216_wait_cmd(FAS216_Info *info, int cmd) +{ + int tout; + int stat; + + fas216_cmd(info, cmd); + + for (tout = 1000; tout; tout -= 1) { + stat = fas216_readb(info, REG_STAT); + if (stat & (STAT_INT|STAT_PARITYERROR)) + break; + udelay(1); + } + + return stat; +} + +static int fas216_get_msg_byte(FAS216_Info *info) +{ + unsigned int stat = fas216_wait_cmd(info, CMD_MSGACCEPTED); + + if ((stat & STAT_INT) == 0) + goto timedout; + + if ((stat & STAT_BUSMASK) != STAT_MESGIN) + goto unexpected_phase_change; + + fas216_readb(info, REG_INST); + + stat = fas216_wait_cmd(info, CMD_TRANSFERINFO); + + if ((stat & STAT_INT) == 0) + goto timedout; + + if (stat & STAT_PARITYERROR) + goto parity_error; + + if ((stat & STAT_BUSMASK) != STAT_MESGIN) + goto unexpected_phase_change; + + fas216_readb(info, REG_INST); + + return fas216_readb(info, REG_FF); + +timedout: + fas216_log(info, LOG_ERROR, "timed out waiting for message byte"); + return -1; + +unexpected_phase_change: + fas216_log(info, LOG_ERROR, "unexpected phase change: status = %02x", stat); return -2; + +parity_error: + fas216_log(info, LOG_ERROR, "parity error during message in phase"); + return -3; } -/* Function: void fas216_message(FAS216_Info *info) - * Purpose : handle a function done interrupt from FAS216 chip - * Params : info - interface which caused function done interrupt +/** + * fas216_message - handle a function done interrupt from FAS216 chip + * @info: interface which caused function done interrupt + * + * Handle a function done interrupt from FAS216 chip */ static void fas216_message(FAS216_Info *info) { unsigned char *message = info->scsi.message; - unsigned int msglen = 1, i; + unsigned int msglen = 1; int msgbyte = 0; fas216_checkmagic(info); - message[0] = inb(REG_FF(info)); + message[0] = fas216_readb(info, REG_FF); if (message[0] == EXTENDED_MESSAGE) { msgbyte = fas216_get_msg_byte(info); @@ -1114,7 +1378,8 @@ } } - info->scsi.msglen = msglen; + if (msgbyte == -3) + goto parity_error; #ifdef DEBUG_MESSAGES { @@ -1135,142 +1400,24 @@ info->scsi.phase = PHASE_MSGIN; } - switch (message[0]) { - case COMMAND_COMPLETE: - if (msglen != 1) - goto unrecognised; - - printk(KERN_ERR "scsi%d.%c: command complete with no " - "status in MESSAGE_IN?\n", - info->host->host_no, fas216_target(info)); - break; - - case SAVE_POINTERS: - if (msglen != 1) - goto unrecognised; - - /* - * Save current data pointer to SAVED data pointer - * SCSI II standard says that we must not acknowledge - * this until we have really saved pointers. - * NOTE: we DO NOT save the command nor status pointers - * as required by the SCSI II standard. These always - * point to the start of their respective areas. - */ - info->SCpnt->SCp = info->scsi.SCp; - info->SCpnt->SCp.sent_command = 0; -#if defined (DEBUG_MESSAGES) || defined (DEBUG_CONNECT) - printk("scsi%d.%c: save data pointers: [%p, %X]\n", - info->host->host_no, fas216_target(info), - info->scsi.SCp.ptr, info->scsi.SCp.this_residual); -#endif - break; - - case RESTORE_POINTERS: - if (msglen != 1) - goto unrecognised; - - /* - * Restore current data pointer from SAVED data pointer - */ - info->scsi.SCp = info->SCpnt->SCp; -#if defined (DEBUG_MESSAGES) || defined (DEBUG_CONNECT) - printk("scsi%d.%c: restore data pointers: [%p, %X]\n", - info->host->host_no, fas216_target(info), - info->scsi.SCp.ptr, info->scsi.SCp.this_residual); -#endif - break; - - case DISCONNECT: - if (msglen != 1) - goto unrecognised; - - info->scsi.phase = PHASE_MSGIN_DISCONNECT; - break; - - case MESSAGE_REJECT: - if (msglen != 1) - goto unrecognised; - - switch (fas216_get_last_msg(info, info->scsi.msgin_fifo)) { - case EXTENDED_MESSAGE | EXTENDED_SDTR << 8: - fas216_handlesync(info, message); - break; - - case EXTENDED_MESSAGE | EXTENDED_WDTR << 8: - fas216_handlewide(info, message); - break; - - default: - printk("scsi%d.%c: reject, last message %04X\n", - info->host->host_no, fas216_target(info), - fas216_get_last_msg(info, info->scsi.msgin_fifo)); - } - break; - - case NOP: - break; - - case SIMPLE_QUEUE_TAG: - if (msglen < 2) - goto unrecognised; - - /* handled above - print a warning since this is untested */ - printk("scsi%d.%c: reconnect queue tag %02X\n", - info->host->host_no, fas216_target(info), - message[1]); - break; - - case EXTENDED_MESSAGE: - if (msglen < 3) - goto unrecognised; - - switch (message[2]) { - case EXTENDED_SDTR: /* Sync transfer negociation request/reply */ - fas216_handlesync(info, message); - break; - - case EXTENDED_WDTR: /* Wide transfer negociation request/reply */ - fas216_handlewide(info, message); - break; - - default: - goto unrecognised; - } - break; - - default: - goto unrecognised; - } - outb(CMD_MSGACCEPTED, REG_CMD(info)); + fas216_parse_message(info, message, msglen); + fas216_cmd(info, CMD_MSGACCEPTED); return; -unrecognised: - printk("scsi%d.%c: unrecognised message, rejecting\n", - info->host->host_no, fas216_target(info)); - printk("scsi%d.%c: message was", info->host->host_no, fas216_target(info)); - for (i = 0; i < msglen; i++) - printk("%s%02X", i & 31 ? " " : "\n ", message[i]); - printk("\n"); - - /* - * Something strange seems to be happening here - - * I can't use SETATN since the chip gives me an - * invalid command interrupt when I do. Weird. - */ -outb(CMD_NOP, REG_CMD(info)); -fas216_dumpstate(info); - outb(CMD_SETATN, REG_CMD(info)); +parity_error: + fas216_cmd(info, CMD_SETATN); msgqueue_flush(&info->scsi.msgs); - msgqueue_addmsg(&info->scsi.msgs, 1, MESSAGE_REJECT); + msgqueue_addmsg(&info->scsi.msgs, 1, MSG_PARITY_ERROR); info->scsi.phase = PHASE_MSGOUT_EXPECT; -fas216_dumpstate(info); - outb(CMD_MSGACCEPTED, REG_CMD(info)); + fas216_cmd(info, CMD_MSGACCEPTED); + return; } -/* Function: void fas216_send_command(FAS216_Info *info) - * Purpose : send a command to a target after all message bytes have been sent - * Params : info - interface which caused bus service +/** + * fas216_send_command - send command after all message bytes have been sent + * @info: interface which caused bus service + * + * Send a command to a target after all message bytes have been sent */ static void fas216_send_command(FAS216_Info *info) { @@ -1278,22 +1425,24 @@ fas216_checkmagic(info); - outb(CMD_NOP|CMD_WITHDMA, REG_CMD(info)); - outb(CMD_FLUSHFIFO, REG_CMD(info)); + fas216_cmd(info, CMD_NOP|CMD_WITHDMA); + fas216_cmd(info, CMD_FLUSHFIFO); /* load command */ for (i = info->scsi.SCp.sent_command; i < info->SCpnt->cmd_len; i++) - outb(info->SCpnt->cmnd[i], REG_FF(info)); + fas216_writeb(info, REG_FF, info->SCpnt->cmnd[i]); - outb(CMD_TRANSFERINFO, REG_CMD(info)); + fas216_cmd(info, CMD_TRANSFERINFO); info->scsi.phase = PHASE_COMMAND; } -/* Function: void fas216_send_messageout(FAS216_Info *info, int start) - * Purpose : handle bus service to send a message - * Params : info - interface which caused bus service - * Note : We do not allow the device to change the data direction! +/** + * fas216_send_messageout - handle bus service to send a message + * @info: interface which caused bus service + * + * Handle bus service to send a message. + * Note: We do not allow the device to change the data direction! */ static void fas216_send_messageout(FAS216_Info *info, int start) { @@ -1301,7 +1450,7 @@ fas216_checkmagic(info); - outb(CMD_FLUSHFIFO, REG_CMD(info)); + fas216_cmd(info, CMD_FLUSHFIFO); if (tot_msglen) { struct message *msg; @@ -1311,37 +1460,38 @@ int i; for (i = start; i < msg->length; i++) - outb(msg->msg[i], REG_FF(info)); + fas216_writeb(info, REG_FF, msg->msg[i]); - msg->fifo = tot_msglen - (inb(REG_CFIS(info)) & CFIS_CF); + msg->fifo = tot_msglen - (fas216_readb(info, REG_CFIS) & CFIS_CF); start = 0; } } else - outb(NOP, REG_FF(info)); + fas216_writeb(info, REG_FF, NOP); - outb(CMD_TRANSFERINFO, REG_CMD(info)); + fas216_cmd(info, CMD_TRANSFERINFO); info->scsi.phase = PHASE_MSGOUT; } -/* Function: void fas216_busservice_intr(FAS216_Info *info, unsigned int stat, unsigned int ssr) - * Purpose : handle a bus service interrupt from FAS216 chip - * Params : info - interface which caused bus service interrupt - * stat - Status register contents - * ssr - SCSI Status register contents +/** + * fas216_busservice_intr - handle bus service interrupt from FAS216 chip + * @info: interface which caused bus service interrupt + * @stat: Status register contents + * @ssr: SCSI Status register contents + * + * Handle a bus service interrupt from FAS216 chip */ static void fas216_busservice_intr(FAS216_Info *info, unsigned int stat, unsigned int ssr) { fas216_checkmagic(info); -#ifdef DEBUG_BUSSERVICE - printk("scsi%d.%c: bus service: stat=%02X ssr=%02X phase=%02X\n", - info->host->host_no, fas216_target(info), stat, ssr, info->scsi.phase); -#endif + fas216_log(info, LOG_BUSSERVICE, + "bus service: stat=%02x ssr=%02x phase=%02x", + stat, ssr, info->scsi.phase); switch (info->scsi.phase) { case PHASE_SELECTION: - if ((ssr & IS_BITS) != 1) + if ((ssr & IS_BITS) != IS_MSGBYTESENT) goto bad_is; break; @@ -1365,7 +1515,7 @@ break; } - outb(CMD_NOP, REG_CMD(info)); + fas216_cmd(info, CMD_NOP); #define STATE(st,ph) ((ph) << 3 | (st)) /* This table describes the legal SCSI state transitions, @@ -1376,15 +1526,17 @@ case STATE(STAT_DATAIN, PHASE_RECONNECTED): fas216_finish_reconnect(info); case STATE(STAT_DATAIN, PHASE_SELSTEPS):/* Sel w/ steps -> Data In */ - case STATE(STAT_DATAIN, PHASE_DATAIN): /* Data In -> Data In */ case STATE(STAT_DATAIN, PHASE_MSGOUT): /* Message Out -> Data In */ case STATE(STAT_DATAIN, PHASE_COMMAND): /* Command -> Data In */ case STATE(STAT_DATAIN, PHASE_MSGIN): /* Message In -> Data In */ - fas216_starttransfer(info, DMA_IN, 0); + info->scsi.phase = PHASE_DATAIN; + fas216_transfer(info); return; + case STATE(STAT_DATAIN, PHASE_DATAIN): /* Data In -> Data In */ case STATE(STAT_DATAOUT, PHASE_DATAOUT):/* Data Out -> Data Out */ - fas216_starttransfer(info, DMA_OUT, 0); + fas216_cleanuptransfer(info); + fas216_transfer(info); return; /* Reselmsgin -> Data Out */ @@ -1394,7 +1546,9 @@ case STATE(STAT_DATAOUT, PHASE_MSGOUT): /* Message Out -> Data Out */ case STATE(STAT_DATAOUT, PHASE_COMMAND):/* Command -> Data Out */ case STATE(STAT_DATAOUT, PHASE_MSGIN): /* Message In -> Data Out */ - fas216_starttransfer(info, DMA_OUT, 1); + fas216_cmd(info, CMD_FLUSHFIFO); + info->scsi.phase = PHASE_DATAOUT; + fas216_transfer(info); return; /* Reselmsgin -> Status */ @@ -1409,7 +1563,7 @@ case STATE(STAT_STATUS, PHASE_COMMAND): /* Command -> Status */ case STATE(STAT_STATUS, PHASE_MSGIN): /* Message In -> Status */ status: - outb(CMD_INITCMDCOMPLETE, REG_CMD(info)); + fas216_cmd(info, CMD_INITCMDCOMPLETE); info->scsi.phase = PHASE_STATUS; return; @@ -1419,17 +1573,17 @@ case STATE(STAT_MESGIN, PHASE_COMMAND): /* Command -> Message In */ case STATE(STAT_MESGIN, PHASE_SELSTEPS):/* Sel w/ steps -> Message In */ case STATE(STAT_MESGIN, PHASE_MSGOUT): /* Message Out -> Message In */ - info->scsi.msgin_fifo = inb(REG_CFIS(info)) & CFIS_CF; - outb(CMD_FLUSHFIFO, REG_CMD(info)); - outb(CMD_TRANSFERINFO, REG_CMD(info)); + info->scsi.msgin_fifo = fas216_readb(info, REG_CFIS) & CFIS_CF; + fas216_cmd(info, CMD_FLUSHFIFO); + fas216_cmd(info, CMD_TRANSFERINFO); info->scsi.phase = PHASE_MSGIN; return; /* Reselmsgin -> Message In */ case STATE(STAT_MESGIN, PHASE_RECONNECTED): case STATE(STAT_MESGIN, PHASE_MSGIN): - info->scsi.msgin_fifo = inb(REG_CFIS(info)) & CFIS_CF; - outb(CMD_TRANSFERINFO, REG_CMD(info)); + info->scsi.msgin_fifo = fas216_readb(info, REG_CFIS) & CFIS_CF; + fas216_cmd(info, CMD_TRANSFERINFO); return; /* Reselmsgin -> Command */ @@ -1440,11 +1594,43 @@ fas216_send_command(info); info->scsi.phase = PHASE_COMMAND; return; - /* Selection -> Message Out */ + + + /* + * Selection -> Message Out + */ case STATE(STAT_MESGOUT, PHASE_SELECTION): fas216_send_messageout(info, 1); return; - /* Any -> Message Out */ + + /* + * Message Out -> Message Out + */ + case STATE(STAT_MESGOUT, PHASE_SELSTEPS): + case STATE(STAT_MESGOUT, PHASE_MSGOUT): + /* + * If we get another message out phase, this usually + * means some parity error occurred. Resend complete + * set of messages. If we have more than one byte to + * send, we need to assert ATN again. + */ + if (info->device[info->SCpnt->target].parity_check) { + /* + * We were testing... good, the device + * supports parity checking. + */ + info->device[info->SCpnt->target].parity_check = 0; + info->device[info->SCpnt->target].parity_enabled = 1; + fas216_writeb(info, REG_CNTL1, info->scsi.cfg[0]); + } + + if (msgqueue_msglength(&info->scsi.msgs) > 1) + fas216_cmd(info, CMD_SETATN); + /*FALLTHROUGH*/ + + /* + * Any -> Message Out + */ case STATE(STAT_MESGOUT, PHASE_MSGOUT_EXPECT): fas216_send_messageout(info, 0); return; @@ -1463,30 +1649,13 @@ printk(KERN_ERR "scsi%d.%c: " "target trying to receive more command bytes\n", info->host->host_no, fas216_target(info)); - outb(CMD_SETATN, REG_CMD(info)); - outb(15, REG_STCL(info)); - outb(0, REG_STCM(info)); - outb(0, REG_STCH(info)); - outb(CMD_PADBYTES | CMD_WITHDMA, REG_CMD(info)); + fas216_cmd(info, CMD_SETATN); + fas216_set_stc(info, 15); + fas216_cmd(info, CMD_PADBYTES | CMD_WITHDMA); msgqueue_flush(&info->scsi.msgs); msgqueue_addmsg(&info->scsi.msgs, 1, INITIATOR_ERROR); info->scsi.phase = PHASE_MSGOUT_EXPECT; return; - - /* Selection -> Message Out */ - case STATE(STAT_MESGOUT, PHASE_SELSTEPS): - case STATE(STAT_MESGOUT, PHASE_MSGOUT): /* Message Out -> Message Out */ - /* If we get another message out phase, this - * usually means some parity error occurred. - * Resend complete set of messages. If we have - * more than 1 byte to send, we need to assert - * ATN again. - */ - if (msgqueue_msglength(&info->scsi.msgs) > 1) - outb(CMD_SETATN, REG_CMD(info)); - - fas216_send_messageout(info, 0); - return; } if (info->scsi.phase == PHASE_MSGIN_DISCONNECT) { @@ -1494,11 +1663,11 @@ info->host->host_no, fas216_target(info), fas216_bus_phase(stat)); msgqueue_flush(&info->scsi.msgs); - outb(CMD_SETATN, REG_CMD(info)); + fas216_cmd(info, CMD_SETATN); msgqueue_addmsg(&info->scsi.msgs, 1, INITIATOR_ERROR); info->scsi.phase = PHASE_MSGOUT_EXPECT; info->scsi.aborting = 1; - outb(CMD_TRANSFERINFO, REG_CMD(info)); + fas216_cmd(info, CMD_TRANSFERINFO); return; } printk(KERN_ERR "scsi%d.%c: bus phase %s after %s?\n", @@ -1509,81 +1678,130 @@ return; bad_is: - printk("scsi%d.%c: bus service at step %d?\n", - info->host->host_no, fas216_target(info), - ssr & IS_BITS); + fas216_log(info, 0, "bus service at step %d?", ssr & IS_BITS); + fas216_dumpstate(info); print_debug_list(); fas216_done(info, DID_ERROR); } -/* Function: void fas216_funcdone_intr(FAS216_Info *info, unsigned int stat, unsigned int ssr) - * Purpose : handle a function done interrupt from FAS216 chip - * Params : info - interface which caused function done interrupt - * stat - Status register contents - * ssr - SCSI Status register contents +/** + * fas216_funcdone_intr - handle a function done interrupt from FAS216 chip + * @info: interface which caused function done interrupt + * @stat: Status register contents + * @ssr: SCSI Status register contents + * + * Handle a function done interrupt from FAS216 chip */ static void fas216_funcdone_intr(FAS216_Info *info, unsigned int stat, unsigned int ssr) { - int status, message; + unsigned int fifo_len = fas216_readb(info, REG_CFIS) & CFIS_CF; + unsigned int status, message; fas216_checkmagic(info); -#ifdef DEBUG_FUNCTIONDONE - printk("scsi%d.%c: function done: stat=%X ssr=%X phase=%02X\n", - info->host->host_no, fas216_target(info), stat, ssr, info->scsi.phase); -#endif + fas216_log(info, LOG_FUNCTIONDONE, + "function done: stat=%02x ssr=%02x phase=%02x", + stat, ssr, info->scsi.phase); + switch (info->scsi.phase) { case PHASE_STATUS: /* status phase - read status and msg */ - status = inb(REG_FF(info)); - message = inb(REG_FF(info)); + if (fifo_len != 2) { + fas216_log(info, 0, "odd number of bytes in FIFO: %d", fifo_len); + } + status = fas216_readb(info, REG_FF); + message = fas216_readb(info, REG_FF); info->scsi.SCp.Message = message; info->scsi.SCp.Status = status; info->scsi.phase = PHASE_DONE; - outb(CMD_MSGACCEPTED, REG_CMD(info)); + fas216_cmd(info, CMD_MSGACCEPTED); break; case PHASE_IDLE: /* reselected? */ case PHASE_MSGIN: /* message in phase */ case PHASE_RECONNECTED: /* reconnected command */ if ((stat & STAT_BUSMASK) == STAT_MESGIN) { - info->scsi.msgin_fifo = inb(REG_CFIS(info)) & CFIS_CF; + info->scsi.msgin_fifo = fifo_len; fas216_message(info); break; } default: - printk("scsi%d.%c: internal phase %s for function done?" - " What do I do with this?\n", - info->host->host_no, fas216_target(info), - fas216_drv_phase(info)); + fas216_log(info, 0, "internal phase %s for function done?" + " What do I do with this?", + fas216_target(info), fas216_drv_phase(info)); } } -/* Function: void fas216_intr(struct Scsi_Host *instance) - * Purpose : handle interrupts from the interface to progress a command - * Params : instance - interface to service +static void fas216_bus_reset(FAS216_Info *info) +{ + neg_t sync_state, wide_state; + int i; + + msgqueue_flush(&info->scsi.msgs); + + info->scsi.reconnected.target = 0; + info->scsi.reconnected.lun = 0; + info->scsi.reconnected.tag = 0; + + wide_state = neg_invalid; + sync_state = neg_invalid; + +#ifdef SCSI2_WIDE + if (info->ifcfg.wide_max_size != 0) + wide_state = neg_wait; +#endif +#ifdef SCSI2_SYNC + if (info->ifcfg.capabilities & (FASCAP_DMA|FASCAP_PSEUDODMA)) + sync_state = neg_wait; +#endif + + info->scsi.phase = PHASE_IDLE; + info->SCpnt = NULL; /* bug! */ + memset(&info->scsi.SCp, 0, sizeof(info->scsi.SCp)); + + for (i = 0; i < 8; i++) { + info->device[i].disconnect_ok = info->ifcfg.disconnect_ok; + info->device[i].sync_state = sync_state; + info->device[i].wide_state = wide_state; + info->device[i].period = info->ifcfg.asyncperiod / 4; + info->device[i].stp = info->scsi.async_stp; + info->device[i].sof = 0; + info->device[i].wide_xfer = 0; + } + + info->rst_bus_status = 1; + wake_up(&info->eh_wait); +} + +/** + * fas216_intr - handle interrupts to progress a command + * @host: interface to service + * + * Handle interrupts from the interface to progress a command */ -void fas216_intr(struct Scsi_Host *instance) +void fas216_intr(struct Scsi_Host *host) { - FAS216_Info *info = (FAS216_Info *)instance->hostdata; + FAS216_Info *info = (FAS216_Info *)host->hostdata; unsigned char isr, ssr, stat; fas216_checkmagic(info); - stat = inb(REG_STAT(info)); - ssr = inb(REG_IS(info)); - isr = inb(REG_INST(info)); + stat = fas216_readb(info, REG_STAT); + ssr = fas216_readb(info, REG_IS); + isr = fas216_readb(info, REG_INST); add_debug_list(stat, ssr, isr, info->scsi.phase); if (stat & STAT_INT) { if (isr & INST_BUSRESET) { - printk(KERN_DEBUG "scsi%d.H: bus reset detected\n", instance->host_no); - scsi_report_bus_reset(instance, 0); + fas216_log(info, 0, "bus reset detected"); + fas216_bus_reset(info); + scsi_report_bus_reset(info->host, 0); } else if (isr & INST_ILLEGALCMD) { - printk(KERN_CRIT "scsi%d.H: illegal command given\n", instance->host_no); + fas216_log(info, LOG_ERROR, "illegal command given\n"); fas216_dumpstate(info); + print_debug_list(); } else if (isr & INST_DISCONNECT) fas216_disconnect_intr(info); else if (isr & INST_RESELECTED) /* reselected */ @@ -1593,107 +1811,120 @@ else if (isr & INST_FUNCDONE) /* function done */ fas216_funcdone_intr(info, stat, ssr); else - printk("scsi%d.%c: unknown interrupt received:" - " phase %s isr %02X ssr %02X stat %02X\n", - instance->host_no, fas216_target(info), + fas216_log(info, 0, "unknown interrupt received:" + " phase %s isr %02X ssr %02X stat %02X", fas216_drv_phase(info), isr, ssr, stat); } } -/* Function: void fas216_kick(FAS216_Info *info) - * Purpose : kick a command to the interface - interface should be idle - * Params : info - our host interface to kick - * Notes : Interrupts are always disabled! - */ -static void fas216_kick(FAS216_Info *info) +static void __fas216_start_command(FAS216_Info *info, Scsi_Cmnd *SCpnt) { - Scsi_Cmnd *SCpnt = NULL; - int tot_msglen, from_queue = 0, disconnect_ok; + int tot_msglen; - fas216_checkmagic(info); + /* following what the ESP driver says */ + fas216_set_stc(info, 0); + fas216_cmd(info, CMD_NOP | CMD_WITHDMA); - /* - * Obtain the next command to process. - */ - do { - if (info->reqSCpnt) { - SCpnt = info->reqSCpnt; - info->reqSCpnt = NULL; - break; - } + /* flush FIFO */ + fas216_cmd(info, CMD_FLUSHFIFO); - if (info->origSCpnt) { - SCpnt = info->origSCpnt; - info->origSCpnt = NULL; - break; + /* load bus-id and timeout */ + fas216_writeb(info, REG_SDID, BUSID(SCpnt->target)); + fas216_writeb(info, REG_STIM, info->ifcfg.select_timeout); + + /* synchronous transfers */ + fas216_set_sync(info, SCpnt->target); + + tot_msglen = msgqueue_msglength(&info->scsi.msgs); + +#ifdef DEBUG_MESSAGES + { + struct message *msg; + int msgnr = 0, i; + + printk("scsi%d.%c: message out: ", + info->host->host_no, '0' + SCpnt->target); + while ((msg = msgqueue_getmsg(&info->scsi.msgs, msgnr++)) != NULL) { + printk("{ "); + for (i = 0; i < msg->length; i++) + printk("%02x ", msg->msg[i]); + printk("} "); } + printk("\n"); + } +#endif - /* retrieve next command */ - if (!SCpnt) { - SCpnt = queue_remove_exclude(&info->queues.issue, - info->busyluns); - from_queue = 1; - break; + if (tot_msglen == 1 || tot_msglen == 3) { + /* + * We have an easy message length to send... + */ + struct message *msg; + int msgnr = 0, i; + + info->scsi.phase = PHASE_SELSTEPS; + + /* load message bytes */ + while ((msg = msgqueue_getmsg(&info->scsi.msgs, msgnr++)) != NULL) { + for (i = 0; i < msg->length; i++) + fas216_writeb(info, REG_FF, msg->msg[i]); + msg->fifo = tot_msglen - (fas216_readb(info, REG_CFIS) & CFIS_CF); } - } while (0); - if (!SCpnt) /* no command pending - just exit */ - return; + /* load command */ + for (i = 0; i < SCpnt->cmd_len; i++) + fas216_writeb(info, REG_FF, SCpnt->cmnd[i]); - if (info->scsi.disconnectable && info->SCpnt) { - queue_add_cmd_tail(&info->queues.disconnected, info->SCpnt); - info->scsi.disconnectable = 0; - info->SCpnt = NULL; - printk("scsi%d.%c: moved command to disconnected queue\n", - info->host->host_no, fas216_target(info)); + if (tot_msglen == 1) + fas216_cmd(info, CMD_SELECTATN); + else + fas216_cmd(info, CMD_SELECTATN3); + } else { + /* + * We have an unusual number of message bytes to send. + * Load first byte into fifo, and issue SELECT with ATN and + * stop steps. + */ + struct message *msg = msgqueue_getmsg(&info->scsi.msgs, 0); + + fas216_writeb(info, REG_FF, msg->msg[0]); + msg->fifo = 1; + + fas216_cmd(info, CMD_SELECTATNSTOP); } +} + +/* + * Decide whether we need to perform a parity test on this device. + * Can also be used to force parity error conditions during initial + * information transfer phase (message out) for test purposes. + */ +static int parity_test(FAS216_Info *info, int target) +{ +#if 0 + if (target == 3) { + info->device[3].parity_check = 0; + return 1; + } +#endif + return info->device[target].parity_check; +} + +static void fas216_start_command(FAS216_Info *info, Scsi_Cmnd *SCpnt) +{ + int disconnect_ok; /* * claim host busy */ info->scsi.phase = PHASE_SELECTION; - info->SCpnt = SCpnt; info->scsi.SCp = SCpnt->SCp; + info->SCpnt = SCpnt; info->dma.transfer_type = fasdma_none; -#ifdef DEBUG_CONNECT - printk("scsi%d.%c: starting cmd %02X", - info->host->host_no, '0' + SCpnt->target, - SCpnt->cmnd[0]); -#endif - - if (from_queue) { -#ifdef SCSI2_TAG - /* - * tagged queuing - allocate a new tag to this command - */ - if (SCpnt->device->tagged_queue && SCpnt->cmnd[0] != REQUEST_SENSE && - SCpnt->cmnd[0] != INQUIRY) { - SCpnt->device->current_tag += 1; - if (SCpnt->device->current_tag == 0) - SCpnt->device->current_tag = 1; - SCpnt->tag = SCpnt->device->current_tag; - } else -#endif - set_bit(SCpnt->target * 8 + SCpnt->lun, info->busyluns); - - info->stats.removes += 1; - switch (SCpnt->cmnd[0]) { - case WRITE_6: - case WRITE_10: - case WRITE_12: - info->stats.writes += 1; - break; - case READ_6: - case READ_10: - case READ_12: - info->stats.reads += 1; - break; - default: - info->stats.miscs += 1; - break; - } - } + if (parity_test(info, SCpnt->target)) + fas216_writeb(info, REG_CNTL1, info->scsi.cfg[0] | CNTL1_PTE); + else + fas216_writeb(info, REG_CNTL1, info->scsi.cfg[0]); /* * Don't allow request sense commands to disconnect. @@ -1738,100 +1969,192 @@ #endif } while (0); + __fas216_start_command(info, SCpnt); +} + +static void fas216_allocate_tag(FAS216_Info *info, Scsi_Cmnd *SCpnt) +{ +#ifdef SCSI2_TAG + /* + * tagged queuing - allocate a new tag to this command + */ + if (SCpnt->device->tagged_queue && SCpnt->cmnd[0] != REQUEST_SENSE && + SCpnt->cmnd[0] != INQUIRY) { + SCpnt->device->current_tag += 1; + if (SCpnt->device->current_tag == 0) + SCpnt->device->current_tag = 1; + SCpnt->tag = SCpnt->device->current_tag; + } else +#endif + set_bit(SCpnt->target * 8 + SCpnt->lun, info->busyluns); + + info->stats.removes += 1; + switch (SCpnt->cmnd[0]) { + case WRITE_6: + case WRITE_10: + case WRITE_12: + info->stats.writes += 1; + break; + case READ_6: + case READ_10: + case READ_12: + info->stats.reads += 1; + break; + default: + info->stats.miscs += 1; + break; + } +} + +static void fas216_do_bus_device_reset(FAS216_Info *info, Scsi_Cmnd *SCpnt) +{ + struct message *msg; + + /* + * claim host busy + */ + info->scsi.phase = PHASE_SELECTION; + info->scsi.SCp = SCpnt->SCp; + info->SCpnt = SCpnt; + info->dma.transfer_type = fasdma_none; + + fas216_log(info, LOG_ERROR, "sending bus device reset"); + + msgqueue_flush(&info->scsi.msgs); + msgqueue_addmsg(&info->scsi.msgs, 1, BUS_DEVICE_RESET); + /* following what the ESP driver says */ - outb(0, REG_STCL(info)); - outb(0, REG_STCM(info)); - outb(0, REG_STCH(info)); - outb(CMD_NOP | CMD_WITHDMA, REG_CMD(info)); + fas216_set_stc(info, 0); + fas216_cmd(info, CMD_NOP | CMD_WITHDMA); /* flush FIFO */ - outb(CMD_FLUSHFIFO, REG_CMD(info)); + fas216_cmd(info, CMD_FLUSHFIFO); /* load bus-id and timeout */ - outb(BUSID(SCpnt->target), REG_SDID(info)); - outb(info->ifcfg.select_timeout, REG_STIM(info)); + fas216_writeb(info, REG_SDID, BUSID(SCpnt->target)); + fas216_writeb(info, REG_STIM, info->ifcfg.select_timeout); /* synchronous transfers */ fas216_set_sync(info, SCpnt->target); - tot_msglen = msgqueue_msglength(&info->scsi.msgs); + msg = msgqueue_getmsg(&info->scsi.msgs, 0); -#ifdef DEBUG_MESSAGES - { - struct message *msg; - int msgnr = 0, i; + fas216_writeb(info, REG_FF, BUS_DEVICE_RESET); + msg->fifo = 1; - printk("scsi%d.%c: message out: ", - info->host->host_no, '0' + SCpnt->target); - while ((msg = msgqueue_getmsg(&info->scsi.msgs, msgnr++)) != NULL) { - printk("{ "); - for (i = 0; i < msg->length; i++) - printk("%02x ", msg->msg[i]); - printk("} "); + fas216_cmd(info, CMD_SELECTATNSTOP); +} + +/** + * fas216_kick - kick a command to the interface + * @info: our host interface to kick + * + * Kick a command to the interface, interface should be idle. + * Notes: Interrupts are always disabled! + */ +static void fas216_kick(FAS216_Info *info) +{ + Scsi_Cmnd *SCpnt = NULL; +#define TYPE_OTHER 0 +#define TYPE_RESET 1 +#define TYPE_QUEUE 2 + int where_from = TYPE_OTHER; + + fas216_checkmagic(info); + + /* + * Obtain the next command to process. + */ + do { + if (info->rstSCpnt) { + SCpnt = info->rstSCpnt; + /* don't remove it */ + where_from = TYPE_RESET; + break; } - printk("\n"); - } -#endif - if (tot_msglen == 1 || tot_msglen == 3) { - /* - * We have an easy message length to send... - */ - struct message *msg; - int msgnr = 0, i; + if (info->reqSCpnt) { + SCpnt = info->reqSCpnt; + info->reqSCpnt = NULL; + break; + } - info->scsi.phase = PHASE_SELSTEPS; + if (info->origSCpnt) { + SCpnt = info->origSCpnt; + info->origSCpnt = NULL; + break; + } - /* load message bytes */ - while ((msg = msgqueue_getmsg(&info->scsi.msgs, msgnr++)) != NULL) { - for (i = 0; i < msg->length; i++) - outb(msg->msg[i], REG_FF(info)); - msg->fifo = tot_msglen - (inb(REG_CFIS(info)) & CFIS_CF); + /* retrieve next command */ + if (!SCpnt) { + SCpnt = queue_remove_exclude(&info->queues.issue, + info->busyluns); + where_from = TYPE_QUEUE; + break; } + } while (0); - /* load command */ - for (i = 0; i < SCpnt->cmd_len; i++) - outb(SCpnt->cmnd[i], REG_FF(info)); + if (!SCpnt) /* no command pending - just exit */ + return; - if (tot_msglen == 1) - outb(CMD_SELECTATN, REG_CMD(info)); - else - outb(CMD_SELECTATN3, REG_CMD(info)); - } else { - /* - * We have an unusual number of message bytes to send. - * Load first byte into fifo, and issue SELECT with ATN and - * stop steps. - */ - struct message *msg = msgqueue_getmsg(&info->scsi.msgs, 0); + if (info->scsi.disconnectable && info->SCpnt) { + fas216_log(info, LOG_CONNECT, + "moved command for %d to disconnected queue", + info->SCpnt->target); + queue_add_cmd_tail(&info->queues.disconnected, info->SCpnt); + info->scsi.disconnectable = 0; + info->SCpnt = NULL; + } - outb(msg->msg[0], REG_FF(info)); - msg->fifo = 1; + fas216_log_command(info, LOG_CONNECT | LOG_MESSAGES, SCpnt, "starting "); - outb(CMD_SELECTATNSTOP, REG_CMD(info)); + switch (where_from) { + case TYPE_QUEUE: + fas216_allocate_tag(info, SCpnt); + case TYPE_OTHER: + fas216_start_command(info, SCpnt); + break; + case TYPE_RESET: + fas216_do_bus_device_reset(info, SCpnt); + break; } -#ifdef DEBUG_CONNECT - printk(", data pointers [%p, %X]\n", + fas216_log(info, LOG_CONNECT, "select: data pointers [%p, %X]", info->scsi.SCp.ptr, info->scsi.SCp.this_residual); -#endif - /* should now get either DISCONNECT or (FUNCTION DONE with BUS SERVICE) intr */ + + /* + * should now get either DISCONNECT or + * (FUNCTION DONE with BUS SERVICE) interrupt + */ +} + +/* + * Clean up from issuing a BUS DEVICE RESET message to a device. + */ +static void +fas216_devicereset_done(FAS216_Info *info, Scsi_Cmnd *SCpnt, unsigned int result) +{ + fas216_log(info, LOG_ERROR, "fas216 device reset complete"); + + info->rstSCpnt = NULL; + info->rst_dev_status = 1; + wake_up(&info->eh_wait); } -/* Function: void fas216_rq_sns_done(info, SCpnt, result) - * Purpose : Finish processing automatic request sense command - * Params : info - interface that completed - * SCpnt - command that completed - * result - driver byte of result +/** + * fas216_rq_sns_done - Finish processing automatic request sense command + * @info: interface that completed + * @SCpnt: command that completed + * @result: driver byte of result + * + * Finish processing automatic request sense command */ static void fas216_rq_sns_done(FAS216_Info *info, Scsi_Cmnd *SCpnt, unsigned int result) { -#ifdef DEBUG_CONNECT - printk("scsi%d.%c: request sense complete, result=%04X%02X%02X\n", - info->host->host_no, '0' + SCpnt->target, result, - SCpnt->SCp.Message, SCpnt->SCp.Status); -#endif + fas216_log_target(info, LOG_CONNECT, SCpnt->target, + "request sense complete, result=0x%04x%02x%02x", + result, SCpnt->SCp.Message, SCpnt->SCp.Status); if (result != DID_OK || SCpnt->SCp.Status != GOOD) /* @@ -1840,7 +2163,8 @@ * confuse the higher levels. */ memset(SCpnt->sense_buffer, 0, sizeof(SCpnt->sense_buffer)); - +//printk("scsi%d.%c: sense buffer: ", info->host->host_no, '0' + SCpnt->target); +//{ int i; for (i = 0; i < 32; i++) printk("%02x ", SCpnt->sense_buffer[i]); printk("\n"); } /* * Note that we don't set SCpnt->result, since that should * reflect the status of the command that we were asked by @@ -1850,11 +2174,13 @@ SCpnt->scsi_done(SCpnt); } -/* Function: void fas216_std_done(info, SCpnt, result) - * Purpose : Finish processing of standard command - * Params : info - interface that completed - * SCpnt - command that completed - * result - driver byte of result +/** + * fas216_std_done - finish processing of standard command + * @info: interface that completed + * @SCpnt: command that completed + * @result: driver byte of result + * + * Finish processing of standard command */ static void fas216_std_done(FAS216_Info *info, Scsi_Cmnd *SCpnt, unsigned int result) @@ -1864,31 +2190,29 @@ SCpnt->result = result << 16 | info->scsi.SCp.Message << 8 | info->scsi.SCp.Status; -#ifdef DEBUG_CONNECT - printk("scsi%d.%c: command complete, result=%08X, command=", - info->host->host_no, '0' + SCpnt->target, SCpnt->result); - print_command(SCpnt->cmnd); -#endif + fas216_log_command(info, LOG_CONNECT, SCpnt, + "command complete, result=0x%08x CDB: ", SCpnt->result); /* - * If the driver detected an error, or the command - * was request sense, then we're all done. + * If the driver detected an error, we're all done. */ - if (result != DID_OK || SCpnt->cmnd[0] == REQUEST_SENSE) + if (host_byte(SCpnt->result) != DID_OK || + msg_byte(SCpnt->result) != COMMAND_COMPLETE) goto done; /* - * If the command returned CHECK_CONDITION status, - * request the sense information. + * If the command returned CHECK_CONDITION or COMMAND_TERMINATED + * status, request the sense information. */ - if (info->scsi.SCp.Status == CHECK_CONDITION) + if (status_byte(SCpnt->result) == CHECK_CONDITION || + status_byte(SCpnt->result) == COMMAND_TERMINATED) goto request_sense; /* * If the command did not complete with GOOD status, * we are all done here. */ - if (info->scsi.SCp.Status != GOOD) + if (status_byte(SCpnt->result) != GOOD) goto done; /* @@ -1902,24 +2226,38 @@ switch (SCpnt->cmnd[0]) { case INQUIRY: case START_STOP: -// case READ_CAPACITY: case MODE_SENSE: break; default: printk(KERN_ERR "scsi%d.%c: incomplete data transfer " - "detected: res=%08X ptr=%p len=%X command=", + "detected: res=%08X ptr=%p len=%X CDB: ", info->host->host_no, '0' + SCpnt->target, SCpnt->result, info->scsi.SCp.ptr, info->scsi.SCp.this_residual); print_command(SCpnt->cmnd); + SCpnt->result &= ~(255 << 16); + SCpnt->result |= DID_BAD_TARGET << 16; + goto request_sense; } } -done: SCpnt->scsi_done(SCpnt); - return; +done: + if (SCpnt->scsi_done) { + SCpnt->scsi_done(SCpnt); + return; + } + + panic("scsi%d.H: null scsi_done function in fas216_done", + info->host->host_no); + request_sense: + if (SCpnt->cmnd[0] == REQUEST_SENSE) + goto done; + + fas216_log_target(info, LOG_CONNECT, SCpnt->target, + "requesting sense"); memset(SCpnt->cmnd, 0, sizeof (SCpnt->cmnd)); SCpnt->cmnd[0] = REQUEST_SENSE; SCpnt->cmnd[1] = SCpnt->lun << 5; @@ -1931,6 +2269,7 @@ SCpnt->SCp.this_residual = sizeof(SCpnt->sense_buffer); SCpnt->SCp.Message = 0; SCpnt->SCp.Status = 0; + SCpnt->request_bufflen = sizeof(SCpnt->sense_buffer); SCpnt->sc_data_direction = SCSI_DATA_READ; SCpnt->use_sg = 0; SCpnt->tag = 0; @@ -1947,15 +2286,18 @@ info->reqSCpnt = SCpnt; } -/* Function: void fas216_done(FAS216_Info *info, unsigned int result) - * Purpose : complete processing for current command - * Params : info - interface that completed - * result - driver byte of result +/** + * fas216_done - complete processing for current command + * @info: interface that completed + * @result: driver byte of result + * + * Complete processing for current command */ static void fas216_done(FAS216_Info *info, unsigned int result) { void (*fn)(FAS216_Info *, Scsi_Cmnd *, unsigned int); Scsi_Cmnd *SCpnt; + unsigned long flags; fas216_checkmagic(info); @@ -1966,12 +2308,8 @@ info->SCpnt = NULL; info->scsi.phase = PHASE_IDLE; - if (!SCpnt->scsi_done) - goto no_done; - if (info->scsi.aborting) { - printk("scsi%d.%c: uncaught abort - returning DID_ABORT\n", - info->host->host_no, fas216_target(info)); + fas216_log(info, 0, "uncaught abort - returning DID_ABORT"); result = DID_ABORT; info->scsi.aborting = 0; } @@ -1982,7 +2320,7 @@ */ if (info->scsi.SCp.ptr && info->scsi.SCp.this_residual == 0) { printk("scsi%d.%c: zero bytes left to transfer, but " - "buffer pointer still valid: ptr=%p len=%08x command=", + "buffer pointer still valid: ptr=%p len=%08x CDB: ", info->host->host_no, '0' + SCpnt->target, info->scsi.SCp.ptr, info->scsi.SCp.this_residual); info->scsi.SCp.ptr = NULL; @@ -1994,29 +2332,33 @@ * the sense information, fas216_kick will re-assert the busy * status. */ + info->device[SCpnt->target].parity_check = 0; clear_bit(SCpnt->target * 8 + SCpnt->lun, info->busyluns); fn = (void (*)(FAS216_Info *, Scsi_Cmnd *, unsigned int))SCpnt->host_scribble; fn(info, SCpnt, result); - if (info->scsi.irq != NO_IRQ) - fas216_kick(info); + if (info->scsi.irq != NO_IRQ) { + spin_lock_irqsave(&info->host_lock, flags); + if (info->scsi.phase == PHASE_IDLE) + fas216_kick(info); + spin_unlock_irqrestore(&info->host_lock, flags); + } return; no_command: panic("scsi%d.H: null command in fas216_done", info->host->host_no); -no_done: - panic("scsi%d.H: null scsi_done function in fas216_done", - info->host->host_no); } -/* Function: int fas216_queue_command(Scsi_Cmnd *SCpnt, void (*done)(Scsi_Cmnd *)) - * Purpose : queue a command for adapter to process. - * Params : SCpnt - Command to queue - * done - done function to call once command is complete - * Returns : 0 - success, else error - * Notes : io_request_lock is held, interrupts are disabled. +/** + * fas216_queue_command - queue a command for adapter to process. + * @SCpnt: Command to queue + * @done: done function to call once command is complete + * + * Queue a command for adapter to process. + * Returns: 0 on success, else error. + * Notes: io_request_lock is held, interrupts are disabled. */ int fas216_queue_command(Scsi_Cmnd *SCpnt, void (*done)(Scsi_Cmnd *)) { @@ -2025,61 +2367,20 @@ fas216_checkmagic(info); -#ifdef DEBUG_CONNECT - printk("scsi%d.%c: received queuable command (%p) %02X\n", - SCpnt->host->host_no, '0' + SCpnt->target, - SCpnt, SCpnt->cmnd[0]); -#endif + fas216_log_command(info, LOG_CONNECT, SCpnt, + "received command (%p) ", SCpnt); SCpnt->scsi_done = done; SCpnt->host_scribble = (void *)fas216_std_done; SCpnt->result = 0; - SCpnt->SCp.Message = 0; - SCpnt->SCp.Status = 0; - - if (SCpnt->use_sg) { - unsigned long len = 0; - int buf; - - SCpnt->SCp.buffer = (struct scatterlist *) SCpnt->buffer; - SCpnt->SCp.buffers_residual = SCpnt->use_sg - 1; - SCpnt->SCp.ptr = (char *) SCpnt->SCp.buffer->address; - SCpnt->SCp.this_residual = SCpnt->SCp.buffer->length; - /* - * Calculate correct buffer length. Some commands - * come in with the wrong request_bufflen. - */ - for (buf = 0; buf <= SCpnt->SCp.buffers_residual; buf++) - len += SCpnt->SCp.buffer[buf].length; - - if (SCpnt->request_bufflen != len) - printk(KERN_WARNING "scsi%d.%c: bad request buffer " - "length %d, should be %ld\n", info->host->host_no, - '0' + SCpnt->target, SCpnt->request_bufflen, len); - SCpnt->request_bufflen = len; - } else { - SCpnt->SCp.buffer = NULL; - SCpnt->SCp.buffers_residual = 0; - SCpnt->SCp.ptr = (unsigned char *)SCpnt->request_buffer; - SCpnt->SCp.this_residual = SCpnt->request_bufflen; - } - /* - * If the upper SCSI layers pass a buffer, but zero length, - * we aren't interested in the buffer pointer. - */ - if (SCpnt->SCp.this_residual == 0 && SCpnt->SCp.ptr) { -#if 0 - printk(KERN_WARNING "scsi%d.%c: zero length buffer passed for " - "command ", info->host->host_no, '0' + SCpnt->target); - print_command(SCpnt->cmnd); -#endif - SCpnt->SCp.ptr = NULL; - } + init_SCp(SCpnt); info->stats.queues += 1; SCpnt->tag = 0; + spin_lock(&info->host_lock); + /* * Add command into execute queue and let it complete under * whatever scheme we're using. @@ -2090,15 +2391,21 @@ * If we successfully added the command, * kick the interface to get it moving. */ - if (result == 0 && (!info->SCpnt || info->scsi.disconnectable)) + if (result == 0 && info->scsi.phase == PHASE_IDLE) fas216_kick(info); + spin_unlock(&info->host_lock); + + fas216_log_target(info, LOG_CONNECT, -1, "queue %s", + result ? "failure" : "success"); return result; } -/* Function: void fas216_internal_done(Scsi_Cmnd *SCpnt) - * Purpose : trigger restart of a waiting thread in fas216_command - * Params : SCpnt - Command to wake +/** + * fas216_internal_done - trigger restart of a waiting thread in fas216_command + * @SCpnt: Command to wake + * + * Trigger restart of a waiting thread in fas216_command */ static void fas216_internal_done(Scsi_Cmnd *SCpnt) { @@ -2109,11 +2416,13 @@ info->internal_done = 1; } -/* Function: int fas216_command(Scsi_Cmnd *SCpnt) - * Purpose : queue a command for adapter to process. - * Params : SCpnt - Command to queue - * Returns : scsi result code - * Notes : io_request_lock is held, interrupts are disabled. +/** + * fas216_command - queue a command for adapter to process. + * @SCpnt: Command to queue + * + * Queue a command for adapter to process. + * Returns: scsi result code. + * Notes: io_request_lock is held, interrupts are disabled. */ int fas216_command(Scsi_Cmnd *SCpnt) { @@ -2148,7 +2457,7 @@ * and go to sleep if we know that the device is going * to be some time (eg, disconnected). */ - if (inb(REG_STAT(info)) & STAT_INT) { + if (fas216_readb(info, REG_STAT) & STAT_INT) { spin_lock_irq(&io_request_lock); fas216_intr(info->host); spin_unlock_irq(&io_request_lock); @@ -2160,23 +2469,42 @@ return SCpnt->result; } -enum res_abort { - res_failed, /* unable to abort */ +/* + * Error handler timeout function. Indicate that we timed out, + * and wake up any error handler process so it can continue. + */ +static void fas216_eh_timer(unsigned long data) +{ + FAS216_Info *info = (FAS216_Info *)data; + + fas216_log(info, LOG_ERROR, "error handling timed out\n"); + + del_timer(&info->eh_timer); + + if (info->rst_bus_status == 0) + info->rst_bus_status = -1; + if (info->rst_dev_status == 0) + info->rst_dev_status = -1; + + wake_up(&info->eh_wait); +} + +enum res_find { + res_failed, /* not found */ res_success, /* command on issue queue */ - res_success_clear, /* command marked tgt/lun busy */ res_hw_abort /* command on disconnected dev */ }; -/* - * Prototype: enum res_abort fas216_do_abort(FAS216_Info *info, Scsi_Cmnd *SCpnt) - * Purpose : decide how to abort a command - * Params : SCpnt - command to abort - * Returns : abort status +/** + * fas216_do_abort - decide how to abort a command + * @SCpnt: command to abort + * + * Decide how to abort a command. + * Returns: abort status */ -static enum res_abort -fas216_do_abort(FAS216_Info *info, Scsi_Cmnd *SCpnt) +static enum res_find fas216_find_command(FAS216_Info *info, Scsi_Cmnd *SCpnt) { - enum res_abort res = res_failed; + enum res_find res = res_failed; if (queue_remove_cmd(&info->queues.issue, SCpnt)) { /* @@ -2224,19 +2552,22 @@ * been set. */ info->origSCpnt = NULL; + clear_bit(SCpnt->target * 8 + SCpnt->lun, info->busyluns); printk("waiting for execution "); - res = res_success_clear; + res = res_success; } else printk("unknown "); return res; } -/* Function: int fas216_eh_abort(Scsi_Cmnd *SCpnt) - * Purpose : abort this command - * Params : SCpnt - command to abort - * Returns : FAILED if unable to abort - * Notes : io_request_lock is taken, and irqs are disabled +/** + * fas216_eh_abort - abort this command + * @SCpnt: command to abort + * + * Abort this command. + * Returns: FAILED if unable to abort + * Notes: io_request_lock is taken, and irqs are disabled */ int fas216_eh_abort(Scsi_Cmnd *SCpnt) { @@ -2247,22 +2578,15 @@ info->stats.aborts += 1; + printk(KERN_WARNING "scsi%d: abort command ", info->host->host_no); + print_command(SCpnt->data_cmnd); + print_debug_list(); fas216_dumpstate(info); - fas216_dumpinfo(info); - - printk(KERN_WARNING "scsi%d: abort ", info->host->host_no); - switch (fas216_do_abort(info, SCpnt)) { - /* - * We managed to find the command and cleared it out. - * We do not expect the command to be executing on the - * target, but we have set the busylun bit. - */ - case res_success_clear: - printk("clear "); - clear_bit(SCpnt->target * 8 + SCpnt->lun, info->busyluns); + printk(KERN_WARNING "scsi%d: abort %p ", info->host->host_no, SCpnt); + switch (fas216_find_command(info, SCpnt)) { /* * We found the command, and cleared it out. Either * the command is still known to be executing on the @@ -2293,155 +2617,192 @@ return result; } -/* Function: void fas216_reset_state(FAS216_Info *info) - * Purpose : Initialise driver internal state - * Params : info - state to initialise +/** + * fas216_eh_device_reset - Reset the device associated with this command + * @SCpnt: command specifing device to reset + * + * Reset the device associated with this command. + * Returns: FAILED if unable to reset. + * Notes: We won't be re-entered, so we'll only have one device + * reset on the go at one time. */ -static void fas216_reset_state(FAS216_Info *info) +int fas216_eh_device_reset(Scsi_Cmnd *SCpnt) { - neg_t sync_state, wide_state; - int i; + FAS216_Info *info = (FAS216_Info *)SCpnt->host->hostdata; + unsigned long flags; + int i, res = FAILED, target = SCpnt->target; - fas216_checkmagic(info); + fas216_log(info, LOG_ERROR, "device reset for target %d", target); - /* - * Clear out all stale info in our state structure - */ - memset(info->busyluns, 0, sizeof(info->busyluns)); - msgqueue_flush(&info->scsi.msgs); - info->scsi.reconnected.target = 0; - info->scsi.reconnected.lun = 0; - info->scsi.reconnected.tag = 0; - info->scsi.disconnectable = 0; - info->scsi.aborting = 0; - info->scsi.phase = PHASE_IDLE; - info->scsi.async_stp = - fas216_syncperiod(info, info->ifcfg.asyncperiod); + spin_lock_irqsave(&info->host_lock, flags); - if (info->ifcfg.wide_max_size == 0) - wide_state = neg_invalid; - else -#ifdef SCSI2_WIDE - wide_state = neg_wait; -#else - wide_state = neg_invalid; -#endif + do { + /* + * If we are currently connected to a device, and + * it is the device we want to reset, there is + * nothing we can do here. Chances are it is stuck, + * and we need a bus reset. + */ + if (info->SCpnt && !info->scsi.disconnectable && + info->SCpnt->target == SCpnt->target) + break; - if (info->host->dma_channel == NO_DMA || !info->dma.setup) - sync_state = neg_invalid; - else -#ifdef SCSI2_SYNC - sync_state = neg_wait; -#else - sync_state = neg_invalid; -#endif + /* + * We're going to be resetting this device. Remove + * all pending commands from the driver. By doing + * so, we guarantee that we won't touch the command + * structures except to process the reset request. + */ + queue_remove_all_target(&info->queues.issue, target); + queue_remove_all_target(&info->queues.disconnected, target); + if (info->origSCpnt && info->origSCpnt->target == target) + info->origSCpnt = NULL; + if (info->reqSCpnt && info->reqSCpnt->target == target) + info->reqSCpnt = NULL; + for (i = 0; i < 8; i++) + clear_bit(target * 8 + i, info->busyluns); - for (i = 0; i < 8; i++) { - info->device[i].disconnect_ok = info->ifcfg.disconnect_ok; - info->device[i].sync_state = sync_state; - info->device[i].wide_state = wide_state; - info->device[i].period = info->ifcfg.asyncperiod / 4; - info->device[i].stp = info->scsi.async_stp; - info->device[i].sof = 0; - info->device[i].wide_xfer = 0; - } + /* + * Hijack this SCSI command structure to send + * a bus device reset message to this device. + */ + SCpnt->host_scribble = (void *)fas216_devicereset_done; - /* - * Drain all commands on disconnected queue - */ - while (queue_remove(&info->queues.disconnected) != NULL); + info->rst_dev_status = 0; + info->rstSCpnt = SCpnt; - /* - * Remove executing commands. - */ - info->SCpnt = NULL; - info->reqSCpnt = NULL; - info->origSCpnt = NULL; -} + if (info->scsi.phase == PHASE_IDLE) + fas216_kick(info); -/* Function: int fas216_eh_device_reset(Scsi_Cmnd *SCpnt) - * Purpose : Reset the device associated with this command - * Params : SCpnt - command specifing device to reset - * Returns : FAILED if unable to reset - */ -int fas216_eh_device_reset(Scsi_Cmnd *SCpnt) -{ - FAS216_Info *info = (FAS216_Info *)SCpnt->host->hostdata; + mod_timer(&info->eh_timer, 30 * HZ); + spin_unlock_irqrestore(&info->host_lock, flags); + + /* + * Wait up to 30 seconds for the reset to complete. + */ + wait_event(info->eh_wait, info->rst_dev_status); + + del_timer_sync(&info->eh_timer); + spin_lock_irqsave(&info->host_lock, flags); + info->rstSCpnt = NULL; - printk("scsi%d.%c: "__FUNCTION__": called\n", - info->host->host_no, '0' + SCpnt->target); - return FAILED; + if (info->rst_dev_status == 1) + res = SUCCESS; + } while (0); + + SCpnt->host_scribble = NULL; + spin_unlock_irqrestore(&info->host_lock, flags); + + fas216_log(info, LOG_ERROR, "device reset complete: %s\n", + res == SUCCESS ? "success" : "failed"); + + return res; } -/* Function: int fas216_eh_bus_reset(Scsi_Cmnd *SCpnt) - * Purpose : Reset the bus associated with the command - * Params : SCpnt - command specifing bus to reset - * Returns : FAILED if unable to reset - * Notes : io_request_lock is taken, and irqs are disabled +/** + * fas216_eh_bus_reset - Reset the bus associated with the command + * @SCpnt: command specifing bus to reset + * + * Reset the bus associated with the command. + * Returns: FAILED if unable to reset. + * Notes: Further commands are blocked. */ int fas216_eh_bus_reset(Scsi_Cmnd *SCpnt) { FAS216_Info *info = (FAS216_Info *)SCpnt->host->hostdata; - int result = FAILED; + unsigned long flags; + Scsi_Device *SDpnt; fas216_checkmagic(info); + fas216_log(info, LOG_ERROR, "resetting bus"); info->stats.bus_resets += 1; - printk("scsi%d.%c: "__FUNCTION__": resetting bus\n", - info->host->host_no, '0' + SCpnt->target); + spin_lock_irqsave(&info->host_lock, flags); + + /* + * Stop all activity on this interface. + */ + fas216_aborttransfer(info); + fas216_writeb(info, REG_CNTL3, info->scsi.cfg[2]); /* - * Attempt to stop all activity on this interface. + * Clear any pending interrupts. */ - outb(info->scsi.cfg[2], REG_CNTL3(info)); - fas216_stoptransfer(info); + while (fas216_readb(info, REG_STAT) & STAT_INT) + fas216_readb(info, REG_INST); + + info->rst_bus_status = 0; /* - * Clear any pending interrupts + * For each attached hard-reset device, clear out + * all command structures. Leave the running + * command in place. */ - while (inb(REG_STAT(info)) & STAT_INT) - inb(REG_INST(info)); + for (SDpnt = info->host->host_queue; SDpnt; SDpnt = SDpnt->next) { + int i; + + if (SDpnt->soft_reset) + continue; + + queue_remove_all_target(&info->queues.issue, SDpnt->id); + queue_remove_all_target(&info->queues.disconnected, SDpnt->id); + if (info->origSCpnt && info->origSCpnt->target == SDpnt->id) + info->origSCpnt = NULL; + if (info->reqSCpnt && info->reqSCpnt->target == SDpnt->id) + info->reqSCpnt = NULL; + info->SCpnt = NULL; + + for (i = 0; i < 8; i++) + clear_bit(SDpnt->id * 8 + i, info->busyluns); + } /* - * Reset the SCSI bus + * Reset the SCSI bus. Device cleanup happens in + * the interrupt handler. */ - outb(CMD_RESETSCSI, REG_CMD(info)); - udelay(5); + fas216_cmd(info, CMD_RESETSCSI); + + mod_timer(&info->eh_timer, jiffies + HZ); + spin_unlock_irqrestore(&info->host_lock, flags); /* - * Clear reset interrupt + * Wait one second for the interrupt. */ - if (inb(REG_STAT(info)) & STAT_INT && - inb(REG_INST(info)) & INST_BUSRESET) - result = SUCCESS; + wait_event(info->eh_wait, info->rst_bus_status); + del_timer_sync(&info->eh_timer); - fas216_reset_state(info); + fas216_log(info, LOG_ERROR, "bus reset complete: %s\n", + info->rst_bus_status == 1 ? "success" : "failed"); - return result; + return info->rst_bus_status == 1 ? SUCCESS : FAILED; } -/* Function: void fas216_init_chip(FAS216_Info *info) - * Purpose : Initialise FAS216 state after reset - * Params : info - state structure for interface +/** + * fas216_init_chip - Initialise FAS216 state after reset + * @info: state structure for interface + * + * Initialise FAS216 state after reset */ static void fas216_init_chip(FAS216_Info *info) { - outb(fas216_clockrate(info->ifcfg.clockrate), REG_CLKF(info)); - outb(info->scsi.cfg[0], REG_CNTL1(info)); - outb(info->scsi.cfg[1], REG_CNTL2(info)); - outb(info->scsi.cfg[2], REG_CNTL3(info)); - outb(info->ifcfg.select_timeout, REG_STIM(info)); - outb(0, REG_SOF(info)); - outb(info->scsi.async_stp, REG_STP(info)); - outb(info->scsi.cfg[0], REG_CNTL1(info)); + fas216_writeb(info, REG_CLKF, fas216_clockrate(info->ifcfg.clockrate)); + fas216_writeb(info, REG_CNTL1, info->scsi.cfg[0]); + fas216_writeb(info, REG_CNTL2, info->scsi.cfg[1]); + fas216_writeb(info, REG_CNTL3, info->scsi.cfg[2]); + fas216_writeb(info, REG_STIM, info->ifcfg.select_timeout); + fas216_writeb(info, REG_SOF, 0); + fas216_writeb(info, REG_STP, info->scsi.async_stp); + fas216_writeb(info, REG_CNTL1, info->scsi.cfg[0]); } -/* Function: int fas216_eh_host_reset(Scsi_Cmnd *SCpnt) - * Purpose : Reset the host associated with this command - * Params : SCpnt - command specifing host to reset - * Returns : FAILED if unable to reset - * Notes : io_request_lock is taken, and irqs are disabled +/** + * fas216_eh_host_reset - Reset the host associated with this command + * @SCpnt: command specifing host to reset + * + * Reset the host associated with this command. + * Returns: FAILED if unable to reset. + * Notes: io_request_lock is taken, and irqs are disabled */ int fas216_eh_host_reset(Scsi_Cmnd *SCpnt) { @@ -2449,13 +2810,13 @@ fas216_checkmagic(info); - printk("scsi%d.%c: "__FUNCTION__": resetting host\n", - info->host->host_no, '0' + SCpnt->target); + printk("scsi%d.%c: %s: resetting host\n", + info->host->host_no, '0' + SCpnt->target, __FUNCTION__); /* * Reset the SCSI chip. */ - outb(CMD_RESETCHIP, REG_CMD(info)); + fas216_cmd(info, CMD_RESETCHIP); /* * Ugly ugly ugly! @@ -2464,13 +2825,13 @@ * IRQs after the sleep. */ spin_unlock_irq(&io_request_lock); - scsi_sleep(25*HZ/100); + scsi_sleep(50 * HZ/100); spin_lock_irq(&io_request_lock); /* * Release the SCSI reset. */ - outb(CMD_NOP, REG_CMD(info)); + fas216_cmd(info, CMD_NOP); fas216_init_chip(info); @@ -2502,56 +2863,56 @@ /* * Reset the chip. */ - outb(CMD_RESETCHIP, REG_CMD(info)); + fas216_writeb(info, REG_CMD, CMD_RESETCHIP); udelay(50); - outb(CMD_NOP, REG_CMD(info)); + fas216_writeb(info, REG_CMD, CMD_NOP); /* * Check to see if control reg 2 is present. */ - outb(0, REG_CNTL3(info)); - outb(CNTL2_S2FE, REG_CNTL2(info)); + fas216_writeb(info, REG_CNTL3, 0); + fas216_writeb(info, REG_CNTL2, CNTL2_S2FE); /* * If we are unable to read back control reg 2 * correctly, it is not present, and we have a * NCR53C90. */ - if ((inb(REG_CNTL2(info)) & (~0xe0)) != CNTL2_S2FE) + if ((fas216_readb(info, REG_CNTL2) & (~0xe0)) != CNTL2_S2FE) return TYPE_NCR53C90; /* * Now, check control register 3 */ - outb(0, REG_CNTL2(info)); - outb(0, REG_CNTL3(info)); - outb(5, REG_CNTL3(info)); + fas216_writeb(info, REG_CNTL2, 0); + fas216_writeb(info, REG_CNTL3, 0); + fas216_writeb(info, REG_CNTL3, 5); /* * If we are unable to read the register back * correctly, we have a NCR53C90A */ - if (inb(REG_CNTL3(info)) != 5) + if (fas216_readb(info, REG_CNTL3) != 5) return TYPE_NCR53C90A; /* * Now read the ID from the chip. */ - outb(0, REG_CNTL3(info)); + fas216_writeb(info, REG_CNTL3, 0); - outb(CNTL3_ADIDCHK, REG_CNTL3(info)); - outb(0, REG_CNTL3(info)); + fas216_writeb(info, REG_CNTL3, CNTL3_ADIDCHK); + fas216_writeb(info, REG_CNTL3, 0); - outb(CMD_RESETCHIP, REG_CMD(info)); - udelay(5); - outb(CMD_WITHDMA | CMD_NOP, REG_CMD(info)); + fas216_writeb(info, REG_CMD, CMD_RESETCHIP); + udelay(50); + fas216_writeb(info, REG_CMD, CMD_WITHDMA | CMD_NOP); - outb(CNTL2_ENF, REG_CNTL2(info)); - outb(CMD_RESETCHIP, REG_CMD(info)); - udelay(5); - outb(CMD_NOP, REG_CMD(info)); + fas216_writeb(info, REG_CNTL2, CNTL2_ENF); + fas216_writeb(info, REG_CMD, CMD_RESETCHIP); + udelay(50); + fas216_writeb(info, REG_CMD, CMD_NOP); - rev = inb(REG1_ID(info)); + rev = fas216_readb(info, REG_ID); family = rev >> 3; rev &= 7; @@ -2577,35 +2938,102 @@ return TYPE_NCR53C9x; } -/* Function: int fas216_init(struct Scsi_Host *instance) - * Purpose : initialise FAS/NCR/AMD SCSI ic. - * Params : instance - a driver-specific filled-out structure - * Returns : 0 on success +/** + * fas216_reset_state - Initialise driver internal state + * @info: state to initialise + * + * Initialise driver internal state */ -int fas216_init(struct Scsi_Host *instance) +static void fas216_reset_state(FAS216_Info *info) { - FAS216_Info *info = (FAS216_Info *)instance->hostdata; - int type; + int i; + + fas216_checkmagic(info); + + fas216_bus_reset(info); + + /* + * Clear out all stale info in our state structure + */ + memset(info->busyluns, 0, sizeof(info->busyluns)); + info->scsi.disconnectable = 0; + info->scsi.aborting = 0; + + for (i = 0; i < 8; i++) { + info->device[i].parity_enabled = 0; + info->device[i].parity_check = 1; + } + + /* + * Drain all commands on disconnected queue + */ + while (queue_remove(&info->queues.disconnected) != NULL); + + /* + * Remove executing commands. + */ + info->SCpnt = NULL; + info->reqSCpnt = NULL; + info->rstSCpnt = NULL; + info->origSCpnt = NULL; +} + +/** + * fas216_init - initialise FAS/NCR/AMD SCSI structures. + * @host: a driver-specific filled-out structure + * + * Initialise FAS/NCR/AMD SCSI structures. + * Returns: 0 on success + */ +int fas216_init(struct Scsi_Host *host) +{ + FAS216_Info *info = (FAS216_Info *)host->hostdata; - info->magic_start = MAGIC; - info->magic_end = MAGIC; - info->host = instance; - info->scsi.cfg[0] = instance->this_id; - info->scsi.cfg[1] = CNTL2_ENF | CNTL2_S2FE; - info->scsi.cfg[2] = info->ifcfg.cntl3 | CNTL3_ADIDCHK | CNTL3_G2CB; + info->magic_start = MAGIC; + info->magic_end = MAGIC; + info->host = host; + info->scsi.cfg[0] = host->this_id | CNTL1_PERE; + info->scsi.cfg[1] = CNTL2_ENF | CNTL2_S2FE; + info->scsi.cfg[2] = info->ifcfg.cntl3 | + CNTL3_ADIDCHK | CNTL3_G2CB | CNTL3_LBTM; + info->scsi.async_stp = fas216_syncperiod(info, info->ifcfg.asyncperiod); + + info->rst_dev_status = -1; + info->rst_bus_status = -1; + init_waitqueue_head(&info->eh_wait); + init_timer(&info->eh_timer); + info->eh_timer.data = (unsigned long)info; + info->eh_timer.function = fas216_eh_timer; + + spin_lock_init(&info->host_lock); memset(&info->stats, 0, sizeof(info->stats)); msgqueue_initialise(&info->scsi.msgs); if (!queue_initialise(&info->queues.issue)) - return 1; + return -ENOMEM; if (!queue_initialise(&info->queues.disconnected)) { queue_free(&info->queues.issue); - return 1; + return -ENOMEM; } + return 0; +} + +/** + * fas216_add - initialise FAS/NCR/AMD SCSI ic. + * @host: a driver-specific filled-out structure + * + * Initialise FAS/NCR/AMD SCSI ic. + * Returns: 0 on success + */ +int fas216_add(struct Scsi_Host *host) +{ + FAS216_Info *info = (FAS216_Info *)host->hostdata; + int type; + fas216_reset_state(info); type = fas216_detect_type(info); info->scsi.type = chip_types[type]; @@ -2622,74 +3050,49 @@ * the resulting reset interrupt, so mask it * out. */ - outb(info->scsi.cfg[0] | CNTL1_DISR, REG_CNTL1(info)); - outb(CMD_RESETSCSI, REG_CMD(info)); + fas216_writeb(info, REG_CNTL1, info->scsi.cfg[0] | CNTL1_DISR); + fas216_writeb(info, REG_CMD, CMD_RESETSCSI); /* * scsi standard says wait 250ms */ spin_unlock_irq(&io_request_lock); - scsi_sleep(25*HZ/100); + scsi_sleep(100*HZ/100); spin_lock_irq(&io_request_lock); - outb(info->scsi.cfg[0], REG_CNTL1(info)); - inb(REG_INST(info)); + fas216_writeb(info, REG_CNTL1, info->scsi.cfg[0]); + fas216_readb(info, REG_INST); fas216_checkmagic(info); return 0; } -/* Function: int fas216_release(struct Scsi_Host *instance) - * Purpose : release all resources and put everything to bed for - * FAS/NCR/AMD SCSI ic. - * Params : instance - a driver-specific filled-out structure - * Returns : 0 on success - */ -int fas216_release(struct Scsi_Host *instance) +void fas216_remove(struct Scsi_Host *host) { - FAS216_Info *info = (FAS216_Info *)instance->hostdata; + FAS216_Info *info = (FAS216_Info *)host->hostdata; fas216_checkmagic(info); - outb(CMD_RESETCHIP, REG_CMD(info)); - queue_free(&info->queues.disconnected); - queue_free(&info->queues.issue); - - return 0; + fas216_writeb(info, REG_CMD, CMD_RESETCHIP); } -/* - * Function: int fas216_info(FAS216_Info *info, char *buffer) - * Purpose : generate a string containing information about this - * host. - * Params : info - FAS216 host information - * buffer - string buffer to build string - * Returns : size of built string +/** + * fas216_release - release all resources for FAS/NCR/AMD SCSI ic. + * @host: a driver-specific filled-out structure + * + * release all resources and put everything to bed for FAS/NCR/AMD SCSI ic. */ -int fas216_info(FAS216_Info *info, char *buffer) +void fas216_release(struct Scsi_Host *host) { - char *p = buffer; + FAS216_Info *info = (FAS216_Info *)host->hostdata; - p += sprintf(p, "(%s) at port 0x%08lX ", - info->scsi.type, info->host->io_port); - - if (info->host->irq != NO_IRQ) - p += sprintf(p, "irq %d ", info->host->irq); - else - p += sprintf(p, "no irq "); - - if (info->host->dma_channel != NO_DMA) - p += sprintf(p, "dma %d ", info->host->dma_channel); - else - p += sprintf(p, "no dma "); - - return p - buffer; + queue_free(&info->queues.disconnected); + queue_free(&info->queues.issue); } int fas216_print_host(FAS216_Info *info, char *buffer) { - return sprintf(buffer, "\n" "Chip : %s\n" @@ -2702,8 +3105,9 @@ int fas216_print_stats(FAS216_Info *info, char *buffer) { - return sprintf(buffer, - "\n" + char *p = buffer; + + p += sprintf(p, "\n" "Command Statistics:\n" " Queued : %u\n" " Issued : %u\n" @@ -2720,6 +3124,8 @@ info->stats.writes, info->stats.miscs, info->stats.disconnects, info->stats.aborts, info->stats.bus_resets, info->stats.host_resets); + + return p - buffer; } int fas216_print_device(FAS216_Info *info, Scsi_Device *scd, char *buffer) @@ -2738,7 +3144,9 @@ scd->tagged_queue ? "en" : "dis", scd->current_tag); - p += sprintf(p, "\n Transfers : %d-bit ", + p += sprintf(p, "%s\n", dev->parity_enabled ? "parity" : ""); + + p += sprintf(p, " Transfers : %d-bit ", 8 << dev->wide_xfer); if (dev->sof) @@ -2750,11 +3158,12 @@ return p - buffer; } -EXPORT_SYMBOL(fas216_info); EXPORT_SYMBOL(fas216_init); +EXPORT_SYMBOL(fas216_add); EXPORT_SYMBOL(fas216_queue_command); EXPORT_SYMBOL(fas216_command); EXPORT_SYMBOL(fas216_intr); +EXPORT_SYMBOL(fas216_remove); EXPORT_SYMBOL(fas216_release); EXPORT_SYMBOL(fas216_eh_abort); EXPORT_SYMBOL(fas216_eh_device_reset); diff -urN orig/drivers/acorn/scsi/fas216.h linux/drivers/acorn/scsi/fas216.h --- orig/drivers/acorn/scsi/fas216.h Tue Oct 3 20:08:24 2000 +++ linux/drivers/acorn/scsi/fas216.h Thu Jan 23 13:04:42 2003 @@ -22,18 +22,18 @@ /* FAS register definitions */ /* transfer count low */ -#define REG_CTCL(x) ((x)->scsi.io_port) -#define REG_STCL(x) ((x)->scsi.io_port) +#define REG_CTCL (0) +#define REG_STCL (0) /* transfer count medium */ -#define REG_CTCM(x) ((x)->scsi.io_port + (1 << (x)->scsi.io_shift)) -#define REG_STCM(x) ((x)->scsi.io_port + (1 << (x)->scsi.io_shift)) +#define REG_CTCM (1) +#define REG_STCM (1) /* fifo data */ -#define REG_FF(x) ((x)->scsi.io_port + (2 << (x)->scsi.io_shift)) +#define REG_FF (2) /* command */ -#define REG_CMD(x) ((x)->scsi.io_port + (3 << (x)->scsi.io_shift)) +#define REG_CMD (3) #define CMD_NOP 0x00 #define CMD_FLUSHFIFO 0x01 #define CMD_RESETCHIP 0x02 @@ -57,7 +57,7 @@ #define CMD_WITHDMA 0x80 /* status register (read) */ -#define REG_STAT(x) ((x)->scsi.io_port + (4 << (x)->scsi.io_shift)) +#define REG_STAT (4) #define STAT_IO (1 << 0) /* IO phase */ #define STAT_CD (1 << 1) /* CD phase */ #define STAT_MSG (1 << 2) /* MSG phase */ @@ -76,11 +76,11 @@ #define STAT_MESGIN (STAT_MSG|STAT_CD|STAT_IO) /* Message In */ /* bus ID for select / reselect */ -#define REG_SDID(x) ((x)->scsi.io_port + (4 << (x)->scsi.io_shift)) +#define REG_SDID (4) #define BUSID(target) ((target) & 7) /* Interrupt status register (read) */ -#define REG_INST(x) ((x)->scsi.io_port + (5 << (x)->scsi.io_shift)) +#define REG_INST (5) #define INST_SELWOATN (1 << 0) /* Select w/o ATN */ #define INST_SELATN (1 << 1) /* Select w/ATN */ #define INST_RESELECTED (1 << 2) /* Reselected */ @@ -91,10 +91,10 @@ #define INST_BUSRESET (1 << 7) /* SCSI Bus reset */ /* Timeout register (write) */ -#define REG_STIM(x) ((x)->scsi.io_port + (5 << (x)->scsi.io_shift)) +#define REG_STIM (5) /* Sequence step register (read) */ -#define REG_IS(x) ((x)->scsi.io_port + (6 << (x)->scsi.io_shift)) +#define REG_IS (6) #define IS_BITS 0x07 #define IS_SELARB 0x00 /* Select & Arb ok */ #define IS_MSGBYTESENT 0x01 /* One byte message sent*/ @@ -104,18 +104,18 @@ #define IS_SOF 0x08 /* Sync off flag */ /* Transfer period step (write) */ -#define REG_STP(x) ((x)->scsi.io_port + (6 << (x)->scsi.io_shift)) +#define REG_STP (6) /* Synchronous Offset (write) */ -#define REG_SOF(x) ((x)->scsi.io_port + (7 << (x)->scsi.io_shift)) +#define REG_SOF (7) /* Fifo state register (read) */ -#define REG_CFIS(x) ((x)->scsi.io_port + (7 << (x)->scsi.io_shift)) +#define REG_CFIS (7) #define CFIS_CF 0x1f /* Num bytes in FIFO */ #define CFIS_IS 0xe0 /* Step */ /* config register 1 */ -#define REG_CNTL1(x) ((x)->scsi.io_port + (8 << (x)->scsi.io_shift)) +#define REG_CNTL1 (8) #define CNTL1_CID (7 << 0) /* Chip ID */ #define CNTL1_STE (1 << 3) /* Self test enable */ #define CNTL1_PERE (1 << 4) /* Parity enable reporting en. */ @@ -124,7 +124,7 @@ #define CNTL1_ETM (1 << 7) /* Extended Timing Mode */ /* Clock conversion factor (read) */ -#define REG_CLKF(x) ((x)->scsi.io_port + (9 << (x)->scsi.io_shift)) +#define REG_CLKF (9) #define CLKF_F37MHZ 0x00 /* 35.01 - 40 MHz */ #define CLKF_F10MHZ 0x02 /* 10 MHz */ #define CLKF_F12MHZ 0x03 /* 10.01 - 15 MHz */ @@ -134,13 +134,13 @@ #define CLKF_F32MHZ 0x07 /* 30.01 - 35 MHz */ /* Chip test register (write) */ -#define REG0_FTM(x) ((x)->scsi.io_port + (10 << (x)->scsi.io_shift)) +#define REG_FTM (10) #define TEST_FTM 0x01 /* Force target mode */ #define TEST_FIM 0x02 /* Force initiator mode */ #define TEST_FHI 0x04 /* Force high impedance mode */ /* Configuration register 2 (read/write) */ -#define REG_CNTL2(x) ((x)->scsi.io_port + (11 << (x)->scsi.io_shift)) +#define REG_CNTL2 (11) #define CNTL2_PGDP (1 << 0) /* Pass Th/Generate Data Parity */ #define CNTL2_PGRP (1 << 1) /* Pass Th/Generate Reg Parity */ #define CNTL2_ACDPE (1 << 2) /* Abort on Cmd/Data Parity Err */ @@ -151,7 +151,7 @@ #define CNTL2_DAE (1 << 7) /* Data Alignment Enable */ /* Configuration register 3 (read/write) */ -#define REG_CNTL3(x) ((x)->scsi.io_port + (12 << (x)->scsi.io_shift)) +#define REG_CNTL3 (12) #define CNTL3_BS8 (1 << 0) /* Burst size 8 */ #define CNTL3_MDM (1 << 1) /* Modify DMA mode */ #define CNTL3_LBTM (1 << 2) /* Last Byte Transfer mode */ @@ -162,14 +162,14 @@ #define CNTL3_ADIDCHK (1 << 7) /* Additional ID check */ /* High transfer count (read/write) */ -#define REG_CTCH(x) ((x)->scsi.io_port + (14 << (x)->scsi.io_shift)) -#define REG_STCH(x) ((x)->scsi.io_port + (14 << (x)->scsi.io_shift)) +#define REG_CTCH (14) +#define REG_STCH (14) -/* ID reigster (read only) */ -#define REG1_ID(x) ((x)->scsi.io_port + (14 << (x)->scsi.io_shift)) +/* ID register (read only) */ +#define REG_ID (14) /* Data alignment */ -#define REG0_DAL(x) ((x)->scsi.io_port + (15 << (x)->scsi.io_shift)) +#define REG_DAL (15) typedef enum { PHASE_IDLE, /* we're not planning on doing anything */ @@ -212,21 +212,36 @@ #define MAGIC 0x441296bdUL #define NR_MSGS 8 +#define FASCAP_DMA (1 << 0) +#define FASCAP_PSEUDODMA (1 << 1) + typedef struct { unsigned long magic_start; + spinlock_t host_lock; struct Scsi_Host *host; /* host */ Scsi_Cmnd *SCpnt; /* currently processing command */ Scsi_Cmnd *origSCpnt; /* original connecting command */ Scsi_Cmnd *reqSCpnt; /* request sense command */ + Scsi_Cmnd *rstSCpnt; /* reset command */ + Scsi_Cmnd *pending_SCpnt[8]; /* per-device pending commands */ + int next_pending; /* next pending device */ + + /* + * Error recovery + */ + wait_queue_head_t eh_wait; + struct timer_list eh_timer; + unsigned int rst_dev_status; + unsigned int rst_bus_status; /* driver information */ struct { + phase_t phase; /* current phase */ unsigned int io_port; /* base address of FAS216 */ unsigned int io_shift; /* shift to adjust reg offsets by */ - unsigned int irq; /* interrupt */ unsigned char cfg[4]; /* configuration registers */ const char *type; /* chip type */ - phase_t phase; /* current phase */ + unsigned int irq; /* interrupt */ struct { unsigned char target; /* reconnected target */ @@ -241,7 +256,6 @@ unsigned int async_stp; /* Async transfer STP value */ unsigned char msgin_fifo; /* bytes in fifo at time of message in */ unsigned char message[256]; /* last message received from device */ - unsigned int msglen; /* length of last message received */ unsigned char disconnectable:1; /* this command can be disconnected */ unsigned char aborting:1; /* aborting command */ @@ -269,6 +283,7 @@ unsigned char wide_max_size; /* Maximum wide transfer size */ unsigned char cntl3; /* Control Reg 3 */ unsigned int asyncperiod; /* Async transfer period (ns) */ + unsigned int capabilities; /* driver capabilities */ unsigned int disconnect_ok:1; /* Disconnects allowed? */ } ifcfg; @@ -281,6 +296,8 @@ /* per-device info */ struct fas216_device { unsigned char disconnect_ok:1; /* device can disconnect */ + unsigned char parity_enabled:1; /* parity checking enabled */ + unsigned char parity_check:1; /* need to check parity checking */ unsigned char period; /* sync xfer period in (*4ns) */ unsigned char stp; /* synchronous transfer period */ unsigned char sof; /* synchronous offset register */ @@ -305,12 +322,19 @@ } FAS216_Info; /* Function: int fas216_init (struct Scsi_Host *instance) - * Purpose : initialise FAS/NCR/AMD SCSI ic. + * Purpose : initialise FAS/NCR/AMD SCSI structures. * Params : instance - a driver-specific filled-out structure * Returns : 0 on success */ extern int fas216_init (struct Scsi_Host *instance); +/* Function: int fas216_add (struct Scsi_Host *instance) + * Purpose : initialise FAS/NCR/AMD SCSI ic. + * Params : instance - a driver-specific filled-out structure + * Returns : 0 on success + */ +extern int fas216_add (struct Scsi_Host *instance); + /* Function: int fas216_abort (Scsi_Cmnd *SCpnt) * Purpose : abort a command if something horrible happens. * Params : SCpnt - Command that is believed to be causing a problem. @@ -347,14 +371,15 @@ */ extern void fas216_intr (struct Scsi_Host *instance); +extern void fas216_remove (struct Scsi_Host *instance); + /* Function: int fas216_release (struct Scsi_Host *instance) * Purpose : release all resources and put everything to bed for FAS/NCR/AMD SCSI ic. * Params : instance - a driver-specific filled-out structure * Returns : 0 on success */ -extern int fas216_release (struct Scsi_Host *instance); +extern void fas216_release (struct Scsi_Host *instance); -extern int fas216_info(FAS216_Info *info, char *buffer); extern int fas216_print_host(FAS216_Info *info, char *buffer); extern int fas216_print_stats(FAS216_Info *info, char *buffer); extern int fas216_print_device(FAS216_Info *info, Scsi_Device *scd, char *buffer); diff -urN orig/drivers/acorn/scsi/powertec.c linux/drivers/acorn/scsi/powertec.c --- orig/drivers/acorn/scsi/powertec.c Fri Nov 16 10:09:50 2001 +++ linux/drivers/acorn/scsi/powertec.c Fri Feb 21 15:44:34 2003 @@ -1,7 +1,7 @@ /* * linux/drivers/acorn/scsi/powertec.c * - * Copyright (C) 1997-2000 Russell King + * Copyright (C) 1997-2003 Russell King * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as @@ -26,8 +26,6 @@ #include #include #include -#include -#include #include #include #include @@ -41,14 +39,10 @@ #include "../../scsi/sd.h" #include "../../scsi/hosts.h" #include "fas216.h" +#include "scsi.h" #include -/* Configuration */ -#define POWERTEC_XTALFREQ 40 -#define POWERTEC_ASYNC_PERIOD 200 -#define POWERTEC_SYNC_DEPTH 7 - /* * List of devices that the driver will recognise */ @@ -70,33 +64,23 @@ #define POWERTEC_INTR_ENABLE 1 #define POWERTEC_INTR_DISABLE 0 -/* - * Version - */ -#define VER_MAJOR 0 -#define VER_MINOR 0 -#define VER_PATCH 5 - -static struct expansion_card *ecs[MAX_ECARDS]; +#define VERSION "1.10 (22/01/2003 2.4.19-rmk5)" /* - * Use term=0,1,0,0,0 to turn terminators on/off + * Use term=0,1,0,0,0 to turn terminators on/off. + * One entry per slot. */ static int term[MAX_ECARDS] = { 1, 1, 1, 1, 1, 1, 1, 1 }; #define NR_SG 256 -typedef struct { +struct powertec_info { FAS216_Info info; - - struct { - unsigned int term_port; - unsigned int terms; - } control; - - /* other info... */ - struct scatterlist sg[NR_SG]; /* Scatter DMA list */ -} PowerTecScsi_Info; + struct expansion_card *ec; + unsigned int term_port; + unsigned int term_ctl; + struct scatterlist sg[NR_SG]; +}; /* Prototype: void powertecscsi_irqenable(ec, irqnr) * Purpose : Enable interrupts on Powertec SCSI card @@ -123,12 +107,8 @@ } static const expansioncard_ops_t powertecscsi_ops = { - powertecscsi_irqenable, - powertecscsi_irqdisable, - NULL, - NULL, - NULL, - NULL + .irqenable = powertecscsi_irqenable, + .irqdisable = powertecscsi_irqdisable, }; /* Prototype: void powertecscsi_terminator_ctl(host, on_off) @@ -139,14 +119,10 @@ static void powertecscsi_terminator_ctl(struct Scsi_Host *host, int on_off) { - PowerTecScsi_Info *info = (PowerTecScsi_Info *)host->hostdata; - - if (on_off) - info->control.terms = POWERTEC_TERM_ENABLE; - else - info->control.terms = 0; + struct powertec_info *info = (struct powertec_info *)host->hostdata; - outb(info->control.terms, info->control.term_port); + info->term_ctl = on_off ? POWERTEC_TERM_ENABLE : 0; + outb(info->term_ctl, info->term_port); } /* Prototype: void powertecscsi_intr(irq, *dev_id, *regs) @@ -175,32 +151,26 @@ powertecscsi_dma_setup(struct Scsi_Host *host, Scsi_Pointer *SCp, fasdmadir_t direction, fasdmatype_t min_type) { - PowerTecScsi_Info *info = (PowerTecScsi_Info *)host->hostdata; + struct powertec_info *info = (struct powertec_info *)host->hostdata; int dmach = host->dma_channel; - if (dmach != NO_DMA && - (min_type == fasdma_real_all || SCp->this_residual >= 512)) { - int bufs = SCp->buffers_residual; - int pci_dir, dma_dir; - - if (bufs) - memcpy(info->sg + 1, SCp->buffer + 1, - sizeof(struct scatterlist) * bufs); - info->sg[0].address = SCp->ptr; - info->sg[0].page = NULL; - info->sg[0].length = SCp->this_residual; + if (info->info.ifcfg.capabilities & FASCAP_DMA && + min_type == fasdma_real_all) { + int bufs, map_dir, dma_dir; + + bufs = copy_SCp_to_sg(&info->sg[0], SCp, NR_SG); if (direction == DMA_OUT) - pci_dir = PCI_DMA_TODEVICE, + map_dir = PCI_DMA_TODEVICE, dma_dir = DMA_MODE_WRITE; else - pci_dir = PCI_DMA_FROMDEVICE, + map_dir = PCI_DMA_FROMDEVICE, dma_dir = DMA_MODE_READ; - pci_map_sg(NULL, info->sg, bufs + 1, pci_dir); + pci_map_sg(NULL, info->sg, bufs, map_dir); disable_dma(dmach); - set_dma_sg(dmach, info->sg, bufs + 1); + set_dma_sg(dmach, info->sg, bufs); set_dma_mode(dmach, dma_dir); enable_dma(dmach); return fasdma_real_all; @@ -225,122 +195,6 @@ disable_dma(host->dma_channel); } -/* Prototype: int powertecscsi_detect(Scsi_Host_Template * tpnt) - * Purpose : initialises PowerTec SCSI driver - * Params : tpnt - template for this SCSI adapter - * Returns : >0 if host found, 0 otherwise. - */ -int -powertecscsi_detect(Scsi_Host_Template *tpnt) -{ - static const card_ids powertecscsi_cids[] = - { POWERTECSCSI_LIST, { 0xffff, 0xffff} }; - int count = 0; - struct Scsi_Host *host; - - tpnt->proc_name = "powertec"; - memset(ecs, 0, sizeof (ecs)); - - ecard_startfind(); - - while (1) { - PowerTecScsi_Info *info; - - ecs[count] = ecard_find(0, powertecscsi_cids); - if (!ecs[count]) - break; - - ecard_claim(ecs[count]); - - host = scsi_register(tpnt, sizeof (PowerTecScsi_Info)); - if (!host) { - ecard_release(ecs[count]); - break; - } - - host->io_port = ecard_address(ecs[count], ECARD_IOC, ECARD_FAST); - host->irq = ecs[count]->irq; - host->dma_channel = ecs[count]->dma; - info = (PowerTecScsi_Info *)host->hostdata; - - if (host->dma_channel != NO_DMA) - set_dma_speed(host->dma_channel, 180); - - info->control.term_port = host->io_port + POWERTEC_TERM_CONTROL; - info->control.terms = term[count] ? POWERTEC_TERM_ENABLE : 0; - powertecscsi_terminator_ctl(host, info->control.terms); - - info->info.scsi.io_port = - host->io_port + POWERTEC_FAS216_OFFSET; - info->info.scsi.io_shift= POWERTEC_FAS216_SHIFT; - info->info.scsi.irq = host->irq; - info->info.ifcfg.clockrate = POWERTEC_XTALFREQ; - info->info.ifcfg.select_timeout = 255; - info->info.ifcfg.asyncperiod = POWERTEC_ASYNC_PERIOD; - info->info.ifcfg.sync_max_depth = POWERTEC_SYNC_DEPTH; - info->info.ifcfg.cntl3 = CNTL3_BS8 | CNTL3_FASTSCSI | CNTL3_FASTCLK; - info->info.ifcfg.disconnect_ok = 1; - info->info.ifcfg.wide_max_size = 0; - info->info.dma.setup = powertecscsi_dma_setup; - info->info.dma.pseudo = NULL; - info->info.dma.stop = powertecscsi_dma_stop; - - ecs[count]->irqaddr = (unsigned char *) - ioaddr(host->io_port + POWERTEC_INTR_STATUS); - ecs[count]->irqmask = POWERTEC_INTR_BIT; - ecs[count]->irq_data = (void *) - (host->io_port + POWERTEC_INTR_CONTROL); - ecs[count]->ops = (expansioncard_ops_t *)&powertecscsi_ops; - - request_region(host->io_port + POWERTEC_FAS216_OFFSET, - 16 << POWERTEC_FAS216_SHIFT, "powertec2-fas"); - - if (host->irq != NO_IRQ && - request_irq(host->irq, powertecscsi_intr, - SA_INTERRUPT, "powertec", host)) { - printk("scsi%d: IRQ%d not free, interrupts disabled\n", - host->host_no, host->irq); - host->irq = NO_IRQ; - info->info.scsi.irq = NO_IRQ; - } - - if (host->dma_channel != NO_DMA && - request_dma(host->dma_channel, "powertec")) { - printk("scsi%d: DMA%d not free, DMA disabled\n", - host->host_no, host->dma_channel); - host->dma_channel = NO_DMA; - } - - fas216_init(host); - ++count; - } - return count; -} - -/* Prototype: int powertecscsi_release(struct Scsi_Host * host) - * Purpose : releases all resources used by this adapter - * Params : host - driver host structure to return info for. - */ -int powertecscsi_release(struct Scsi_Host *host) -{ - int i; - - fas216_release(host); - - if (host->irq != NO_IRQ) - free_irq(host->irq, host); - if (host->dma_channel != NO_DMA) - free_dma(host->dma_channel); - release_region(host->io_port + POWERTEC_FAS216_OFFSET, - 16 << POWERTEC_FAS216_SHIFT); - - for (i = 0; i < MAX_ECARDS; i++) - if (ecs[i] && - host->io_port == ecard_address(ecs[i], ECARD_IOC, ECARD_FAST)) - ecard_release(ecs[i]); - return 0; -} - /* Prototype: const char *powertecscsi_info(struct Scsi_Host * host) * Purpose : returns a descriptive string about this interface, * Params : host - driver host structure to return info for. @@ -348,15 +202,12 @@ */ const char *powertecscsi_info(struct Scsi_Host *host) { - PowerTecScsi_Info *info = (PowerTecScsi_Info *)host->hostdata; - static char string[100], *p; + struct powertec_info *info = (struct powertec_info *)host->hostdata; + static char string[150]; - p = string; - p += sprintf(p, "%s ", host->hostt->name); - p += fas216_info(&info->info, p); - p += sprintf(p, "v%d.%d.%d terminators o%s", - VER_MAJOR, VER_MINOR, VER_PATCH, - info->control.terms ? "n" : "ff"); + sprintf(string, "%s (%s) in slot %d v%s terminators o%s", + host->hostt->name, info->info.scsi.type, info->ec->slot_no, + VERSION, info->term_ctl ? "n" : "ff"); return string; } @@ -409,31 +260,25 @@ int length, int host_no, int inout) { int pos, begin; - struct Scsi_Host *host = scsi_hostlist; - PowerTecScsi_Info *info; + struct Scsi_Host *host; + struct powertec_info *info; Scsi_Device *scd; - while (host) { - if (host->host_no == host_no) - break; - host = host->next; - } + host = scsi_host_hn_get(host_no); if (!host) return 0; if (inout == 1) return powertecscsi_set_proc_info(host, buffer, length); - info = (PowerTecScsi_Info *)host->hostdata; + info = (struct powertec_info *)host->hostdata; begin = 0; - pos = sprintf(buffer, - "PowerTec SCSI driver version %d.%d.%d\n", - VER_MAJOR, VER_MINOR, VER_PATCH); + pos = sprintf(buffer, "PowerTec SCSI driver v%s\n", VERSION); pos += fas216_print_host(&info->info, buffer + pos); pos += sprintf(buffer + pos, "Term : o%s\n", - info->control.terms ? "n" : "ff"); + info->term_ctl ? "n" : "ff"); pos += fas216_print_stats(&info->info, buffer + pos); @@ -458,27 +303,185 @@ return pos; } +static int powertecscsi_probe(struct expansion_card *ec); + +/* Prototype: int powertecscsi_detect(Scsi_Host_Template * tpnt) + * Purpose : initialises PowerTec SCSI driver + * Params : tpnt - template for this SCSI adapter + * Returns : >0 if host found, 0 otherwise. + */ +static int powertecscsi_detect(Scsi_Host_Template *tpnt) +{ + static const card_ids powertecscsi_cids[] = + { POWERTECSCSI_LIST, { 0xffff, 0xffff} }; + struct expansion_card *ec; + int count = 0, ret; + + ecard_startfind(); + + while (1) { + ec = ecard_find(0, powertecscsi_cids); + if (!ec) + break; + + ecard_claim(ec); + + ret = powertecscsi_probe(ec); + if (ret) { + ecard_release(ec); + break; + } + ++count; + } + return count; +} + +static void powertecscsi_remove(struct Scsi_Host *host); + +/* Prototype: int powertecscsi_release(struct Scsi_Host * host) + * Purpose : releases all resources used by this adapter + * Params : host - driver host structure to return info for. + */ +static int powertecscsi_release(struct Scsi_Host *host) +{ + powertecscsi_remove(host); + return 0; +} + static Scsi_Host_Template powertecscsi_template = { - module: THIS_MODULE, - proc_info: powertecscsi_proc_info, - name: "PowerTec SCSI", - detect: powertecscsi_detect, - release: powertecscsi_release, - info: powertecscsi_info, - bios_param: scsicam_bios_param, - can_queue: 1, - this_id: 7, - sg_tablesize: SG_ALL, - cmd_per_lun: 1, - use_clustering: ENABLE_CLUSTERING, - command: fas216_command, - queuecommand: fas216_queue_command, - eh_host_reset_handler: fas216_eh_host_reset, - eh_bus_reset_handler: fas216_eh_bus_reset, - eh_device_reset_handler: fas216_eh_device_reset, - eh_abort_handler: fas216_eh_abort, - use_new_eh_code: 1 + .module = THIS_MODULE, + .proc_info = powertecscsi_proc_info, + .name = "PowerTec SCSI", + .detect = powertecscsi_detect, + .release = powertecscsi_release, + .info = powertecscsi_info, + .bios_param = scsicam_bios_param, + .command = fas216_command, + .queuecommand = fas216_queue_command, + .eh_host_reset_handler = fas216_eh_host_reset, + .eh_bus_reset_handler = fas216_eh_bus_reset, + .eh_device_reset_handler = fas216_eh_device_reset, + .eh_abort_handler = fas216_eh_abort, + .use_new_eh_code = 1, + + .can_queue = 1, + .this_id = 7, + .sg_tablesize = SG_ALL, + .cmd_per_lun = 1, + .use_clustering = ENABLE_CLUSTERING, + .proc_name = "powertec", }; + +static int +powertecscsi_probe(struct expansion_card *ec) +{ + struct Scsi_Host *host; + struct powertec_info *info; + unsigned long base; + int ret; + + base = ecard_address(ec, ECARD_IOC, ECARD_FAST); + + request_region(base + POWERTEC_FAS216_OFFSET, + 16 << POWERTEC_FAS216_SHIFT, "powertec2-fas"); + + host = scsi_register(&powertecscsi_template, + sizeof (struct powertec_info)); + if (!host) { + ret = -ENOMEM; + goto out_region; + } + + host->io_port = base; + host->irq = ec->irq; + host->dma_channel = ec->dma; + + ec->irqaddr = (unsigned char *)ioaddr(base + POWERTEC_INTR_STATUS); + ec->irqmask = POWERTEC_INTR_BIT; + ec->irq_data = (void *)(base + POWERTEC_INTR_CONTROL); + ec->ops = (expansioncard_ops_t *)&powertecscsi_ops; + + info = (struct powertec_info *)host->hostdata; + info->ec = ec; + info->term_port = base + POWERTEC_TERM_CONTROL; + powertecscsi_terminator_ctl(host, term[ec->slot_no]); + + info->info.scsi.io_port = host->io_port + POWERTEC_FAS216_OFFSET; + info->info.scsi.io_shift = POWERTEC_FAS216_SHIFT; + info->info.scsi.irq = host->irq; + info->info.ifcfg.clockrate = 40; /* MHz */ + info->info.ifcfg.select_timeout = 255; + info->info.ifcfg.asyncperiod = 200; /* ns */ + info->info.ifcfg.sync_max_depth = 7; + info->info.ifcfg.cntl3 = CNTL3_BS8 | CNTL3_FASTSCSI | CNTL3_FASTCLK; + info->info.ifcfg.disconnect_ok = 1; + info->info.ifcfg.wide_max_size = 0; + info->info.ifcfg.capabilities = 0; + info->info.dma.setup = powertecscsi_dma_setup; + info->info.dma.pseudo = NULL; + info->info.dma.stop = powertecscsi_dma_stop; + + ret = fas216_init(host); + if (ret) + goto out_free; + + ret = request_irq(host->irq, powertecscsi_intr, + SA_INTERRUPT, "powertec", host); + if (ret) { + printk("scsi%d: IRQ%d not free: %d\n", + host->host_no, host->irq, ret); + goto out_release; + } + + if (host->dma_channel != NO_DMA) { + if (request_dma(host->dma_channel, "powertec")) { + printk("scsi%d: DMA%d not free, using PIO\n", + host->host_no, host->dma_channel); + host->dma_channel = NO_DMA; + } else { + set_dma_speed(host->dma_channel, 180); + info->info.ifcfg.capabilities |= FASCAP_DMA; + } + } + + ret = fas216_add(host); + if (ret == 0) + goto out; + + if (host->dma_channel != NO_DMA) + free_dma(host->dma_channel); + free_irq(host->irq, host); + + out_release: + fas216_release(host); + + out_free: + scsi_unregister(host); + + out_region: + release_region(base + POWERTEC_FAS216_OFFSET, + 16 << POWERTEC_FAS216_SHIFT); + + out: + return ret; +} + +static void powertecscsi_remove(struct Scsi_Host *host) +{ + struct powertec_info *info = (struct powertec_info *)host->hostdata; + + fas216_remove(host); + + if (host->dma_channel != NO_DMA) + free_dma(host->dma_channel); + free_irq(host->irq, host); + + release_region(host->io_port + POWERTEC_FAS216_OFFSET, + 16 << POWERTEC_FAS216_SHIFT); + + fas216_release(host); + ecard_release(info->ec); +} static int __init powertecscsi_init(void) { diff -urN orig/drivers/acorn/scsi/queue.c linux/drivers/acorn/scsi/queue.c --- orig/drivers/acorn/scsi/queue.c Fri Oct 26 16:46:02 2001 +++ linux/drivers/acorn/scsi/queue.c Wed Nov 13 14:09:51 2002 @@ -229,6 +229,27 @@ } /* + * Function: queue_remove_all_target(queue, target) + * Purpose : remove all SCSI commands from the queue for a specified target + * Params : queue - queue to remove command from + * target - target device id + * Returns : nothing + */ +void queue_remove_all_target(Queue_t *queue, int target) +{ + unsigned long flags; + struct list_head *l; + + spin_lock_irqsave(&queue->queue_lock, flags); + list_for_each(l, &queue->head) { + QE_t *q = list_entry(l, QE_t, list); + if (q->SCpnt->target == target) + __queue_remove(queue, l); + } + spin_unlock_irqrestore(&queue->queue_lock, flags); +} + +/* * Function: int queue_probetgtlun (queue, target, lun) * Purpose : check to see if we have a command in the queue for the specified * target/lun. @@ -290,6 +311,7 @@ EXPORT_SYMBOL(queue_remove_exclude); EXPORT_SYMBOL(queue_remove_tgtluntag); EXPORT_SYMBOL(queue_remove_cmd); +EXPORT_SYMBOL(queue_remove_all_target); EXPORT_SYMBOL(queue_probetgtlun); MODULE_AUTHOR("Russell King"); diff -urN orig/drivers/acorn/scsi/queue.h linux/drivers/acorn/scsi/queue.h --- orig/drivers/acorn/scsi/queue.h Tue Oct 3 20:08:24 2000 +++ linux/drivers/acorn/scsi/queue.h Wed Nov 13 14:09:51 2002 @@ -74,6 +74,15 @@ extern Scsi_Cmnd *queue_remove_tgtluntag (Queue_t *queue, int target, int lun, int tag); /* + * Function: queue_remove_all_target(queue, target) + * Purpose : remove all SCSI commands from the queue for a specified target + * Params : queue - queue to remove command from + * target - target device id + * Returns : nothing + */ +extern void queue_remove_all_target(Queue_t *queue, int target); + +/* * Function: int queue_probetgtlun (queue, target, lun) * Purpose : check to see if we have a command in the queue for the specified * target/lun. diff -urN orig/drivers/acorn/scsi/scsi.h linux/drivers/acorn/scsi/scsi.h --- orig/drivers/acorn/scsi/scsi.h Thu Jan 1 01:00:00 1970 +++ linux/drivers/acorn/scsi/scsi.h Fri Feb 21 16:51:30 2003 @@ -0,0 +1,123 @@ +/* + * linux/drivers/acorn/scsi/scsi.h + * + * Copyright (C) 2002 Russell King + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + * + * Commonly used scsi driver functions. + */ + +#define BELT_AND_BRACES + +/* + * The scatter-gather list handling. This contains all + * the yucky stuff that needs to be fixed properly. + */ +static inline int copy_SCp_to_sg(struct scatterlist *sg, Scsi_Pointer *SCp, int max) +{ + int bufs = SCp->buffers_residual; + + BUG_ON(bufs + 1 > max); + + sg->address = SCp->ptr; + sg->length = SCp->this_residual; + + if (bufs) + memcpy(sg + 1, SCp->buffer + 1, + sizeof(struct scatterlist) * bufs); + return bufs + 1; +} + +static inline int next_SCp(Scsi_Pointer *SCp) +{ + int ret = SCp->buffers_residual; + if (ret) { + SCp->buffer++; + SCp->buffers_residual--; + SCp->ptr = (unsigned char *)SCp->buffer->address; + SCp->this_residual = SCp->buffer->length; + } else { + SCp->ptr = NULL; + SCp->this_residual = 0; + } + return ret; +} + +static inline unsigned char get_next_SCp_byte(Scsi_Pointer *SCp) +{ + char c = *SCp->ptr; + + SCp->ptr += 1; + SCp->this_residual -= 1; + + return c; +} + +static inline void put_next_SCp_byte(Scsi_Pointer *SCp, unsigned char c) +{ + *SCp->ptr = c; + SCp->ptr += 1; + SCp->this_residual -= 1; +} + +static inline void init_SCp(Scsi_Cmnd *SCpnt) +{ + memset(&SCpnt->SCp, 0, sizeof(struct scsi_pointer)); + + if (SCpnt->use_sg) { + unsigned long len = 0; + int buf; + + SCpnt->SCp.buffer = (struct scatterlist *) SCpnt->buffer; + SCpnt->SCp.buffers_residual = SCpnt->use_sg - 1; + SCpnt->SCp.ptr = (char *) SCpnt->SCp.buffer->address; + SCpnt->SCp.this_residual = SCpnt->SCp.buffer->length; + +#ifdef BELT_AND_BRACES + /* + * Calculate correct buffer length. Some commands + * come in with the wrong request_bufflen. + */ + for (buf = 0; buf <= SCpnt->SCp.buffers_residual; buf++) + len += SCpnt->SCp.buffer[buf].length; + + if (SCpnt->request_bufflen != len) + printk(KERN_WARNING "scsi%d.%c: bad request buffer " + "length %d, should be %ld\n", SCpnt->host->host_no, + '0' + SCpnt->target, SCpnt->request_bufflen, len); + SCpnt->request_bufflen = len; +#endif + } else { + SCpnt->SCp.ptr = (unsigned char *)SCpnt->request_buffer; + SCpnt->SCp.this_residual = SCpnt->request_bufflen; + } + + /* + * If the upper SCSI layers pass a buffer, but zero length, + * we aren't interested in the buffer pointer. + */ + if (SCpnt->SCp.this_residual == 0 && SCpnt->SCp.ptr) { +#if 0 //def BELT_AND_BRACES + printk(KERN_WARNING "scsi%d.%c: zero length buffer passed for " + "command ", SCpnt->host->host_no, '0' + SCpnt->target); + print_command(SCpnt->cmnd); +#endif + SCpnt->SCp.ptr = NULL; + } +} + +static inline struct Scsi_Host *scsi_host_hn_get(int host_no) +{ + struct Scsi_Host *host = scsi_hostlist; + + while (host) { + if (host->host_no == host_no) + break; + host = host->next; + } + + return host; +} diff -urN orig/drivers/block/Makefile linux/drivers/block/Makefile --- orig/drivers/block/Makefile Mon Aug 5 13:30:41 2002 +++ linux/drivers/block/Makefile Mon Aug 5 16:00:30 2002 @@ -27,11 +27,17 @@ obj-$(CONFIG_BLK_DEV_PS2) += ps2esdi.o obj-$(CONFIG_BLK_DEV_XD) += xd.o obj-$(CONFIG_BLK_CPQ_DA) += cpqarray.o -obj-$(CONFIG_BLK_CPQ_CISS_DA) += cciss.o +obj-$(CONFIG_BLK_CPQ_CISS_DA) += cciss.o obj-$(CONFIG_BLK_DEV_DAC960) += DAC960.o obj-$(CONFIG_BLK_DEV_UMEM) += umem.o obj-$(CONFIG_BLK_DEV_NBD) += nbd.o subdir-$(CONFIG_PARIDE) += paride + +ifeq ($(CONFIG_ARCH_ACORN),y) +mod-subdirs += ../acorn/block +subdir-y += ../acorn/block +obj-y += ../acorn/block/acorn-block.o +endif include $(TOPDIR)/Rules.make diff -urN orig/drivers/block/ll_rw_blk.c linux/drivers/block/ll_rw_blk.c --- orig/drivers/block/ll_rw_blk.c Mon Aug 5 13:30:42 2002 +++ linux/drivers/block/ll_rw_blk.c Mon Aug 5 13:43:12 2002 @@ -31,6 +31,19 @@ #include #include +/* Maybe something to cleanup in 2.3? + * We shouldn't touch 0x3f2 on machines which don't have a PC floppy controller + * - it may contain something else which could cause a system hang. This is + * now selected by a configuration option, but maybe it ought to be in the + * floppy code itself? - rmk + */ +#if defined(__i386__) || (defined(__arm__) && defined(CONFIG_ARCH_ACORN)) +#define FLOPPY_BOOT_DISABLE +#endif +#ifdef CONFIG_BLK_DEV_FD +#undef FLOPPY_BOOT_DISABLE +#endif + /* * MAC Floppy IWM hooks */ @@ -428,7 +441,7 @@ elevator_init(&q->elevator, ELEVATOR_LINUS); blk_init_free_list(q); q->request_fn = rfn; - q->back_merge_fn = ll_back_merge_fn; + q->back_merge_fn = ll_back_merge_fn; q->front_merge_fn = ll_front_merge_fn; q->merge_requests_fn = ll_merge_requests_fn; q->make_request_fn = __make_request; @@ -1349,7 +1362,7 @@ mfm_init(); #endif #ifdef CONFIG_PARIDE - { extern void paride_init(void); paride_init(); }; + { extern void paride_init(void); paride_init(); } #endif #ifdef CONFIG_MAC_FLOPPY swim3_init(); @@ -1363,12 +1376,14 @@ #ifdef CONFIG_ATARI_FLOPPY atari_floppy_init(); #endif +#ifdef CONFIG_BLK_DEV_FD1772 + fd1772_init(); +#endif #ifdef CONFIG_BLK_DEV_FD floppy_init(); -#else -#if defined(__i386__) /* Do we even need this? */ - outb_p(0xc, 0x3f2); #endif +#ifdef FLOPPY_BOOT_DISABLE + outb_p(0xc, 0x3f2); #endif #ifdef CONFIG_CDU31A cdu31a_init(); @@ -1426,7 +1441,7 @@ jsfd_init(); #endif return 0; -}; +} EXPORT_SYMBOL(io_request_lock); EXPORT_SYMBOL(end_that_request_first); diff -urN orig/drivers/cdrom/cdrom.c linux/drivers/cdrom/cdrom.c --- orig/drivers/cdrom/cdrom.c Mon Aug 5 13:30:43 2002 +++ linux/drivers/cdrom/cdrom.c Mon Aug 5 13:43:16 2002 @@ -246,8 +246,8 @@ #define CD_DVD 0x80 /* Define this to remove _all_ the debugging messages */ -/* #define ERRLOGMASK CD_NOTHING */ -#define ERRLOGMASK (CD_WARNING) +#define ERRLOGMASK CD_NOTHING +/* #define ERRLOGMASK (CD_WARNING) */ /* #define ERRLOGMASK (CD_WARNING|CD_OPEN|CD_COUNT_TRACKS|CD_CLOSE) */ /* #define ERRLOGMASK (CD_WARNING|CD_REG_UNREG|CD_DO_IOCTL|CD_OPEN|CD_CLOSE|CD_COUNT_TRACKS) */ diff -urN orig/drivers/char/Config.in linux/drivers/char/Config.in --- orig/drivers/char/Config.in Mon Aug 5 13:30:43 2002 +++ linux/drivers/char/Config.in Thu Jan 23 20:07:01 2003 @@ -14,10 +14,10 @@ if [ "$CONFIG_IA64" = "y" ]; then bool 'Support for serial console port described by EFI HCDP table' CONFIG_SERIAL_HCDP fi - if [ "$CONFIG_ARCH_ACORN" = "y" ]; then - tristate ' Atomwide serial port support' CONFIG_ATOMWIDE_SERIAL - tristate ' Dual serial port support' CONFIG_DUALSP_SERIAL - fi +fi +if [ "$CONFIG_ARCH_ACORN" = "y" ]; then + dep_tristate ' Atomwide serial port support' CONFIG_ATOMWIDE_SERIAL $CONFIG_SERIAL + dep_tristate ' Dual serial port support' CONFIG_DUALSP_SERIAL $CONFIG_SERIAL fi dep_mbool 'Extended dumb serial driver options' CONFIG_SERIAL_EXTENDED $CONFIG_SERIAL if [ "$CONFIG_SERIAL_EXTENDED" = "y" ]; then @@ -84,18 +84,6 @@ bool ' Leave port 1 alone (for kgdb or audio)' CONFIG_SIBYTE_SB1250_DUART_NO_PORT_1 fi fi - fi -fi -if [ "$CONFIG_EXPERIMENTAL" = "y" -a "$CONFIG_ZORRO" = "y" ]; then - tristate 'Commodore A2232 serial support (EXPERIMENTAL)' CONFIG_A2232 -fi -if [ "$CONFIG_FOOTBRIDGE" = "y" ]; then - bool 'DC21285 serial port support' CONFIG_SERIAL_21285 - if [ "$CONFIG_SERIAL_21285" = "y" ]; then - if [ "$CONFIG_OBSOLETE" = "y" ]; then - bool ' Use /dev/ttyS0 device (OBSOLETE)' CONFIG_SERIAL_21285_OLD - fi - bool ' Console on DC21285 serial port' CONFIG_SERIAL_21285_CONSOLE fi if [ "$CONFIG_MIPS" = "y" ]; then bool ' TMPTX3912/PR31700 serial port support' CONFIG_SERIAL_TX3912 @@ -123,6 +111,13 @@ fi bool 'Enable Smart Card Reader 0 Support ' CONFIG_IT8172_SCR0 fi + +source drivers/serial/Config.in + +if [ "$CONFIG_ARCH_ANAKIN" = "y" ]; then + tristate 'Anakin touchscreen support' CONFIG_TOUCHSCREEN_ANAKIN +fi + bool 'Unix98 PTY support' CONFIG_UNIX98_PTYS if [ "$CONFIG_UNIX98_PTYS" = "y" ]; then int 'Maximum number of Unix98 PTYs in use (0-2048)' CONFIG_UNIX98_PTY_COUNT 256 @@ -137,6 +132,12 @@ source drivers/i2c/Config.in +if [ "$CONFIG_I2C" != "n" ]; then + dep_tristate ' DS1307 RTC' CONFIG_I2C_DS1307 $CONFIG_I2C +fi + +source drivers/l3/Config.in + mainmenu_option next_comment comment 'Mice' tristate 'Bus Mouse Support' CONFIG_BUSMOUSE @@ -182,11 +183,11 @@ tristate ' ALi M7101 PMU Watchdog Timer' CONFIG_ALIM7101_WDT tristate ' AMD "Elan" SC520 Watchdog Timer' CONFIG_SC520_WDT tristate ' Berkshire Products PC Watchdog' CONFIG_PCWATCHDOG - if [ "$CONFIG_FOOTBRIDGE" = "y" ]; then - tristate ' DC21285 watchdog' CONFIG_21285_WATCHDOG - if [ "$CONFIG_ARCH_NETWINDER" = "y" ]; then - tristate ' NetWinder WB83C977 watchdog' CONFIG_977_WATCHDOG - fi + if [ "$CONFIG_ARM" = "y" ]; then + dep_tristate ' DC21285 watchdog' CONFIG_21285_WATCHDOG $CONFIG_FOOTBRIDGE + dep_tristate ' NetWinder WB83C977 watchdog' CONFIG_977_WATCHDOG $CONFIG_ARCH_NETWINDER + dep_tristate ' SA1100 watchdog' CONFIG_SA1100_WATCHDOG $CONFIG_ARCH_SA1100 + dep_tristate ' Omaha watchdog' CONFIG_OMAHA_WATCHDOG $CONFIG_ARCH_OMAHA fi tristate ' Eurotech CPU-1220/1410 Watchdog Timer' CONFIG_EUROTECH_WDT tristate ' IB700 SBC Watchdog Timer' CONFIG_IB700_WDT @@ -234,6 +235,12 @@ fi if [ "$CONFIG_OBSOLETE" = "y" -a "$CONFIG_ALPHA_BOOK1" = "y" ]; then bool 'Tadpole ANA H8 Support' CONFIG_H8 +fi +if [ "$CONFIG_ARCH_SA1100" = "y" ]; then + tristate 'SA1100 Real Time Clock' CONFIG_SA1100_RTC +fi +if [ "$CONFIG_ARCH_OMAHA" = "y" ]; then + tristate 'Omaha Real Time Clock' CONFIG_OMAHA_RTC fi tristate 'Double Talk PC internal speech card support' CONFIG_DTLK diff -urN orig/drivers/char/Makefile linux/drivers/char/Makefile --- orig/drivers/char/Makefile Mon Aug 5 13:30:43 2002 +++ linux/drivers/char/Makefile Thu Oct 24 14:58:03 2002 @@ -72,11 +72,36 @@ endif ifeq ($(ARCH),arm) - ifneq ($(CONFIG_PC_KEYMAP),y) - KEYMAP = + KEYMAP := + KEYBD := + ifeq ($(CONFIG_PC_KEYMAP),y) + KEYMAP := defkeymap.o endif - ifneq ($(CONFIG_PC_KEYB),y) - KEYBD = + ifeq ($(CONFIG_PC_KEYB),y) + KEYBD += pc_keyb.o + endif + ifeq ($(CONFIG_KMI_KEYB),y) + KEYBD += amba_kmi_keyb.o + endif + ifeq ($(CONFIG_SA1111),y) + KEYBD += sa1111_keyb.o + endif + ifeq ($(CONFIG_ARCH_EDB7211),y) + KEYBD += edb7211_keyb.o + endif + ifeq ($(CONFIG_ARCH_AUTCPU12),y) + KEYMAP := defkeymap.o + KEYBD += clps711x_keyb.o + endif + ifeq ($(CONFIG_SA1100_GRAPHICSCLIENT),y) + KEYMAP = gckeymap.o + KEYBD += gc_keyb.o + endif + ifeq ($(CONFIG_SA1100_CERF_CPLD),y) + KEYBD += cerf_keyb.o + endif + ifeq ($(CONFIG_ARCH_FORTUNET),y) + KEYMAP := defkeymap.o endif endif @@ -139,10 +164,8 @@ obj-$(CONFIG_VT) += vt.o vc_screen.o consolemap.o consolemap_deftbl.o $(CONSOLE) selection.o obj-$(CONFIG_SERIAL) += $(SERIAL) obj-$(CONFIG_SERIAL_HCDP) += hcdp_serial.o -obj-$(CONFIG_SERIAL_21285) += serial_21285.o -obj-$(CONFIG_SERIAL_SA1100) += serial_sa1100.o -obj-$(CONFIG_SERIAL_AMBA) += serial_amba.o obj-$(CONFIG_TS_AU1000_ADS7846) += au1000_ts.o +obj-$(CONFIG_TOUCHSCREEN_ANAKIN) += anakin_ts.o ifndef CONFIG_SUN_KEYBOARD obj-$(CONFIG_VT) += keyboard.o $(KEYMAP) $(KEYBD) @@ -205,6 +228,8 @@ obj-$(CONFIG_MK712_MOUSE) += mk712.o obj-$(CONFIG_RTC) += rtc.o obj-$(CONFIG_EFI_RTC) += efirtc.o +obj-$(CONFIG_SA1100_RTC) += sa1100-rtc.o +obj-$(CONFIG_OMAHA_RTC) += omaha-rtc.o ifeq ($(CONFIG_PPC),) obj-$(CONFIG_NVRAM) += nvram.o endif @@ -235,6 +260,7 @@ obj-$(CONFIG_DZ) += dz.o obj-$(CONFIG_NWBUTTON) += nwbutton.o obj-$(CONFIG_NWFLASH) += nwflash.o +obj-$(CONFIG_SA1100_CONSUS) += consusbutton.o # Only one watchdog can succeed. We probe the hardware watchdog # drivers first, then the softdog driver. This means if your hardware @@ -261,11 +287,22 @@ obj-$(CONFIG_INDYDOG) += indydog.o obj-$(CONFIG_SC1200_WDT) += sc1200wdt.o obj-$(CONFIG_WAFER_WDT) += wafer5823wdt.o +obj-$(CONFIG_SA1100_WATCHDOG) += sa1100_wdt.o +obj-$(CONFIG_OMAHA_WATCHDOG) += omaha_wdt.o obj-$(CONFIG_SOFT_WATCHDOG) += softdog.o +# I2C char devices +obj-$(CONFIG_I2C_DS1307) += ds1307.o + subdir-$(CONFIG_MWAVE) += mwave ifeq ($(CONFIG_MWAVE),y) obj-y += mwave/mwave.o +endif + +ifeq ($(CONFIG_ARCH_ACORN),y) +mod-subdirs += ../acorn/char +subdir-y += ../acorn/char +obj-y += ../acorn/char/acorn-char.o endif include $(TOPDIR)/Rules.make diff -urN orig/drivers/char/amba_kmi_keyb.c linux/drivers/char/amba_kmi_keyb.c --- orig/drivers/char/amba_kmi_keyb.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/char/amba_kmi_keyb.c Wed Feb 27 11:50:07 2002 @@ -0,0 +1,999 @@ +/* + * linux/drivers/char/amba_kmi_keyb.c + * + * AMBA Keyboard and Mouse Interface Driver + * + * Copyright (C) 2000 Deep Blue Solutions Ltd. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + * + * This keyboard driver drives a PS/2 keyboard and mouse connected + * to the KMI interfaces. The KMI interfaces are nothing more than + * a uart; there is no inteligence in them to do keycode translation. + * We leave all that up to the keyboard itself. + * + * FIXES: + * dirk.uffmann@nokia.com: enabled PS/2 reconnection + */ +#include +#include +#include +#include /* for in_interrupt */ +#include +#include +#include /* for udelay */ +#include /* for keyboard_tasklet */ +#include + +#include +#include +#include +#include + +//#define DEBUG(s) printk s +#define DEBUG(s) do { } while (0) + +#define CONFIG_AMBA_PS2_RECONNECT + +#define KMI_BASE (kmi->base) + +#define KMI_RESET 0x00 +#define KMI_RESET_POR 0x01 +#define KMI_RESET_DONE 0x02 + +#define KMI_NO_ACK 0xffff + +#define PS2_O_RESET 0xff +#define PS2_O_RESEND 0xfe +#define PS2_O_DISABLE 0xf5 +#define PS2_O_ENABLE 0xf4 +#define PS2_O_ECHO 0xee + +/* + * Keyboard + */ +#define PS2_O_SET_DEFAULT 0xf6 +#define PS2_O_SET_RATE_DELAY 0xf3 +#define PS2_O_SET_SCANSET 0xf0 +#define PS2_O_INDICATORS 0xed + +/* + * Mouse + */ +#define PS2_O_SET_SAMPLE 0xf3 +#define PS2_O_SET_STREAM 0xea +#define PS2_O_SET_RES 0xe8 +#define PS2_O_SET_SCALE21 0xe7 +#define PS2_O_SET_SCALE11 0xe6 +#define PS2_O_REQ_STATUS 0xe9 + +/* + * Responses + */ +#define PS2_I_RESEND 0xfe +#define PS2_I_DIAGFAIL 0xfc +#define PS2_I_ACK 0xfa +#define PS2_I_BREAK 0xf0 +#define PS2_I_ECHO 0xee +#define PS2_I_BAT_OK 0xaa + +static char *kmi_type[] = { "Keyboard", "Mouse" }; + +static struct kmi_info *kmi_keyb; +static struct kmi_info *kmi_mouse; + +static inline void __kmi_send(struct kmi_info *kmi, u_int val) +{ + u_int status; + + do { + status = __raw_readb(KMISTAT); + } while (!(status & KMISTAT_TXEMPTY)); + + kmi->resend_count += 1; + __raw_writeb(val, KMIDATA); +} + +static void kmi_send(struct kmi_info *kmi, u_int val) +{ + kmi->last_tx = val; + kmi->resend_count = -1; + __kmi_send(kmi, val); +} + +static u_int kmi_send_and_wait(struct kmi_info *kmi, u_int val, u_int timeo) +{ + DECLARE_WAITQUEUE(wait, current); + + if (kmi->present == 0) + return KMI_NO_ACK; + + kmi->res = KMI_NO_ACK; + kmi->last_tx = val; + kmi->resend_count = -1; + + if (current->pid != 0 && !in_interrupt()) { + add_wait_queue(&kmi->wait_q, &wait); + set_current_state(TASK_UNINTERRUPTIBLE); + __kmi_send(kmi, val); + schedule_timeout(timeo); + current->state = TASK_RUNNING; + remove_wait_queue(&kmi->wait_q, &wait); + } else { + int i; + + __kmi_send(kmi, val); + for (i = 0; i < 1000; i++) { + if (kmi->res != KMI_NO_ACK) + break; + udelay(100); + } + } + + return kmi->res; +} + +/* + * This lot should probably be separated into a separate file... + */ +#ifdef CONFIG_KMI_MOUSE + +#include /* for struct file_ops */ +#include /* for poll_table */ +#include /* for struct miscdev */ +#include /* for add_mouse_randomness */ +#include /* for kmalloc */ +#include /* for {un,}lock_kernel */ +#include + +#include + +#define BUF_SZ 2048 + +static spinlock_t kmi_mouse_lock; +static int kmi_mouse_count; +static struct queue { + u_int head; + u_int tail; + struct fasync_struct *fasync; + unsigned char buf[BUF_SZ]; +} *queue; + +#define queue_empty() (queue->head == queue->tail) + +static u_char get_from_queue(void) +{ + unsigned long flags; + u_char res; + + spin_lock_irqsave(&kmi_mouse_lock, flags); + res = queue->buf[queue->tail]; + queue->tail = (queue->tail + 1) & (BUF_SZ-1); + spin_unlock_irqrestore(&kmi_mouse_lock, flags); + + return res; +} + +static ssize_t +kmi_mouse_read(struct file *file, char *buf, size_t count, loff_t *ppos) +{ + ssize_t i = count; + + if (queue_empty()) { + int ret; + + if (file->f_flags & O_NONBLOCK) + return -EAGAIN; + ret = wait_event_interruptible(kmi_mouse->wait_q, !queue_empty()); + if (ret) + return ret; + } + while (i > 0 && !queue_empty()) { + u_char c; + c = get_from_queue(); + put_user(c, buf++); + i--; + } + if (count - i) + file->f_dentry->d_inode->i_atime = CURRENT_TIME; + return count - i; +} + +static ssize_t +kmi_mouse_write(struct file *file, const char *buf, size_t count, loff_t *ppos) +{ + ssize_t retval = 0; + + if (count > 32) + count = 32; + + do { + char c; + get_user(c, buf++); + kmi_send_and_wait(kmi_mouse, c, HZ); + retval++; + } while (--count); + + if (retval) + file->f_dentry->d_inode->i_mtime = CURRENT_TIME; + + return retval; +} + +static unsigned int +kmi_mouse_poll(struct file *file, poll_table *wait) +{ + poll_wait(file, &kmi_mouse->wait_q, wait); + return (!queue_empty()) ? POLLIN | POLLRDNORM : 0; +} + +static int +kmi_mouse_release(struct inode *inode, struct file *file) +{ + lock_kernel(); + fasync_helper(-1, file, 0, &queue->fasync); + if (--kmi_mouse_count == 0) + kmi_send_and_wait(kmi_mouse, PS2_O_DISABLE, HZ); + unlock_kernel(); + return 0; +} + +static int +kmi_mouse_open(struct inode *inode, struct file *file) +{ + if (kmi_mouse_count++) + return 0; + queue->head = queue->tail = 0; + kmi_send_and_wait(kmi_mouse, PS2_O_ENABLE, HZ); + return 0; +} + +static int +kmi_mouse_fasync(int fd, struct file *filp, int on) +{ + int retval = fasync_helper(fd, filp, on, &queue->fasync); + if (retval > 0) + retval = 0; + return retval; +} + +static struct file_operations ps_fops = { + read: kmi_mouse_read, + write: kmi_mouse_write, + poll: kmi_mouse_poll, + open: kmi_mouse_open, + release: kmi_mouse_release, + fasync: kmi_mouse_fasync, +}; + +static struct miscdevice ps_mouse = { + minor: PSMOUSE_MINOR, + name: "psaux", + fops: &ps_fops, +}; + +static u_char kmi_mse_init_string[] = { + PS2_O_DISABLE, + PS2_O_SET_SAMPLE, 100, + PS2_O_SET_RES, 3, + PS2_O_SET_SCALE21 +}; + +/* + * The "normal" mouse scancode processing + */ +static void kmi_mse_intr(struct kmi_info *kmi, u_int val, struct pt_regs *regs) +{ + u_int head; + + add_mouse_randomness(val); + +#ifdef CONFIG_AMBA_PS2_RECONNECT + /* Try to detect a hot-plug event on the PS/2 mouse port */ + switch (kmi->hotplug_state) { + case 0: + /* Maybe we lost contact... */ + if (val == PS2_I_BAT_OK) { + kmi->hotplug_state++; + DEBUG(("%s: Saw 0xAA. Going to hotplug state %d\n", kmi->name, kmi->hotplug_state)); + } + break; + + case 1: + /* Again, maybe (but only maybe) we lost contact... */ + if (val == 0) { + kmi->hotplug_state++; + kmi_send(kmi, PS2_O_REQ_STATUS); + DEBUG(("%s: Got 0xAA 0x00. Sent Status Request\n", kmi->name)); + } else { + kmi->hotplug_state = 0; + DEBUG(("%s: No 0x00 followed 0xAA. No reconnect.\n", kmi->name)); + } + break; + + case 2: + /* Eat up acknowledge */ + if (val == PS2_I_ACK) + kmi->hotplug_state++; + else { + kmi->hotplug_state = 0; + DEBUG(("%s: didn't get ack (0x%2.2x)\n", kmi->name, val)); + } + break; + + case 3: + /* check if data reporting is still enabled, then no POR has happend */ + kmi->reconnect = !(val & 1<<5); + DEBUG(("%s: Data reporting disabled?: (%d)\n", kmi->name, kmi->reconnect)); + kmi->hotplug_state++; + DEBUG(("%s: Going to hotplug state %d\n", kmi->name, kmi->hotplug_state)); + break; + + case 4: + /* Eat up one status byte */ + kmi->hotplug_state++; + DEBUG(("%s: Going to hotplug state %d\n", kmi->name, kmi->hotplug_state)); + break; + + case 5: + /* Eat up another status byte */ + if (kmi->reconnect) { + kmi->config_num = 0; + kmi_send(kmi, kmi_mse_init_string[kmi->config_num]); + kmi->config_num++; + kmi->hotplug_state++; + DEBUG(("%s: Sending byte %d of PS/2 init string.\n", kmi->name, kmi->config_num)); + } else { + kmi->hotplug_state = 0; + DEBUG(("%s: False Alarm...\n", kmi->name)); + } + break; + + case 6: + if (val == PS2_I_ACK && kmi->config_num < sizeof(kmi_mse_init_string)) { + kmi_send(kmi, kmi_mse_init_string[kmi->config_num]); + kmi->config_num++; + DEBUG(("%s: Sending byte %d of PS/2 init string.\n", kmi->name, kmi->config_num)); + } else { + if (val == PS2_I_ACK) { + DEBUG(("%s: Now enable the mouse again...\n", kmi->name)); + queue->head = queue->tail = 0; + kmi_send(kmi, PS2_O_ENABLE); + kmi->hotplug_state++; + } else { + kmi->hotplug_state = 0; + DEBUG(("%s: didn't get ack (0x%2.2x)\n", kmi->name, val)); + } + } + break; + + case 7: + /* Eat up last acknowledge from enable */ + if (val == PS2_I_ACK) + printk(KERN_ERR "%s: reconnected\n", kmi->name); + else + DEBUG(("%s: didn't get ack (0x%2.2x)\n", kmi->name, val)); + + kmi->hotplug_state = 0; + break; + + } /* switch (kmi->hotplug_state) */ + + /* while inside hotplug mechanism, don't misinterpret values */ + if (kmi->hotplug_state > 2) + return; +#endif + + /* We are waiting for the mouse to respond to a kmi_send_and_wait() */ + if (kmi->res == KMI_NO_ACK) { + if (val == PS2_I_RESEND) { + if (kmi->resend_count < 5) + __kmi_send(kmi, kmi->last_tx); + else { + printk(KERN_ERR "%s: too many resends\n", kmi->name); + return; + } + } + + if (val == PS2_I_ACK) { + kmi->res = val; + wake_up(&kmi->wait_q); + } + return; + } + + /* The mouse autonomously send new data, so wake up mouse_read() */ + if (queue) { + head = queue->head; + queue->buf[head] = val; + head = (head + 1) & (BUF_SZ - 1); + if (head != queue->tail) { + queue->head = head; + kill_fasync(&queue->fasync, SIGIO, POLL_IN); + wake_up_interruptible(&kmi->wait_q); + } + } +} + +static int kmi_init_mouse(struct kmi_info *kmi) +{ + u_int ret, i; + + if (kmi->present) { + kmi->rx = kmi_mse_intr; + + for (i = 0; i < sizeof(kmi_mse_init_string); i++) { + ret = kmi_send_and_wait(kmi, kmi_mse_init_string[i], HZ); + if (ret != PS2_I_ACK) + printk("%s: didn't get ack (0x%2.2x)\n", + kmi->name, ret); + } + } + + queue = kmalloc(sizeof(*queue), GFP_KERNEL); + if (queue) { + memset(queue, 0, sizeof(*queue)); + misc_register(&ps_mouse); + ret = 0; + } else + ret = -ENOMEM; + + return ret; +} +#endif /* CONFIG_KMI_MOUSE */ + +/* + * The "program" we send to the keyboard to set it up how we want it: + * - default typematic delays + * - scancode set 1 + */ +static u_char kmi_kbd_init_string[] = { + PS2_O_DISABLE, + PS2_O_SET_DEFAULT, + PS2_O_SET_SCANSET, 0x01, + PS2_O_ENABLE +}; + +static void kmi_kbd_intr(struct kmi_info *kmi, u_int val, struct pt_regs *regs); + +static int __kmi_init_keyboard(struct kmi_info *kmi) +{ + u_int ret, i; + + if (!kmi->present) + return 0; + + kmi->rx = kmi_kbd_intr; + + for (i = 0; i < sizeof(kmi_kbd_init_string); i++) { + ret = kmi_send_and_wait(kmi, kmi_kbd_init_string[i], HZ); + if (ret != PS2_I_ACK) + printk("%s: didn't ack (0x%2.2x)\n", + kmi->name, ret); + } + + return 0; +} + +static void kmi_kbd_init_tasklet(unsigned long k) +{ + struct kmi_info *kmi = (struct kmi_info *)k; + __kmi_init_keyboard(kmi); +} + +static DECLARE_TASKLET_DISABLED(kmikbd_init_tasklet, kmi_kbd_init_tasklet, 0); + +/* + * The "normal" keyboard scancode processing + */ +static void kmi_kbd_intr(struct kmi_info *kmi, u_int val, struct pt_regs *regs) +{ +#ifdef CONFIG_AMBA_PS2_RECONNECT + /* Try to detect a hot-plug event on the PS/2 keyboard port */ + switch (kmi->hotplug_state) { + case 0: + /* Maybe we lost contact... */ + if (val == PS2_I_BAT_OK) { + kmi_send(kmi, PS2_O_SET_SCANSET); + kmi->hotplug_state++; + DEBUG(("%s: Saw 0xAA. Going to hotplug state %d\n", kmi->name, kmi->hotplug_state)); + } + break; + + case 1: + /* Eat up acknowledge */ + if (val == PS2_I_ACK) { + /* Request scan code set: '2' if POR has happend, '1' is false alarm */ + kmi_send(kmi, 0); + kmi->hotplug_state++; + } + else { + kmi->hotplug_state = 0; + DEBUG(("%s: didn't get ack (0x%2.2x)\n", kmi->name, val)); + } + break; + + case 2: + /* Eat up acknowledge */ + if (val == PS2_I_ACK) + kmi->hotplug_state++; + else { + kmi->hotplug_state = 0; + DEBUG(("%s: didn't get ack (0x%2.2x)\n", kmi->name, val)); + } + break; + + case 3: + kmi->hotplug_state = 0; + if (val == 2) { + DEBUG(("%s: POR detected. Scan code is: (%d)\n", kmi->name, val)); + kmi->present = 1; + tasklet_schedule(&kmikbd_init_tasklet); + printk(KERN_ERR "%s: reconnected\n", kmi->name); + return; + } + else + DEBUG(("%s: False Alarm...\n", kmi->name)); + break; + + } /* switch (kmi->hotplug_state) */ +#endif + + if (val == PS2_I_DIAGFAIL) { + printk(KERN_ERR "%s: diagnostic failed\n", kmi->name); + return; + } + + /* We are waiting for the keyboard to respond to a kmi_send_and_wait() */ + if (kmi->res == KMI_NO_ACK) { + if (val == PS2_I_RESEND) { + if (kmi->resend_count < 5) + __kmi_send(kmi, kmi->last_tx); + else { + printk(KERN_ERR "%s: too many resends\n", kmi->name); + return; + } + } + + if (val >= 0xee) { + kmi->res = val; + wake_up(&kmi->wait_q); + } + return; + } + +#ifdef CONFIG_VT + kbd_pt_regs = regs; + handle_scancode(val, !(val & 0x80)); + tasklet_schedule(&keyboard_tasklet); +#endif +} + +static void kmi_intr(int nr, void *devid, struct pt_regs *regs) +{ + struct kmi_info *kmi = devid; + u_int status = __raw_readb(KMIIR); + + if (status & KMIIR_RXINTR) { + u_int val = __raw_readb(KMIDATA); + + if (kmi->rx) + kmi->rx(kmi, val, regs); + } +} + +static int kmi_init_keyboard(struct kmi_info *kmi) +{ + kmikbd_init_tasklet.data = (unsigned long)kmi; + tasklet_enable(&kmikbd_init_tasklet); + + return __kmi_init_keyboard(kmi); +} + +/* + * Reset interrupt handler + */ +static void __init +kmi_reset_intr(struct kmi_info *kmi, u_int val, struct pt_regs *regs) +{ + if (kmi->state == KMI_RESET) { + if (val == PS2_I_ACK) + kmi->state = KMI_RESET_POR; + else { + val = KMI_NO_ACK; + goto finished; + } + } else if (kmi->state == KMI_RESET_POR) { +finished: + kmi->res = val; + kmi->state = KMI_RESET_DONE; + kmi->rx = NULL; + wake_up(&kmi->wait_q); + } +} + +/* + * Reset the device plugged into this interface + */ +static int __init kmi_reset(struct kmi_info *kmi) +{ + u_int res; + int ret = 0; + + kmi->state = KMI_RESET; + kmi->rx = kmi_reset_intr; + res = kmi_send_and_wait(kmi, PS2_O_RESET, HZ); + kmi->rx = NULL; + + if (res != PS2_I_BAT_OK) { + printk(KERN_ERR "%s: reset failed; ", kmi->name); + if (kmi->res != KMI_NO_ACK) + printk("code 0x%2.2x\n", kmi->res); + else + printk("no ack\n"); + ret = -EINVAL; + } + return ret; +} + +static int __init kmi_init_one_interface(struct kmi_info *kmi) +{ + u_int stat; + int ret = -ENODEV; + + init_waitqueue_head(&kmi->wait_q); + + printk(KERN_INFO "%s at 0x%8.8x on irq %d (%s)\n", kmi->name, + kmi->base, kmi->irq, kmi_type[kmi->type]); + + /* + * Initialise the KMI interface + */ + __raw_writeb(kmi->divisor, KMICLKDIV); + __raw_writeb(KMICR_EN, KMICR); + + /* + * Check that the data and clock lines are OK. + */ + stat = __raw_readb(KMISTAT); + if ((stat & (KMISTAT_IC|KMISTAT_ID)) != (KMISTAT_IC|KMISTAT_ID)) { + printk(KERN_ERR "%s: %s%s%sline%s stuck low\n", kmi->name, + (stat & KMISTAT_IC) ? "" : "clock ", + (stat & (KMISTAT_IC | KMISTAT_ID)) ? "" : "and ", + (stat & KMISTAT_ID) ? "" : "data ", + (stat & (KMISTAT_IC | KMISTAT_ID)) ? "" : "s"); + goto bad; + } + + /* + * Claim the appropriate interrupts + */ + ret = request_irq(kmi->irq, kmi_intr, 0, kmi->name, kmi); + if (ret) + goto bad; + + /* + * Enable the receive interrupt, and reset the device. + */ + __raw_writeb(KMICR_EN | KMICR_RXINTREN, KMICR); + kmi->present = 1; + kmi->present = kmi_reset(kmi) == 0; + + switch (kmi->type) { + case KMI_KEYBOARD: + ret = kmi_init_keyboard(kmi); + break; + +#ifdef CONFIG_KMI_MOUSE + case KMI_MOUSE: + ret = kmi_init_mouse(kmi); + break; +#endif + } + + return ret; + +bad: + /* + * Oh dear, the interface was bad, disable it. + */ + __raw_writeb(0, KMICR); + return ret; +} + +#ifdef CONFIG_VT +/* + * The fragment between #ifdef above and #endif * CONFIG_VT * + * is from the pc_keyb.c driver. It is not copyrighted under the + * above notice. This code is by various authors; please see + * drivers/char/pc_keyb.c for further information. + */ + +/* + * Translation of escaped scancodes to keycodes. + * This is now user-settable. + * The keycodes 1-88,96-111,119 are fairly standard, and + * should probably not be changed - changing might confuse X. + * X also interprets scancode 0x5d (KEY_Begin). + * + * For 1-88 keycode equals scancode. + */ + +#define E0_KPENTER 96 +#define E0_RCTRL 97 +#define E0_KPSLASH 98 +#define E0_PRSCR 99 +#define E0_RALT 100 +#define E0_BREAK 101 /* (control-pause) */ +#define E0_HOME 102 +#define E0_UP 103 +#define E0_PGUP 104 +#define E0_LEFT 105 +#define E0_RIGHT 106 +#define E0_END 107 +#define E0_DOWN 108 +#define E0_PGDN 109 +#define E0_INS 110 +#define E0_DEL 111 + +#define E1_PAUSE 119 + +/* BTC */ +#define E0_MACRO 112 +/* LK450 */ +#define E0_F13 113 +#define E0_F14 114 +#define E0_HELP 115 +#define E0_DO 116 +#define E0_F17 117 +#define E0_KPMINPLUS 118 +/* + * My OmniKey generates e0 4c for the "OMNI" key and the + * right alt key does nada. [kkoller@nyx10.cs.du.edu] + */ +#define E0_OK 124 +/* + * New microsoft keyboard is rumoured to have + * e0 5b (left window button), e0 5c (right window button), + * e0 5d (menu button). [or: LBANNER, RBANNER, RMENU] + * [or: Windows_L, Windows_R, TaskMan] + */ +#define E0_MSLW 125 +#define E0_MSRW 126 +#define E0_MSTM 127 + +static u_char e0_keys[128] = { + 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, + E0_KPENTER, E0_RCTRL, 0, 0, + 0, 0, 0, 0, + 0, 0, 0, 0, + 0, 0, 0, 0, + 0, 0, 0, 0, + 0, 0, 0, 0, + 0, E0_KPSLASH, 0, E0_PRSCR, + E0_RALT, 0, 0, 0, + 0, E0_F13, E0_F14, E0_HELP, + E0_DO, E0_F17, 0, 0, + 0, 0, E0_BREAK, E0_HOME, + E0_UP, E0_PGUP, 0, E0_LEFT, + E0_OK, E0_RIGHT, E0_KPMINPLUS, E0_END, + E0_DOWN, E0_PGDN, E0_INS, E0_DEL, + 0, 0, 0, 0, + 0, 0, 0, E0_MSLW, + E0_MSRW, E0_MSTM, 0, 0, + 0, 0, 0, 0, + 0, 0, 0, 0, + 0, 0, 0, 0, + 0, 0, 0, E0_MACRO, + 0, 0, 0, 0, + 0, 0, 0, 0, + 0, 0, 0, 0, + 0, 0, 0, 0 +}; + +#ifdef CONFIG_MAGIC_SYSRQ +u_char kmi_kbd_sysrq_xlate[128] = + "\000\0331234567890-=\177\t" /* 0x00 - 0x0f */ + "qwertyuiop[]\r\000as" /* 0x10 - 0x1f */ + "dfghjkl;'`\000\\zxcv" /* 0x20 - 0x2f */ + "bnm,./\000*\000 \000\201\202\203\204\205" /* 0x30 - 0x3f */ + "\206\207\210\211\212\000\000789-456+1" /* 0x40 - 0x4f */ + "230\177\000\000\213\214\000\000\000\000\000\000\000\000\000\000" /* 0x50 - 0x5f */ + "\r\000/"; /* 0x60 - 0x6f */ +#endif + +int kmi_kbd_setkeycode(u_int scancode, u_int keycode) +{ + if (scancode < 128 || scancode > 255 || keycode > 127) + return -EINVAL; + e0_keys[scancode - 128] = keycode; + return 0; +} + +int kmi_kbd_getkeycode(u_int scancode) +{ + if (scancode < 128 || scancode > 255) + return -EINVAL; + return e0_keys[scancode - 128]; +} + +int kmi_kbd_translate(u_char scancode, u_char *keycode, char raw_mode) +{ + static int prev_scancode = 0; + + /* special prefix scancodes.. */ + if (scancode == 0xe0 || scancode == 0xe1) { + prev_scancode = scancode; + return 0; + } + + /* 0xff is sent by a few keyboards, ignore it. 0x00 is error */ + if (scancode == 0x00 || scancode == 0xff) { + prev_scancode = 0; + return 0; + } + + scancode &= 0x7f; + + if (prev_scancode) { + int old_scancode = prev_scancode; + + prev_scancode = 0; + switch (old_scancode) { + case 0xe0: + /* + * The keyboard maintains its own internal caps lock + * and num lock status. In caps lock mode, E0 AA + * precedes make code and E0 2A follows break code. + * In numlock mode, E0 2A precedes make code, and + * E0 AA follows break code. We do our own book- + * keeping, so we will just ignore these. + * + * For my keyboard there is no caps lock mode, but + * there are both Shift-L and Shift-R modes. The + * former mode generates E0 2A / E0 AA pairs, the + * latter E0 B6 / E0 36 pairs. So, we should also + * ignore the latter. - aeb@cwi.nl + */ + if (scancode == 0x2a || scancode == 0x36) + return 0; + if (e0_keys[scancode]) + *keycode = e0_keys[scancode]; + else { + if (!raw_mode) + printk(KERN_INFO "kbd: unknown " + "scancode e0 %02x\n", + scancode); + return 0; + } + break; + + case 0xe1: + if (scancode == 0x1d) + prev_scancode = 0x100; + else { + if (!raw_mode) + printk(KERN_INFO "kbd: unknown " + "scancode e1 %02x\n", + scancode); + return 0; + } + break; + + case 0x100: + if (scancode == 0x45) + *keycode = E1_PAUSE; + else { + if (!raw_mode) + printk(KERN_INFO "kbd: unknown " + "scan code e1 1d %02x\n", + scancode); + return 0; + } + break; + } + } else + *keycode = scancode; + return 1; +} + +char kmi_kbd_unexpected_up(u_char keycode) +{ + return 0x80; +} + +void kmi_kbd_leds(u_char leds) +{ + struct kmi_info *kmi = kmi_keyb; + u_int ret; + + if (kmi) { + ret = kmi_send_and_wait(kmi, PS2_O_INDICATORS, HZ); + if (ret != KMI_NO_ACK) + ret = kmi_send_and_wait(kmi, leds, HZ); + if (ret == KMI_NO_ACK) + kmi->present = 0; + } +} + +int __init kmi_kbd_init(void) +{ + int ret = -ENODEV; + + if (kmi_keyb) { + strcpy(kmi_keyb->name, "kmikbd"); + ret = kmi_init_one_interface(kmi_keyb); + } + + if (ret == 0) { + k_setkeycode = kmi_kbd_setkeycode; + k_getkeycode = kmi_kbd_getkeycode; + k_translate = kmi_kbd_translate; + k_unexpected_up = kmi_kbd_unexpected_up; + k_leds = kmi_kbd_leds; +#ifdef CONFIG_MAGIC_SYSRQ + k_sysrq_xlate = kmi_kbd_sysrq_xlate; + k_sysrq_key = 0x54; +#endif + } + + return ret; +} + +#endif /* CONFIG_VT */ + +int register_kmi(struct kmi_info *kmi) +{ + struct kmi_info **kmip = NULL; + int ret; + + if (kmi->type == KMI_KEYBOARD) + kmip = &kmi_keyb; + else if (kmi->type == KMI_MOUSE) + kmip = &kmi_mouse; + + ret = -EINVAL; + if (kmip) { + ret = -EBUSY; + if (!*kmip) { + *kmip = kmi; + ret = 0; + } + } + + return ret; +} + +#ifdef CONFIG_KMI_MOUSE +static int __init kmi_init(void) +{ + int ret = -ENODEV; + + if (kmi_mouse) { + strcpy(kmi_mouse->name, "kmimouse"); + ret = kmi_init_one_interface(kmi_mouse); + } + + return ret; +} + +__initcall(kmi_init); +#endif diff -urN orig/drivers/char/anakin_ts.c linux/drivers/char/anakin_ts.c --- orig/drivers/char/anakin_ts.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/char/anakin_ts.c Tue Oct 23 17:06:03 2001 @@ -0,0 +1,208 @@ +/* + * linux/drivers/char/anakin_ts.c + * + * Copyright (C) 2001 Aleph One Ltd. for Acunia N.V. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + * + * Changelog: + * 18-Apr-2001 TTC Created + * 23-Oct-2001 dwmw2 Cleanup + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include + +/* + * TSBUF_SIZE must be a power of two + */ +#define ANAKIN_TS_MINOR 16 +#define TSBUF_SIZE 256 +#define NEXT(index) (((index) + 1) & (TSBUF_SIZE - 1)) + +static unsigned short buffer[TSBUF_SIZE][4]; +static int head, tail; +static DECLARE_WAIT_QUEUE_HEAD(queue); +static DECLARE_MUTEX(open_sem); +static spinlock_t tailptr_lock = SPIN_LOCK_UNLOCKED; +static struct fasync_struct *fasync; + +/* + * Interrupt handler and standard file operations + */ +static void +anakin_ts_handler(int irq, void *dev_id, struct pt_regs *regs) +{ + unsigned int status = __raw_readl(IO_BASE + IO_CONTROLLER + 0x24); + + /* + * iPAQ format (u16 pressure, x, y, millisecs) + */ + switch (status >> 20 & 3) { + case 0: + return; + case 2: + buffer[head][0] = 0; + break; + default: + buffer[head][0] = 0x7f; + } + + if (unlikely((volatile int)tail == NEXT(head))) { + /* Run out of space in the buffer. Move the tail pointer */ + spin_lock(&tailptr_lock); + + if ((volatile int)tail == NEXT(head)) { + tail = NEXT(NEXT(head)); + } + spin_unlock(&tailptr_lock); + } + + buffer[head][1] = status >> 2 & 0xff; + buffer[head][2] = status >> 12 & 0xff; + buffer[head][3] = jiffies; + mb(); + head = NEXT(head); + + wake_up_interruptible(&queue); + kill_fasync(&fasync, SIGIO, POLL_IN); + +} + +static ssize_t +anakin_ts_read(struct file *filp, char *buf, size_t count, loff_t *l) +{ + unsigned short data[4]; + ssize_t written = 0; + + if (head == tail) { + if (filp->f_flags & O_NONBLOCK) + return -EAGAIN; + if (wait_event_interruptible(queue, (volatile int)head != (volatile int)tail)) + return -ERESTARTSYS; + } + + while ((volatile int)head != (volatile int)tail && count >= sizeof data) { + /* Copy the data out with the spinlock held, so the + interrupt can't fill the buffer and move the tail + pointer while we're doing it */ + spin_lock_irq(&tailptr_lock); + + memcpy(data, buffer[tail], sizeof data); + tail = NEXT(tail); + + spin_unlock_irq(&tailptr_lock); + + if (copy_to_user(buf, data, sizeof data)) + return -EFAULT; + count -= sizeof data; + buf += sizeof data; + written += sizeof data; + } + return written ? written : -EINVAL; +} + +static unsigned int +anakin_ts_poll(struct file *filp, poll_table *wait) +{ + poll_wait(filp, &queue, wait); + return head != tail ? POLLIN | POLLRDNORM : 0; +} + +static int +anakin_ts_ioctl(struct inode *inode, struct file *filp, + unsigned int cmd, unsigned long arg) +{ + /* + * Future ioctl goes here + */ + return 0; +} + +static int +anakin_ts_open(struct inode *inode, struct file *filp) +{ + if (down_trylock(&open_sem)) + return -EBUSY; + return 0; +} + +static int +anakin_ts_fasync(int fd, struct file *filp, int on) +{ + return fasync_helper(fd, filp, on, &fasync); +} + +static int +anakin_ts_release(struct inode *inode, struct file *filp) +{ + anakin_ts_fasync(-1, filp, 0); + up(&open_sem); + return 0; +} + +static struct file_operations anakin_ts_fops = { + owner: THIS_MODULE, + read: anakin_ts_read, + poll: anakin_ts_poll, + ioctl: anakin_ts_ioctl, + open: anakin_ts_open, + release: anakin_ts_release, + fasync: anakin_ts_fasync, +}; + +static struct miscdevice anakin_ts_miscdev = { + ANAKIN_TS_MINOR, + "anakin_ts", + &anakin_ts_fops +}; + +/* + * Initialization and exit routines + */ +int __init +anakin_ts_init(void) +{ + int retval; + + if ((retval = request_irq(IRQ_TOUCHSCREEN, anakin_ts_handler, + SA_INTERRUPT, "anakin_ts", 0))) { + printk(KERN_WARNING "anakin_ts: failed to get IRQ\n"); + return retval; + } + __raw_writel(1, IO_BASE + IO_CONTROLLER + 8); + misc_register(&anakin_ts_miscdev); + + printk(KERN_NOTICE "Anakin touchscreen driver initialised\n"); + + return 0; +} + +void __exit +anakin_ts_exit(void) +{ + __raw_writel(0, IO_BASE + IO_CONTROLLER + 8); + free_irq(IRQ_TOUCHSCREEN, 0); + misc_deregister(&anakin_ts_miscdev); +} + +module_init(anakin_ts_init); +module_exit(anakin_ts_exit); + +MODULE_AUTHOR("Tak-Shing Chan "); +MODULE_DESCRIPTION("Anakin touchscreen driver"); +MODULE_SUPPORTED_DEVICE("touchscreen/anakin"); diff -urN orig/drivers/char/cerf_keyb.c linux/drivers/char/cerf_keyb.c --- orig/drivers/char/cerf_keyb.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/char/cerf_keyb.c Wed Oct 17 16:13:27 2001 @@ -0,0 +1,380 @@ +/* + cerf_keyb.c: This is the end. Daniel is writing a device driver!!! +*/ +#include + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include + +#include + +#define KBD_REPORT_UNKN + +#define KBD_REPORT_ERR /* Report keyboard errors */ +#define KBD_REPORT_UNKN /* Report unknown scan codes */ +#define KBD_REPORT_TIMEOUTS /* Report keyboard timeouts */ +#define KBD_NO_DATA (-1) /* No data */ +#define KBD_REPEAT_START (0x20) +#define KBD_REPEAT_CONTINUE (0x05) +#define KBD_KEY_DOWN_MAX (0x10) +#define UINT_LEN (20) +#define SC_LIM (69) +#define KBD_ROWS (5) +#define KBD_COLUMNS (8) + +#define KBD_KEYUP (0x80) +#define KBD_MODESCAN (0x7f) +#define KBD_CAPSSCAN (0x3a) +#define KBD_SHIFTSCAN (0x2a) +#define KBD_NUMCURSCAN (0x7c) +#define KBD_CTRLSCAN (0x1d) +#define KBD_ALTSCAN (0x38) + +#define KBD_UP_OFF (0) +#define KBD_UP_ON (1) +#define KBD_DOWN (2) +#define KBD_DOWN_HOLD (3) + + + +static unsigned char handle_kbd_event(void); +static unsigned char kbd_read_input(void); +static void column_set(unsigned int column); +static int scancodes(unsigned char codeval[KBD_ROWS][KBD_COLUMNS]); + +static spinlock_t kbd_controller_lock = SPIN_LOCK_UNLOCKED; +static struct timer_list kbd_timer; + +static short mode_ena = 0; +static short numcur_ena = 0; +static short shift_ena = 0; + +#define E0_KPENTER 96 +#define E0_RCTRL 97 +#define E0_KPSLASH 98 +#define E0_PRSCR 99 +#define E0_RALT 100 +#define E0_BREAK 101 /* (control-pause) */ +#define E0_HOME 102 +#define E0_UP 103 +#define E0_PGUP 104 +#define E0_LEFT 105 +#define E0_RIGHT 106 +#define E0_END 107 +#define E0_DOWN 108 +#define E0_PGDN 109 +#define E0_INS 110 +#define E0_DEL 111 +#define E1_PAUSE 119 +#define E0_MACRO 112 +#define E0_F13 113 +#define E0_F14 114 +#define E0_HELP 115 +#define E0_DO 116 +#define E0_F17 117 +#define E0_KPMINPLUS 118 +#define E0_OK 124 +#define E0_MSLW 125 +#define E0_MSRW 126 +#define E0_MSTM 127 + +static unsigned char e0_keys[128] = { + 0, 0, 0, 0, 0, 0, 0, 0, /* 0x00-0x07 */ + 0, 0, 0, 0, 0, 0, 0, 0, /* 0x08-0x0f */ + 0, 0, 0, 0, 0, 0, 0, 0, /* 0x10-0x17 */ + 0, 0, 0, 0, E0_KPENTER, E0_RCTRL, 0, 0, /* 0x18-0x1f */ + 0, 0, 0, 0, 0, 0, 0, 0, /* 0x20-0x27 */ + 0, 0, 0, 0, 0, 0, 0, 0, /* 0x28-0x2f */ + 0, 0, 0, 0, 0, E0_KPSLASH, 0, E0_PRSCR, /* 0x30-0x37 */ + E0_RALT, 0, 0, 0, 0, E0_F13, E0_F14, E0_HELP, /* 0x38-0x3f */ + E0_DO, E0_F17, 0, 0, 0, 0, E0_BREAK, E0_HOME, /* 0x40-0x47 */ + E0_UP, E0_PGUP, 0, E0_LEFT, E0_OK, E0_RIGHT, E0_KPMINPLUS, E0_END,/* 0x48-0x4f */ + E0_DOWN, E0_PGDN, E0_INS, E0_DEL, 0, 0, 0, 0, /* 0x50-0x57 */ + 0, 0, 0, E0_MSLW, E0_MSRW, E0_MSTM, 0, 0, /* 0x58-0x5f */ + 0, 0, 0, 0, 0, 0, 0, 0, /* 0x60-0x67 */ + 0, 0, 0, 0, 0, 0, 0, E0_MACRO, /* 0x68-0x6f */ + 0, 0, 0, 0, 0, 0, 0, 0, /* 0x70-0x77 */ + 0, 0, 0, 0, 0, 0, 0, 0 /* 0x78-0x7f */ +}; + +static unsigned char cerf_normal_map[KBD_ROWS][KBD_COLUMNS] = { + {KBD_ALTSCAN, KBD_MODESCAN, 0x1e, 0x30, 0x2e, 0x20, 0x00, 0x00}, + {0x12, 0x21, 0x22, 0x23, 0x17, 0x24, 0x25, 0x00}, + {0x26, 0x32, 0x31, 0x18, 0x19, 0x10, 0x13, 0x00}, + {0x1f, 0x14, 0x16, 0x2f, 0x11, 0x2d, 0x15, 0x00}, + {0x2c, KBD_SHIFTSCAN, KBD_CTRLSCAN, 0x39, KBD_NUMCURSCAN, 0x2b, 0x1c, 0x00} +}; + +static unsigned char cerf_mode_map[KBD_ROWS][KBD_COLUMNS] = { + {0x00, 0x00, 0x02, 0x03, 0x04, 0x05, 0x00, 0x00}, + {0x06, 0x07, 0x08, 0x09, 0x0a, 0x0b, 0x00, 0x00}, // + {0x0d, 0x0c, 0x37, 0x35, 0x0d, 0x48, 0x28, 0x00}, + {0x01, 0x33, 0x34, 0x00, 0x4b, 0x27, 0x4d, 0x00}, // + {0x0f, 0x00, KBD_CAPSSCAN, 0x0e, 0x00, 0x50, 0x00, 0x00} +}; + +static unsigned char cerf_numcur_map[KBD_ROWS][KBD_COLUMNS] = { + {0x00, 0x00, 0x02, 0x03, 0x04, 0x05, 0x00, 0x00}, + {0x06, 0x07, 0x08, 0x09, 0x0a, 0x0b, 0x00, 0x00}, + {0x0d, 0x0c, 0x00, 0x00, 0x00, 0x48, 0x00, 0x00}, + {0x00, 0x00, 0x00, 0x00, 0x4b, 0x00, 0x4d, 0x00}, + {0x00, 0x00, 0x00, 0x00, 0x00, 0x50, 0x00, 0x00} +}; + +static void column_set(unsigned int column) +{ + if (column < 0) + { + CERF_PDA_CPLD_UnSet(CERF_PDA_CPLD_KEYPAD_A, 0xFF, 0xFF); + CERF_PDA_CPLD_UnSet(CERF_PDA_CPLD_KEYPAD_B, 0xFF, 0xFF); + } + else + { + if(column < 4) + { + CERF_PDA_CPLD_Set(CERF_PDA_CPLD_KEYPAD_A, 1 << (column % 4), 0xFF); + CERF_PDA_CPLD_UnSet(CERF_PDA_CPLD_KEYPAD_B, 0xFF, 0xFF); + } + else + { + CERF_PDA_CPLD_UnSet(CERF_PDA_CPLD_KEYPAD_A, 0xFF, 0xFF); + CERF_PDA_CPLD_Set(CERF_PDA_CPLD_KEYPAD_B, 1 << (column % 4), 0xFF); + } + } +} + +static int scancodes(unsigned char codeval[KBD_ROWS][KBD_COLUMNS]) +{ + int i, j; + + for(i = 0; i < KBD_COLUMNS; i++) + { + column_set(i); + udelay(50); + for(j = 0; j < KBD_ROWS; j++) + { + if(mode_ena) + codeval[j][i] = (GPLR & (1 << (20 + j)))?(cerf_mode_map[j][i]?cerf_mode_map[j][i]:cerf_normal_map[j][i]):0; + else if(numcur_ena) + codeval[j][i] = (GPLR & (1 << (20 + j)))?(cerf_numcur_map[j][i]?cerf_numcur_map[j][i]:cerf_normal_map[j][i]):0; + else + codeval[j][i] = (GPLR & (1 << (20 + j)))?cerf_normal_map[j][i]:0; + } + } + column_set(-1); + + return 0; +} + +static unsigned char kbd_read_input(void) +{ + int i, j, k, l; + unsigned char prev; + static unsigned char count = 0; + + static unsigned char oldcodes[KBD_ROWS][KBD_COLUMNS]={{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}, + {0,0,0,0,0,0,0,0}}; + unsigned char inputcode[KBD_ROWS][KBD_COLUMNS]; + + memset(inputcode, 0, sizeof(unsigned char) * (KBD_ROWS * KBD_COLUMNS)); + scancodes(inputcode); + + for(i = 0; i < KBD_COLUMNS; i++) + { + for(j = 0; j < KBD_ROWS; j++) + { +// if(oldcodes[j][i] == 0xe0) +// oldcodes[j][i] = + if(oldcodes[j][i] != inputcode[j][i]) + { + // Value of the key before entering this function + prev = oldcodes[j][i]; + + // KEYUP + if(inputcode[j][i] == 0 && oldcodes[j][i] != 0 && !(oldcodes[j][i] & KBD_KEYUP)) + { + oldcodes[j][i] |= KBD_KEYUP; + + if(mode_ena == KBD_UP_ON) + mode_ena = KBD_UP_OFF; + if(prev == KBD_MODESCAN) + if(mode_ena == KBD_DOWN_HOLD) + mode_ena = KBD_UP_OFF; + else if(mode_ena == KBD_DOWN) + mode_ena = KBD_UP_ON; + if(mode_ena == KBD_DOWN) + mode_ena = KBD_DOWN_HOLD; + } + // RESET KEYUP + else if(oldcodes[j][i] & KBD_KEYUP) + oldcodes[j][i] = 0; + // KEY DOWN + else + { + oldcodes[j][i] = inputcode[j][i]; + + // Parse out mode modifiers before the keyboard interpreter can touch them + if(inputcode[j][i] == KBD_MODESCAN) + { + if(!mode_ena) + mode_ena = KBD_DOWN; + continue; + } + if(inputcode[j][i] == KBD_NUMCURSCAN) + { + numcur_ena = numcur_ena?0:1; + continue; + } + } + //printk("Modified: (%#x,%#x), ipv:%#x, To: (%#.2x), From: (%#.2x), Flags:%d,%d,%d\r\n", j, i, inputcode[j][i], oldcodes[j][i], prev, mode_ena, shift_ena, numcur_ena); + return oldcodes[j][i]; + } + } + } + + return (unsigned char)(KBD_NO_DATA); +} + +int cerf_kbd_translate(unsigned char scancode, unsigned char *keycode, + char raw_mode) +{ + static int prev_scancode; + + if (scancode == 0xe0 || scancode == 0xe1) { + prev_scancode = scancode; + return 0; + } + + if (scancode == 0x00 || scancode == 0xff) { + prev_scancode = 0; + return 0; + } + + scancode &= 0x7f; + + if (prev_scancode) { + if (prev_scancode != 0xe0) { + if (prev_scancode == 0xe1 && scancode == 0x1d) { + prev_scancode = 0x100; + return 0; + } else if (prev_scancode == 0x100 && scancode == 0x45) { + prev_scancode = 0; + } else { +#ifdef KBD_REPORT_UNKN + if (!raw_mode) + printk(KERN_INFO "keyboard: unknown e1 escape sequence\n"); +#endif + prev_scancode = 0; + return 0; + } + } else { + prev_scancode = 0; + if (scancode == 0x2a || scancode == 0x36) + return 0; + else { +#ifdef KBD_REPORT_UNKN + if (!raw_mode) + printk(KERN_INFO "keyboard: unknown scancode e0 %02x\n", + scancode); +#endif + return 0; + } + } + } else + *keycode = scancode; + return 1; +} + +static inline void handle_keyboard_event(unsigned char scancode) +{ + if(scancode != (unsigned char)(KBD_NO_DATA)) + { +#ifdef CONFIG_VT + handle_scancode(scancode, !(scancode & KBD_KEYUP)); +#endif + tasklet_schedule(&keyboard_tasklet); + } +} + +static unsigned char handle_kbd_event(void) +{ + unsigned char scancode; + + scancode = kbd_read_input(); + handle_keyboard_event(scancode); + + return 0; +} + +/* Handle the automatic interrupts handled by the timer */ +static void keyboard_interrupt(unsigned long foo) +{ + spin_lock_irq(&kbd_controller_lock); + handle_kbd_event(); + spin_unlock_irq(&kbd_controller_lock); + + kbd_timer.expires = 8 + jiffies; + kbd_timer.data = 0x00000000; + kbd_timer.function = (void(*)(unsigned long))&keyboard_interrupt; + + add_timer(&kbd_timer); +} + +void cerf_leds(unsigned char leds) +{ +} +char cerf_unexpected_up(unsigned char keycode) +{ +return 0; +} +int cerf_getkeycode(unsigned int scancode) +{ +return 0; +} +int cerf_setkeycode(unsigned int scancode, unsigned int keycode) +{ +return 0; +} + +void cerf_kbd_init_hw(void) +{ + printk("Starting Cerf PDA Keyboard Driver... "); + + k_setkeycode = cerf_setkeycode; + k_getkeycode = cerf_getkeycode; + k_translate = cerf_kbd_translate; + k_unexpected_up = cerf_unexpected_up; + k_leds = cerf_leds; + + GPDR &= ~(GPIO_GPIO(20) | GPIO_GPIO(21) | GPIO_GPIO(22) | GPIO_GPIO(23) | GPIO_GPIO(24)); + kbd_timer.expires = 40 + jiffies; + kbd_timer.data = 0x00000000; + kbd_timer.function = (void(*)(unsigned long))&keyboard_interrupt; + + add_timer(&kbd_timer); + + printk("Done\r\n"); +} diff -urN orig/drivers/char/clps711x_keyb.c linux/drivers/char/clps711x_keyb.c --- orig/drivers/char/clps711x_keyb.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/char/clps711x_keyb.c Thu Nov 29 20:14:55 2001 @@ -0,0 +1,547 @@ +/* + * drivers/char/clps711x_keyb.c + * + * Copyright (C) 2001 Thomas Gleixner + * + * based on drivers/edb7211_keyb.c, which is copyright (C) 2000 Bluemug Inc. + * + * Keyboard driver for ARM Linux on EP7xxx and CS89712 processors + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. See the file COPYING + * in the main directory of this archive for more details. + * + * + * Hardware: + * + * matrix scan keyboards based on EP7209,7211,7212,7312 and CS89712 + * on chip keyboard scanner. + * Adaption for different machines is done in init function. + * + * Basic Function: + * + * Basicly the driver is interrupt driven. It sets all column drivers + * high. If any key is pressed, a interrupt occures. Now a seperate scan of + * each column is done. This scan is timer based, because we use a keyboard + * interface with decoupling capacitors (neccecary if you want to survive + * EMC compliance tests). Always one line is set high. When next timer event + * occures the scan data on port A are valid. This makes also sure, that no + * spurious keys are scanned. The kbd int on these CPU's is not deglitched! + * After scanning all columns, we switch back to int mode, if no key is + * pressed. If any is pressed we reschedule the scan within a programmable + * delay. If we would switch back to interrupt mode as long as a key is pressed, + * we come right back to the interrupt, because the int. is level triggered ! + * The timer based scan of the seperate columns can also be done in one + * timer event (set fastscan to 1). + * + * Summary: + * The design of this keyboard controller chip is stupid at all ! + * + * Matrix translation: + * The matrix translation table is based on standard XT scancodes. Maybe + * you have to adjust the KEYISPRINTABLE macro if you set other codes. + * + * HandyKey: + * + * On small matrix keyboards you don't have enough keys for operation. + * The intention was to implement a operation mode as it's used on handys. + * You can rotate trough four scancode levels and produce e.g. with a 4x3 + * matrix 4*3*4 = 48 different keycodes. That's basicly enough for editing + * filenames or things like that. The HandyKey function takes care about + * nonprintable keys like cursors, backspace, del ... + * If a key is pressed and is a printable keycode, the code is put to the + * main keyboard handler and a cursor left is applied. If you press the same + * key again, the current character is deleted and the next level character + * is applied. (e.g. 1, a, b, c, 1 ....). If you press a different key, the + * driver applies cursor right, before processing the new key. + * The autocomplete feature moves the cursor right, if you do not press a + * key within a programmable time. + * If HandyKey is off, the keyboard behaviour is that of a standard keyboard + * HandyKey can be en/disabled from userspace with the proc/keyboard entry + * + * proc/keyboard: + * + * Read access gives back the actual state of the HandyKey function + * h:0 Disabled + * h:1 Enabled + * Write access has two functions. Changing the HandyKey mode and applying + * a different scancode translation table. + * Syntax is: h:0 disable Handykey + * h:1 enabled Handykey + * t:array[256] of bytes Transfer translation table + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include + +#include +#include + +void clps711x_kbd_init_hw(void); + +/* + * Values for the keyboard column scan control register. + */ +#define KBSC_HI 0x0 /* All driven high */ +#define KBSC_LO 0x1 /* All driven low */ +#define KBSC_X 0x2 /* All high impedance */ +#define KBSC_COL0 0x8 /* Column 0 high, others high impedance */ +#define KBSC_COL1 0x9 /* Column 1 high, others high impedance */ +#define KBSC_COL2 0xa /* Column 2 high, others high impedance */ +#define KBSC_COL3 0xb /* Column 3 high, others high impedance */ +#define KBSC_COL4 0xc /* Column 4 high, others high impedance */ +#define KBSC_COL5 0xd /* Column 5 high, others high impedance */ +#define KBSC_COL6 0xe /* Column 6 high, others high impedance */ +#define KBSC_COL7 0xf /* Column 7 high, others high impedance */ + +/* +* Keycodes for cursor left/right and delete (used by HandyKey) +*/ +#define KEYCODE_CLEFT 0x4b +#define KEYCODE_CRIGHT 0x4d +#define KEYCODE_DEL 0x53 +#define KEYISPRINTABLE(code) ( (code > 0x01 && code < 0x37 && code != 0x1c \ + && code != 0x0e) || code == 0x39) + +/* Simple translation table for the SysRq keys */ +#ifdef CONFIG_MAGIC_SYSRQ +unsigned char clps711x_kbd_sysrq_xlate[128] = + "\000\0331234567890-=\177\t" /* 0x00 - 0x0f */ + "qwertyuiop[]\r\000as" /* 0x10 - 0x1f */ + "dfghjkl;'`\000\\zxcv" /* 0x20 - 0x2f */ + "bnm,./\000*\000 \000\201\202\203\204\205" /* 0x30 - 0x3f */ + "\206\207\210\211\212\000\000789-456+1" /* 0x40 - 0x4f */ + "230\177\000\000\213\214\000\000\000\000\000\000\000\000\000\000" /* 0x50 - 0x5f */ + "\r\000/"; /* 0x60 - 0x6f */ +#endif + +/* + * This table maps row/column keyboard matrix positions to XT scancodes. + * It's a default table, which can be overriden by writing to proc/keyboard + */ +#ifdef CONFIG_ARCH_AUTCPU12 +static unsigned char autcpu12_scancode[256] = +{ +/* Column: + Row 0 1 2 3 4 5 6 7 */ +/* A0 */ 0x08, 0x09, 0x0a, 0x0e, 0x05, 0x06, 0x00, 0x00, +/* A1 */ 0x07, 0x53, 0x02, 0x03, 0x04, 0x0f, 0x00, 0x00, +/* A2 */ 0x0c, 0x0b, 0x33, 0x1c, 0xff, 0x4b, 0x00, 0x00, +/* A3 */ 0x48, 0x50, 0x4d, 0x3b, 0x3c, 0x3d, 0x00, 0x00, +/* A4 */ 0x3e, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, +/* A5 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, +/* A6 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, +/* A7 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + +/* A0 */ 0x1e, 0x20, 0x22, 0x0e, 0x24, 0x32, 0x00, 0x00, +/* A1 */ 0x19, 0x53, 0x1f, 0x2f, 0x15, 0x0f, 0x00, 0x00, +/* A2 */ 0x0c, 0x39, 0x34, 0x1c, 0xff, 0x4b, 0x00, 0x00, +/* A3 */ 0x48, 0x50, 0x4d, 0x3b, 0x3c, 0x3d, 0x00, 0x00, +/* A4 */ 0x3e, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, +/* A5 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, +/* A6 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, +/* A7 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + +/* A0 */ 0x30, 0x12, 0x23, 0x0e, 0x25, 0x31, 0x00, 0x00, +/* A1 */ 0x10, 0x53, 0x14, 0x11, 0x2c, 0x0f, 0x00, 0x00, +/* A2 */ 0x0c, 0x0b, 0x27, 0x1c, 0xff, 0x4b, 0x00, 0x00, +/* A3 */ 0x48, 0x50, 0x4d, 0x3b, 0x3c, 0x3d, 0x00, 0x00, +/* A4 */ 0x3e, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, +/* A5 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, +/* A6 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, +/* A7 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + +/* A0 */ 0x2e, 0x21, 0x17, 0x0e, 0x26, 0x18, 0x00, 0x00, +/* A1 */ 0x13, 0x53, 0x16, 0x2D, 0x04, 0x0f, 0x00, 0x00, +/* A2 */ 0x0c, 0x39, 0x35, 0x1c, 0xff, 0x4b, 0x00, 0x00, +/* A3 */ 0x48, 0x50, 0x4d, 0x3b, 0x3c, 0x3d, 0x00, 0x00, +/* A4 */ 0x3e, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, +/* A5 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, +/* A6 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, +/* A7 */ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, +}; +#endif + +static int keys[8]; +static int new_keys[8]; +static int previous_keys[8]; + +static int fastscan; +static int scan_interval; +static int scan_delay; +static int last_column; +static int key_is_pressed; + +static unsigned char *act_scancode; + +static struct kbd_handy_key { + int ena; + int code; + int shift; + int autocomplete; + unsigned long expires; + unsigned long delay; + unsigned char left; + unsigned char right; + unsigned char del; +} khandy; + +static struct tq_struct kbd_process_task; +static struct timer_list clps711x_kbd_timer; +static struct timer_list clps711x_kbdhandy_timer; +static struct proc_dir_entry *clps711x_keyboard_proc_entry = NULL; + +/* + * Translate a raw keycode to an XT keyboard scancode. + */ +static int clps711x_translate(unsigned char scancode, unsigned char *keycode, + char raw_mode) +{ + *keycode = act_scancode[scancode]; + return 1; +} + +/* +* Initialize handykey structure +* clear code, clear shift +* scan scancode for cursor right/left and delete +*/ +static void clps711x_handykey_init(void) { + + int i; + + khandy.ena = 0; + khandy.code = 0; + khandy.shift = 0; + khandy.autocomplete = 0; + for(i = 0; i < 64; i++) { + switch(act_scancode[i]) { + case KEYCODE_CLEFT: khandy.left = i; break; + case KEYCODE_CRIGHT: khandy.right = i; break; + case KEYCODE_DEL: khandy.del = i; break; + } + } +} + +/* +* Check for handy key and process it +*/ +void inline clps711x_checkhandy(int col, int row) { + + int scode, down; + unsigned char kcode; + + scode = (row<<3) + col; + down = keys[col]>>row & 0x01; + kcode = act_scancode[scode]; + + if (!khandy.ena) { + if (khandy.code) { + handle_scancode(khandy.right,1); + handle_scancode(khandy.right,0); + } + khandy.code = 0; + khandy.shift = 0; + khandy.autocomplete = 0; + } + + if(!kcode) + return; + + if (!down || !khandy.ena) { + if (khandy.ena && KEYISPRINTABLE(act_scancode[scode])) + khandy.autocomplete = 1; + else + handle_scancode(scode + khandy.shift, down); + return; + } + + khandy.autocomplete = 0; + if (KEYISPRINTABLE(kcode)) { + if (khandy.code) { + if(khandy.code == (scode|0x100)) { + handle_scancode(khandy.del,1); + handle_scancode(khandy.del,0); + khandy.shift = khandy.shift < 3*64 ? khandy.shift + 64 : 0 ; + } else { + handle_scancode(khandy.right,1); + handle_scancode(khandy.right,0); + khandy.shift = 0; + } + } + handle_scancode(scode + khandy.shift, 1); + handle_scancode(scode + khandy.shift, 0); + khandy.code = scode | 0x100; + handle_scancode(khandy.left,1); + handle_scancode(khandy.left,0); + } else { + if (khandy.code) { + khandy.code = 0; + handle_scancode(khandy.right,1); + handle_scancode(khandy.right,0); + } + khandy.shift = 0; + handle_scancode(scode, down); + } +} + + +/* + * Process the new key data + */ +static void clps711x_kbd_process(void* data) +{ + int col,row,res; + + for (col = 0; col < 8; col++) { + if (( res = previous_keys[col] ^ keys[col]) == 0) + continue; + for(row = 0; row < 8; row++) { + if ( ((res >> row) & 0x01) != 0) + clps711x_checkhandy(col,row); + } + } + /* Update the state variables. */ + memcpy(previous_keys, keys, 8 * sizeof(int)); + + /* reschedule, if autocomplete pending */ + if (khandy.autocomplete) { + khandy.expires = jiffies + khandy.delay; + mod_timer(&clps711x_kbdhandy_timer,khandy.expires); + } + +} + +static char clps711x_unexpected_up(unsigned char scancode) +{ + return 0200; +} + +/* +* Handle timer event, for autocomplete function +* Reschedule keyboard process task +*/ +static void clps711x_kbdhandy_timeout(unsigned long data) +{ + if(khandy.autocomplete) { + khandy.code = 0; + khandy.shift = 0; + khandy.autocomplete = 0; + handle_scancode(khandy.right,1); + handle_scancode(khandy.right,0); + } +} + +/* +* Handle timer event, while in pollmode +*/ +static void clps711x_kbd_timeout(unsigned long data) +{ + int i; + unsigned long flags; + /* + * read bits of actual column or all columns in fastscan-mode + */ + for (i = 0; i < 8; i++) { + new_keys[last_column - KBSC_COL0] = clps_readb(PADR) & 0xff; + key_is_pressed |= new_keys[last_column - KBSC_COL0]; + last_column = last_column < KBSC_COL7 ? last_column + 1 : KBSC_COL0; + local_irq_save(flags); + clps_writel( (clps_readl(SYSCON1) & ~SYSCON1_KBDSCANMASK) + | last_column, SYSCON1); + local_irq_restore(flags); + /* + * For fastscan, apply a short delay to settle scanlines + * else break and wait for next timeout + */ + if (fastscan) + udelay(5); + else + break; + } + + if (key_is_pressed) + khandy.autocomplete = 0; + + /* + * switch to interupt mode, if all columns scanned and no key pressed + * else reschedule scan + */ + if (last_column == KBSC_COL0) { + if (!key_is_pressed) { + local_irq_save(flags); + clps_writel( (clps_readl(SYSCON1) & ~SYSCON1_KBDSCANMASK) + | KBSC_HI, SYSCON1); + local_irq_restore(flags); + clps_writel(0,KBDEOI); + enable_irq(IRQ_KBDINT); + } else { + clps711x_kbd_timer.expires = jiffies + scan_interval; + add_timer(&clps711x_kbd_timer); + } + key_is_pressed = 0; + memcpy(keys, new_keys, 8 * sizeof(int)); + for (i = 0; i < 8; i++) { + if (previous_keys[i] != keys[i]) { + queue_task(&kbd_process_task, &tq_timer); + return; + } + } + } else { + clps711x_kbd_timer.expires = jiffies + scan_delay; + add_timer(&clps711x_kbd_timer); + } +} + +/* +* Keyboard interrupt, change to scheduling mode +*/ +static void clps711x_kbd_int(int irq, void *dev_id, struct pt_regs *regs) +{ + +#ifdef CONFIG_VT + kbd_pt_regs = regs; +#endif + disable_irq(IRQ_KBDINT); + khandy.autocomplete = 0; + clps_writel( (clps_readl(SYSCON1) & ~SYSCON1_KBDSCANMASK) + | KBSC_COL0, SYSCON1); + clps711x_kbd_timer.expires = jiffies + scan_delay; + add_timer(&clps711x_kbd_timer); +} + + +static int clps711x_kbd_proc_keyboard_read(char *page, char **start, off_t off, + int count, int *eof, void *data) +{ + if (count < 2) + return -EINVAL; + + return sprintf(page,"h:%d\n",khandy.ena); +} + +static int clps711x_kbd_proc_keyboard_write(struct file *file, const char *buffer, + unsigned long count, void *data) +{ + unsigned char buf[260]; + + if (count < 3|| count > 258) + return -EINVAL; + if (copy_from_user(buf, buffer, count)) + return -EFAULT; + if (buf[1] != ':') + return -EINVAL; + + if (buf[0] == 'h') { + switch (buf[2]) { + case '0': + case '1': + case '2': khandy.ena = buf[2]-'0'; return count; + } + } + + if (buf[0] == 't' && count == 258) { + memcpy(act_scancode,buf+2,256); + /* rescan cursor left/right and del */ + clps711x_handykey_init(); + return count; + } + + return -EINVAL; +} + + +/* + * Initialize the keyboard hardware. + * Set all columns high + * Install interrupt handler + * + * Machine dependent parameters: + * + * fastscan: 0 = timer based scan for each column + * 1 = full scan is done in one timer event + * scan_delay: time between column scans + * setup even if you use fastscan (leeds to timer mode) + * scan_interval: time between full scans + * handy.delay: timeout before last entry get's automatically valid + * + */ +void __init clps711x_kbd_init_hw(void) +{ + + /* + * put here machine dependent init stuff + */ + if (machine_is_autcpu12()) { + fastscan = 0; + scan_interval = 50*HZ/1000; + scan_delay = 20*HZ/1000; + khandy.delay = 750*HZ/1000; + act_scancode = autcpu12_scancode; + } else { + printk("No initialization, keyboard killed\n"); + return; + } + + last_column = KBSC_COL0; + key_is_pressed = 0; + + clps711x_handykey_init(); + + /* Register the /proc entry */ + clps711x_keyboard_proc_entry = create_proc_entry("keyboard", 0444, + &proc_root); + if (clps711x_keyboard_proc_entry == NULL) + printk("Couldn't create the /proc entry for the keyboard\n"); + else { + clps711x_keyboard_proc_entry->read_proc = + &clps711x_kbd_proc_keyboard_read; + clps711x_keyboard_proc_entry->write_proc = + &clps711x_kbd_proc_keyboard_write; + } + + /* Initialize the matrix processing task. */ + k_translate = clps711x_translate; + k_unexpected_up = clps711x_unexpected_up; + kbd_process_task.routine = clps711x_kbd_process; + kbd_process_task.data = 0; + + /* Setup the timer for keyboard polling, after kbd int */ + init_timer(&clps711x_kbd_timer); + clps711x_kbd_timer.function = clps711x_kbd_timeout; + clps711x_kbd_timer.data = 0; + init_timer(&clps711x_kbdhandy_timer); + clps711x_kbdhandy_timer.function = clps711x_kbdhandy_timeout; + clps711x_kbdhandy_timer.data = 1; + + /* Initialise scan hardware, request int */ + clps_writel( (clps_readl(SYSCON1) & ~SYSCON1_KBDSCANMASK) + | KBSC_HI, SYSCON1); + request_irq(IRQ_KBDINT, clps711x_kbd_int, 0,"keyboard", NULL); + + printk("clps711x keyboard init done\n"); + +} diff -urN orig/drivers/char/console.c linux/drivers/char/console.c --- orig/drivers/char/console.c Mon Aug 5 13:30:43 2002 +++ linux/drivers/char/console.c Mon Aug 5 13:43:18 2002 @@ -72,8 +72,14 @@ * * Removed console_lock, enabled interrupts across all console operations * 13 March 2001, Andrew Morton + * + * Split out con_write_ctrl_* functions from do_con_write & changed + * vc_state to function pointer + * by Russell King , July 1998 */ +#define CONSOLE_WIP + #include #include #include @@ -226,7 +232,7 @@ static inline unsigned short *screenpos(int currcons, int offset, int viewed) { unsigned short *p; - + if (!viewed) p = (unsigned short *)(origin + offset); else if (!sw->con_screen_pos) @@ -720,7 +726,7 @@ else { unsigned short *p = (unsigned short *) kmalloc(ss, GFP_USER); if (!p) { - for (i = first; i < currcons; i++) + for (i = first; i< currcons; i++) if (newscreens[i]) kfree(newscreens[i]); return -ENOMEM; @@ -838,7 +844,7 @@ /* the default colour table, for VGA+ colour systems */ int default_red[] = {0x00,0xaa,0x00,0xaa,0x00,0xaa,0x00,0xaa, 0x55,0xff,0x55,0xff,0x55,0xff,0x55,0xff}; -int default_grn[] = {0x00,0x00,0xaa,0x55,0x00,0x00,0xaa,0xaa, +int default_grn[] = {0x00,0x00,0xaa,0xaa,0x00,0x00,0xaa,0xaa, 0x55,0x55,0xff,0xff,0x55,0x55,0xff,0xff}; int default_blu[] = {0x00,0x00,0x00,0x00,0xaa,0xaa,0xaa,0xaa, 0x55,0x55,0x55,0x55,0xff,0xff,0xff,0xff}; @@ -1384,6 +1390,19 @@ need_wrap = 0; } +static int con_write_ctrl_ESnormal(int, struct tty_struct *, unsigned int); +static int con_write_ctrl_ESesc(int, struct tty_struct *, unsigned int); +static int con_write_ctrl_ESnonstd(int, struct tty_struct *, unsigned int); +static int con_write_ctrl_ESpalette(int, struct tty_struct *, unsigned int); +static int con_write_ctrl_ESsquare(int, struct tty_struct *, unsigned int); +static int con_write_ctrl_ESgetpars(int, struct tty_struct *, unsigned int); +static int con_write_ctrl_ESgotpars(int, struct tty_struct *, unsigned int); +static int con_write_ctrl_ESpercent(int, struct tty_struct *, unsigned int); +static int con_write_ctrl_ESfunckey(int, struct tty_struct *, unsigned int); +static int con_write_ctrl_EShash(int, struct tty_struct *, unsigned int); +static int con_write_ctrl_ESsetG0(int, struct tty_struct *, unsigned int); +static int con_write_ctrl_ESsetG1(int, struct tty_struct *, unsigned int); + enum { ESnormal, ESesc, ESsquare, ESgetpars, ESgotpars, ESfunckey, EShash, ESsetG0, ESsetG1, ESpercent, ESignore, ESnonstd, ESpalette }; @@ -1393,7 +1412,7 @@ { top = 0; bottom = video_num_lines; - vc_state = ESnormal; + vc_state = con_write_ctrl_ESnormal; ques = 0; translate = set_translate(LAT1_MAP,currcons); G0_charset = LAT1_MAP; @@ -1488,328 +1507,426 @@ disp_ctrl = 0; return; case 24: case 26: - vc_state = ESnormal; + vc_state = con_write_ctrl_ESnormal; return; case 27: - vc_state = ESesc; + vc_state = con_write_ctrl_ESesc; return; case 127: del(currcons); return; case 128+27: - vc_state = ESsquare; + vc_state = con_write_ctrl_ESsquare; return; } - switch(vc_state) { - case ESesc: - vc_state = ESnormal; - switch (c) { - case '[': - vc_state = ESsquare; - return; - case ']': - vc_state = ESnonstd; - return; - case '%': - vc_state = ESpercent; - return; - case 'E': - cr(currcons); - lf(currcons); - return; - case 'M': - ri(currcons); - return; - case 'D': - lf(currcons); - return; - case 'H': - tab_stop[x >> 5] |= (1 << (x & 31)); - return; - case 'Z': - respond_ID(tty); - return; - case '7': - save_cur(currcons); - return; - case '8': - restore_cur(currcons); - return; - case '(': - vc_state = ESsetG0; - return; - case ')': - vc_state = ESsetG1; - return; - case '#': - vc_state = EShash; - return; - case 'c': - reset_terminal(currcons,1); - return; - case '>': /* Numeric keypad */ - clr_kbd(kbdapplic); - return; - case '=': /* Appl. keypad */ - set_kbd(kbdapplic); - return; - } - return; - case ESnonstd: - if (c=='P') { /* palette escape sequence */ - for (npar=0; npar='0'&&c<='9') || (c>='A'&&c<='F') || (c>='a'&&c<='f') ) { - par[npar++] = (c>'9' ? (c&0xDF)-'A'+10 : c-'0') ; - if (npar==7) { - int i = par[0]*3, j = 1; - palette[i] = 16*par[j++]; - palette[i++] += par[j++]; - palette[i] = 16*par[j++]; - palette[i++] += par[j++]; - palette[i] = 16*par[j++]; - palette[i] += par[j]; - set_palette(currcons); - vc_state = ESnormal; - } - } else - vc_state = ESnormal; - return; - case ESsquare: - for(npar = 0 ; npar < NPAR ; npar++) - par[npar] = 0; - npar = 0; - vc_state = ESgetpars; - if (c == '[') { /* Function key */ - vc_state=ESfunckey; - return; + vc_state(currcons, tty, c); +} + +static int con_write_utf(int currcons, int c) +{ + unsigned int chr; + + /* Combine UTF-8 into Unicode */ + /* Incomplete characters silently ignored */ + if (c < 0x80) { + utf_count = 0; + return c; + } + + if (utf_count > 0 && (c & 0xc0) == 0x80) { + chr = (utf_char << 6) | (c & 0x3f); + utf_count--; + if (utf_count == 0) + return chr; + } else { + unsigned int count; + if ((c & 0xe0) == 0xc0) { + count = 1; + chr = (c & 0x1f); + } else if ((c & 0xf0) == 0xe0) { + count = 2; + chr = (c & 0x0f); + } else if ((c & 0xf8) == 0xf0) { + count = 3; + chr = (c & 0x07); + } else if ((c & 0xfc) == 0xf8) { + count = 4; + chr = (c & 0x03); + } else if ((c & 0xfe) == 0xfc) { + count = 5; + chr = (c & 0x01); + } else { + count = 0; + chr = 0; } - ques = (c=='?'); - if (ques) - return; - case ESgetpars: - if (c==';' && npar='0' && c<='9') { - par[npar] *= 10; - par[npar] += c-'0'; - return; - } else vc_state=ESgotpars; - case ESgotpars: - vc_state = ESnormal; - switch(c) { - case 'h': - set_mode(currcons,1); - return; - case 'l': - set_mode(currcons,0); - return; - case 'c': - if (ques) { - if (par[0]) - cursor_type = par[0] | (par[1]<<8) | (par[2]<<16); - else - cursor_type = CUR_DEFAULT; - return; - } - break; - case 'm': - if (ques) { - clear_selection(); - if (par[0]) - complement_mask = par[0]<<8 | par[1]; - else - complement_mask = s_complement_mask; - return; - } - break; - case 'n': - if (!ques) { - if (par[0] == 5) - status_report(tty); - else if (par[0] == 6) - cursor_report(currcons,tty); - } - return; + utf_count = count; + } + + utf_char = chr; + return -1; +} + +static int con_write_ctrl_ESnormal(int currcons, struct tty_struct *tty, unsigned int c) +{ + return 0; +} + +static int con_write_ctrl_ESesc(int currcons, struct tty_struct *tty, unsigned int c) +{ + vc_state = con_write_ctrl_ESnormal; + switch (c) { + case '[': + vc_state = con_write_ctrl_ESsquare; + break; + case ']': + vc_state = con_write_ctrl_ESnonstd; + break; + case '%': + vc_state = con_write_ctrl_ESpercent; + break; + case 'E': + cr(currcons); + lf(currcons); + break; + case 'M': + ri(currcons); + break; + case 'D': + lf(currcons); + break; + case 'H': + tab_stop[x >> 5] |= (1 << (x & 31)); + break; + case 'Z': + respond_ID(tty); + break; + case '7': + save_cur(currcons); + break; + case '8': + restore_cur(currcons); + return 1; + case '(': + vc_state = con_write_ctrl_ESsetG0; + break; + case ')': + vc_state = con_write_ctrl_ESsetG1; + break; + case '#': + vc_state = con_write_ctrl_EShash; + break; + case 'c': + reset_terminal(currcons,1); + return 1; + case '>': /* Numeric keypad */ + clr_kbd(kbdapplic); + break; + case '=': /* Appl. keypad */ + set_kbd(kbdapplic); + break; + } + return 0; +} + +static int con_write_ctrl_ESnonstd(int currcons, struct tty_struct *tty, unsigned int c) +{ + switch (c) { + case 'P': /* palette escape sequence */ + for (npar=0; npar='0'&&c<='9') || (c>='A'&&c<='F') || (c>='a'&&c<='f') ) { + par[npar++] = (c>'9' ? (c&0xDF)-'A'+10 : c-'0') ; + if (npar==7) { + int i = par[0]*3, j = 1; + palette[i] = 16*par[j++]; + palette[i++] += par[j++]; + palette[i] = 16*par[j++]; + palette[i++] += par[j++]; + palette[i] = 16*par[j++]; + palette[i] += par[j]; + set_palette(currcons); + vc_state = con_write_ctrl_ESnormal; } + } else + vc_state = con_write_ctrl_ESnormal; + return 0; +} + +static int con_write_ctrl_ESsquare(int currcons, struct tty_struct *tty, unsigned int c) +{ + for(npar = 0 ; npar < NPAR ; npar++) + par[npar] = 0; + npar = 0; + vc_state = con_write_ctrl_ESgetpars; + if (c == '[') { /* Function key */ + vc_state = con_write_ctrl_ESfunckey; + return 0; + } + ques = (c=='?'); + if (ques) + return 0; + return con_write_ctrl_ESgetpars(currcons, tty, c); +} + +static int con_write_ctrl_ESgetpars(int currcons, struct tty_struct *tty, unsigned int c) +{ + if (c==';' && npar='0' && c<='9') { + par[npar] *= 10; + par[npar] += c-'0'; + return 0; + } else vc_state = con_write_ctrl_ESgotpars; + return con_write_ctrl_ESgotpars(currcons, tty, c); +} + +static int con_write_ctrl_ESgotpars(int currcons, struct tty_struct *tty, unsigned int c) +{ + vc_state = con_write_ctrl_ESnormal; + switch(c) { + case 'h': + set_mode(currcons,1); + return 0; + case 'l': + set_mode(currcons,0); + return 0; + case 'c': if (ques) { - ques = 0; - return; + if (par[0]) + cursor_type = par[0] | (par[1]<<8) | (par[2]<<16); + else + cursor_type = CUR_DEFAULT; + return 0; } - switch(c) { - case 'G': case '`': - if (par[0]) par[0]--; - gotoxy(currcons,par[0],y); - return; - case 'A': - if (!par[0]) par[0]++; - gotoxy(currcons,x,y-par[0]); - return; - case 'B': case 'e': - if (!par[0]) par[0]++; - gotoxy(currcons,x,y+par[0]); - return; - case 'C': case 'a': - if (!par[0]) par[0]++; - gotoxy(currcons,x+par[0],y); - return; - case 'D': - if (!par[0]) par[0]++; - gotoxy(currcons,x-par[0],y); - return; - case 'E': - if (!par[0]) par[0]++; - gotoxy(currcons,0,y+par[0]); - return; - case 'F': - if (!par[0]) par[0]++; - gotoxy(currcons,0,y-par[0]); - return; - case 'd': - if (par[0]) par[0]--; - gotoxay(currcons,x,par[0]); - return; - case 'H': case 'f': - if (par[0]) par[0]--; - if (par[1]) par[1]--; - gotoxay(currcons,par[1],par[0]); - return; - case 'J': - csi_J(currcons,par[0]); - return; - case 'K': - csi_K(currcons,par[0]); - return; - case 'L': - csi_L(currcons,par[0]); - return; - case 'M': - csi_M(currcons,par[0]); - return; - case 'P': - csi_P(currcons,par[0]); - return; - case 'c': - if (!par[0]) - respond_ID(tty); - return; - case 'g': - if (!par[0]) - tab_stop[x >> 5] &= ~(1 << (x & 31)); - else if (par[0] == 3) { - tab_stop[0] = - tab_stop[1] = - tab_stop[2] = - tab_stop[3] = - tab_stop[4] = 0; - } - return; - case 'm': - csi_m(currcons); - return; - case 'q': /* DECLL - but only 3 leds */ - /* map 0,1,2,3 to 0,1,2,4 */ - if (par[0] < 4) - setledstate(kbd_table + currcons, - (par[0] < 3) ? par[0] : 4); - return; - case 'r': - if (!par[0]) - par[0]++; - if (!par[1]) - par[1] = video_num_lines; - /* Minimum allowed region is 2 lines */ - if (par[0] < par[1] && - par[1] <= video_num_lines) { - top=par[0]-1; - bottom=par[1]; - gotoxay(currcons,0,0); - } - return; - case 's': - save_cur(currcons); - return; - case 'u': - restore_cur(currcons); - return; - case 'X': - csi_X(currcons, par[0]); - return; - case '@': - csi_at(currcons,par[0]); - return; - case ']': /* setterm functions */ - setterm_command(currcons); - return; + break; + case 'm': + if (ques) { + clear_selection(); + if (par[0]) + complement_mask = par[0]<<8 | par[1]; + else + complement_mask = s_complement_mask; + return 0; } - return; - case ESpercent: - vc_state = ESnormal; - switch (c) { - case '@': /* defined in ISO 2022 */ - utf = 0; - return; - case 'G': /* prelim official escape code */ - case '8': /* retained for compatibility */ - utf = 1; - return; + break; + case 'n': + if (!ques) { + if (par[0] == 5) + status_report(tty); + else if (par[0] == 6) + cursor_report(currcons,tty); } - return; - case ESfunckey: - vc_state = ESnormal; - return; - case EShash: - vc_state = ESnormal; - if (c == '8') { - /* DEC screen alignment test. kludge :-) */ - video_erase_char = - (video_erase_char & 0xff00) | 'E'; - csi_J(currcons, 2); - video_erase_char = - (video_erase_char & 0xff00) | ' '; - do_update_region(currcons, origin, screenbuf_size/2); + return 0; + } + if (ques) { + ques = 0; + return 0; + } + switch(c) { + case 'G': case '`': + if (par[0]) par[0]--; + gotoxy(currcons,par[0],y); + break; + case 'A': + if (!par[0]) par[0]++; + gotoxy(currcons,x,y-par[0]); + break; + case 'B': case 'e': + if (!par[0]) par[0]++; + gotoxy(currcons,x,y+par[0]); + break; + case 'C': case 'a': + if (!par[0]) par[0]++; + gotoxy(currcons,x+par[0],y); + break; + case 'D': + if (!par[0]) par[0]++; + gotoxy(currcons,x-par[0],y); + break; + case 'E': + if (!par[0]) par[0]++; + gotoxy(currcons,0,y+par[0]); + break; + case 'F': + if (!par[0]) par[0]++; + gotoxy(currcons,0,y-par[0]); + break; + case 'd': + if (par[0]) par[0]--; + gotoxay(currcons,x,par[0]); + break; + case 'H': case 'f': + if (par[0]) par[0]--; + if (par[1]) par[1]--; + gotoxay(currcons,par[1],par[0]); + break; + case 'J': + csi_J(currcons,par[0]); + break; + case 'K': + csi_K(currcons,par[0]); + break; + case 'L': + csi_L(currcons,par[0]); + break; + case 'M': + csi_M(currcons,par[0]); + break; + case 'P': + csi_P(currcons,par[0]); + break; + case 'c': + if (!par[0]) + respond_ID(tty); + break; + case 'g': + if (!par[0]) + tab_stop[x >> 5] &= ~(1 << (x & 31)); + else if (par[0] == 3) { + tab_stop[0] = + tab_stop[1] = + tab_stop[2] = + tab_stop[3] = + tab_stop[4] = 0; + } + break; + case 'm': + csi_m(currcons); + return 1; + case 'q': /* DECLL - but only 3 leds */ + /* map 0,1,2,3 to 0,1,2,4 */ + if (par[0] < 4) + setledstate(kbd_table + currcons, + (par[0] < 3) ? par[0] : 4); + break; + case 'r': + if (!par[0]) + par[0]++; + if (!par[1]) + par[1] = video_num_lines; + /* Minimum allowed region is 2 lines */ + if (par[0] < par[1] && + par[1] <= video_num_lines) { + top=par[0]-1; + bottom=par[1]; + gotoxay(currcons,0,0); } - return; - case ESsetG0: - if (c == '0') - G0_charset = GRAF_MAP; - else if (c == 'B') - G0_charset = LAT1_MAP; - else if (c == 'U') - G0_charset = IBMPC_MAP; - else if (c == 'K') - G0_charset = USER_MAP; - if (charset == 0) - translate = set_translate(G0_charset,currcons); - vc_state = ESnormal; - return; - case ESsetG1: - if (c == '0') - G1_charset = GRAF_MAP; - else if (c == 'B') - G1_charset = LAT1_MAP; - else if (c == 'U') - G1_charset = IBMPC_MAP; - else if (c == 'K') - G1_charset = USER_MAP; - if (charset == 1) - translate = set_translate(G1_charset,currcons); - vc_state = ESnormal; - return; - default: - vc_state = ESnormal; + break; + case 's': + save_cur(currcons); + break; + case 'u': + restore_cur(currcons); + return 1; + case 'X': + csi_X(currcons, par[0]); + break; + case '@': + csi_at(currcons,par[0]); + break; + case ']': /* setterm functions */ + setterm_command(currcons); + break; + } + return 0; +} + +static int con_write_ctrl_ESpercent(int currcons, struct tty_struct *tty, unsigned int c) +{ + vc_state = con_write_ctrl_ESnormal; + switch (c) { + case '@': /* defined in ISO 2022 */ + utf = 0; + break; + case 'G': /* prelim official escape code */ + case '8': /* retained for compatibility */ + utf = 1; + break; + } + return 0; +} + +static int con_write_ctrl_ESfunckey(int currcons, struct tty_struct *tty, unsigned int c) +{ + vc_state = con_write_ctrl_ESnormal; + return 0; +} + +static int con_write_ctrl_EShash(int currcons, struct tty_struct *tty, unsigned int c) +{ + vc_state = con_write_ctrl_ESnormal; + if (c == '8') { + /* DEC screen alignment test. kludge :-) */ + video_erase_char = + (video_erase_char & 0xff00) | 'E'; + csi_J(currcons, 2); + video_erase_char = + (video_erase_char & 0xff00) | ' '; + do_update_region(currcons, origin, screenbuf_size/2); + } + return 0; +} + +static int con_write_ctrl_ESsetG0(int currcons, struct tty_struct *tty, unsigned int c) +{ + switch (c) { + case '0': + G0_charset = GRAF_MAP; + break; + case 'B': + G0_charset = LAT1_MAP; + break; + case 'U': + G0_charset = IBMPC_MAP; + break; + case 'K': + G0_charset = USER_MAP; + break; + } + if (charset == 0) { + translate = set_translate(G0_charset,currcons); + return 1; + } + vc_state = con_write_ctrl_ESnormal; + return 0; +} + +static int con_write_ctrl_ESsetG1(int currcons, struct tty_struct *tty, unsigned int c) +{ + switch (c) { + case '0': + G1_charset = GRAF_MAP; + break; + case 'B': + G1_charset = LAT1_MAP; + break; + case 'U': + G1_charset = IBMPC_MAP; + break; + case 'K': + G1_charset = USER_MAP; + break; + } + if (charset == 1) { + translate = set_translate(G1_charset,currcons); + return 1; } + vc_state = con_write_ctrl_ESnormal; + return 0; } /* This is a temporary buffer used to prepare a tty console write @@ -1843,7 +1960,7 @@ unsigned long draw_from = 0, draw_to = 0; struct vt_struct *vt = (struct vt_struct *)tty->driver_data; u16 himask, charmask; - const unsigned char *orig_buf = NULL; + const unsigned char *orig_buf; int orig_count; if (in_interrupt()) @@ -1901,42 +2018,12 @@ count--; if (utf) { - /* Combine UTF-8 into Unicode */ - /* Incomplete characters silently ignored */ - if(c > 0x7f) { - if (utf_count > 0 && (c & 0xc0) == 0x80) { - utf_char = (utf_char << 6) | (c & 0x3f); - utf_count--; - if (utf_count == 0) - tc = c = utf_char; - else continue; - } else { - if ((c & 0xe0) == 0xc0) { - utf_count = 1; - utf_char = (c & 0x1f); - } else if ((c & 0xf0) == 0xe0) { - utf_count = 2; - utf_char = (c & 0x0f); - } else if ((c & 0xf8) == 0xf0) { - utf_count = 3; - utf_char = (c & 0x07); - } else if ((c & 0xfc) == 0xf8) { - utf_count = 4; - utf_char = (c & 0x03); - } else if ((c & 0xfe) == 0xfc) { - utf_count = 5; - utf_char = (c & 0x01); - } else - utf_count = 0; + tc = con_write_utf(currcons, c); + if (tc < 0) continue; - } - } else { - tc = c; - utf_count = 0; - } - } else { /* no utf */ - tc = translate[toggle_meta ? (c|0x80) : c]; - } + c = tc; + } else /* no utf */ + tc = translate[toggle_meta ? (c|0x80) : c]; /* If the original code was a control character we * only allow a glyph to be displayed if the code is @@ -1954,7 +2041,7 @@ && (c != 127 || disp_ctrl) && (c != 128+27); - if (vc_state == ESnormal && ok) { + if (vc_state == con_write_ctrl_ESnormal && ok) { /* Now try to find out how to display it */ tc = conv_uni_to_pc(vc_cons[currcons].d, tc); if ( tc == -4 ) { @@ -2097,7 +2184,7 @@ if (kmsg_redirect && vc_cons_allocated(kmsg_redirect - 1)) currcons = kmsg_redirect - 1; - /* read `x' only after setting currecons properly (otherwise + /* read `x' only after setting currcons properly (otherwise the `x' macro will read the x of the foreground console). */ myx = x; @@ -2460,8 +2547,8 @@ console_driver.init_termios = tty_std_termios; console_driver.flags = TTY_DRIVER_REAL_RAW | TTY_DRIVER_RESET_TERMIOS; /* Tell tty_register_driver() to skip consoles because they are - * registered before kmalloc() is ready. We'll patch them in later. - * See comments at console_init(); see also con_init_devfs(). + * registered before kmalloc() is ready. We'll patch them in later. + * See comments at console_init(); see also con_init_devfs(). */ console_driver.flags |= TTY_DRIVER_NO_DEVFS; console_driver.refcount = &console_refcount; @@ -2704,7 +2791,7 @@ if (console_blank_hook && console_blank_hook(1)) return; - if (vesa_blank_mode) + if (vesa_blank_mode) sw->con_blank(vc_cons[currcons].d, vesa_blank_mode + 1); } @@ -2872,7 +2959,7 @@ if (!op->height) { /* Need to guess font height [compat] */ int h, i; u8 *charmap = op->data, tmp; - + /* If from KDFONTOP ioctl, don't allow things which can be done in userland, so that we can get rid of this soon */ if (!(op->flags & KD_FONT_FLAG_OLD)) @@ -2919,18 +3006,18 @@ op->data = old_op.data; if (!rc && !set) { int c = (op->width+7)/8 * 32 * op->charcount; - + if (op->data && op->charcount > old_op.charcount) rc = -ENOSPC; if (!(op->flags & KD_FONT_FLAG_OLD)) { - if (op->width > old_op.width || + if (op->width > old_op.width || op->height > old_op.height) rc = -ENOSPC; } else { if (op->width != 8) rc = -EIO; else if ((old_op.height && op->height > old_op.height) || - op->height > 32) + op->height > 32) rc = -ENOSPC; } if (!rc && op->data && copy_to_user(op->data, temp, c)) diff -urN orig/drivers/char/ds1307.c linux/drivers/char/ds1307.c --- orig/drivers/char/ds1307.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/char/ds1307.c Mon Aug 5 23:06:51 2002 @@ -0,0 +1,604 @@ +/* + * ds1307.c + * + * Device driver for Dallas Semiconductor's Real Time Controller DS1307. + * + * Copyright (C) 2002 Intrinsyc Software Inc. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + * + */ + +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "ds1307.h" + +#define DEBUG 0 + +#if DEBUG +static unsigned int rtc_debug = DEBUG; +#else +#define rtc_debug 0 /* gcc will remove all the debug code for us */ +#endif + +static unsigned short slave_address = DS1307_I2C_SLAVE_ADDR; + +struct i2c_driver ds1307_driver; +struct i2c_client *ds1307_i2c_client = 0; + +static unsigned short ignore[] = { I2C_CLIENT_END }; +static unsigned short normal_addr[] = { DS1307_I2C_SLAVE_ADDR, I2C_CLIENT_END }; + +static struct i2c_client_address_data addr_data = { + normal_i2c: normal_addr, + normal_i2c_range: ignore, + probe: ignore, + probe_range: ignore, + ignore: ignore, + ignore_range: ignore, + force: ignore, +}; + +static int ds1307_rtc_ioctl( struct inode *, struct file *, unsigned int, unsigned long); +static int ds1307_rtc_open(struct inode *inode, struct file *file); +static int ds1307_rtc_release(struct inode *inode, struct file *file); + +static struct file_operations rtc_fops = { + owner: THIS_MODULE, + ioctl: ds1307_rtc_ioctl, + open: ds1307_rtc_open, + release: ds1307_rtc_release, +}; + +static struct miscdevice ds1307_rtc_miscdev = { + RTC_MINOR, + "rtc", + &rtc_fops +}; + +static int ds1307_probe(struct i2c_adapter *adap); +static int ds1307_detach(struct i2c_client *client); +static int ds1307_command(struct i2c_client *client, unsigned int cmd, void *arg); + +struct i2c_driver ds1307_driver = { + name: "DS1307", + id: I2C_DRIVERID_DS1307, + flags: I2C_DF_NOTIFY, + attach_adapter: ds1307_probe, + detach_client: ds1307_detach, + command: ds1307_command +}; + +static spinlock_t ds1307_rtc_lock = SPIN_LOCK_UNLOCKED; + +#define DAT(x) ((unsigned int)((x)->data)) /* keep the control register info */ + +static int +ds1307_readram( char *buf, int len) +{ + unsigned long flags; + unsigned char ad[1] = { 0 }; + int ret; + struct i2c_msg msgs[2] = { + { ds1307_i2c_client->addr , 0, 1, ad }, + { ds1307_i2c_client->addr , I2C_M_RD, len, buf } }; + + spin_lock_irqsave(&ds1307_rtc_lock, flags); + ret = i2c_transfer(ds1307_i2c_client->adapter, msgs, 2); + spin_unlock_irqrestore(&ds1307_rtc_lock,flags); + + return ret; +} + +static void +ds1307_dumpram( void) +{ + unsigned char buf[DS1307_RAM_SIZE]; + int ret; + + ret = ds1307_readram( buf, DS1307_RAM_SIZE); + + if( ret > 0) + { + int i; + for( i=0; iaddr , 0, 1, ad }, + { ds1307_i2c_client->addr , I2C_M_RD, 1, buf } + }; + unsigned char ctrl_info; + int ret; + + if( enable) + ctrl_info = SQW_ENABLE | RATE_32768HZ; + else + ctrl_info = SQW_DISABLE; + ds1307_command(ds1307_i2c_client, DS1307_SETCTRL, &ctrl_info); + + /* read addr 0 (Clock-Halt bit and second counter */ + ret = i2c_transfer(ds1307_i2c_client->adapter, msgs, 2); + + if( enable) + buf[1] = buf[0] & ~CLOCK_HALT; /* clear Clock-Halt bit */ + else + buf[1] = buf[0] | CLOCK_HALT; /* set Clock-Halt bit */ + buf[0] = 0; /* control register address on DS1307 */ + + ret = i2c_master_send(ds1307_i2c_client, (char *)buf, 2); +} + +static int +ds1307_attach(struct i2c_adapter *adap, int addr, unsigned short flags,int kind) +{ + struct i2c_client *c; + unsigned char buf[1], ad[1] = { 7 }; + struct i2c_msg msgs[2] = { + { addr , 0, 1, ad }, + { addr , I2C_M_RD, 1, buf } + }; + int ret; + + c = (struct i2c_client *)kmalloc(sizeof(*c), GFP_KERNEL); + if (!c) + return -ENOMEM; + + strcpy(c->name, "DS1307"); + c->id = ds1307_driver.id; + c->flags = 0; + c->addr = addr; + c->adapter = adap; + c->driver = &ds1307_driver; + c->data = NULL; + + ret = i2c_transfer(c->adapter, msgs, 2); + + if ( ret == 2 ) + { + DAT(c) = buf[0]; + } + else + printk ("ds1307_attach(): i2c_transfer() returned %d.\n",ret); + + ds1307_i2c_client = c; + ds1307_enable_clock( 1); + + return i2c_attach_client(c); +} + +static int +ds1307_probe(struct i2c_adapter *adap) +{ + return i2c_probe(adap, &addr_data, ds1307_attach); +} + +static int +ds1307_detach(struct i2c_client *client) +{ + i2c_detach_client(client); + ds1307_enable_clock( 0); + + return 0; +} + +static void +ds1307_convert_to_time( struct rtc_time *dt, char *buf) +{ + dt->tm_sec = BCD_TO_BIN(buf[0]); + dt->tm_min = BCD_TO_BIN(buf[1]); + + if ( TWELVE_HOUR_MODE(buf[2]) ) + { + dt->tm_hour = HOURS_12(buf[2]); + if (HOURS_AP(buf[2])) /* PM */ + { + dt->tm_hour += 12; + } + } + else /* 24-hour-mode */ + { + dt->tm_hour = HOURS_24(buf[2]); + } + + dt->tm_mday = BCD_TO_BIN(buf[4]); + /* dt->tm_mon is zero-based */ + dt->tm_mon = BCD_TO_BIN(buf[5]) - 1; + /* year is 1900 + dt->tm_year */ + dt->tm_year = BCD_TO_BIN(buf[6]) + 100; + + if( rtc_debug > 2) + { + printk("ds1307_get_datetime: year = %d\n", dt->tm_year); + printk("ds1307_get_datetime: mon = %d\n", dt->tm_mon); + printk("ds1307_get_datetime: mday = %d\n", dt->tm_mday); + printk("ds1307_get_datetime: hour = %d\n", dt->tm_hour); + printk("ds1307_get_datetime: min = %d\n", dt->tm_min); + printk("ds1307_get_datetime: sec = %d\n", dt->tm_sec); + } +} + +static int +ds1307_get_datetime(struct i2c_client *client, struct rtc_time *dt) +{ + unsigned char buf[7], addr[1] = { 0 }; + struct i2c_msg msgs[2] = { + { client->addr, 0, 1, addr }, + { client->addr, I2C_M_RD, 7, buf } + }; + int ret = -EIO; + + memset(buf, 0, sizeof(buf)); + + ret = i2c_transfer(client->adapter, msgs, 2); + + if (ret == 2) { + ds1307_convert_to_time( dt, buf); + ret = 0; + } + else + printk("ds1307_get_datetime(), i2c_transfer() returned %d\n",ret); + + return ret; +} + +static int +ds1307_set_datetime(struct i2c_client *client, struct rtc_time *dt, int datetoo) +{ + unsigned char buf[8]; + int ret, len = 4; + + if( rtc_debug > 2) + { + printk("ds1307_set_datetime: tm_year = %d\n", dt->tm_year); + printk("ds1307_set_datetime: tm_mon = %d\n", dt->tm_mon); + printk("ds1307_set_datetime: tm_mday = %d\n", dt->tm_mday); + printk("ds1307_set_datetime: tm_hour = %d\n", dt->tm_hour); + printk("ds1307_set_datetime: tm_min = %d\n", dt->tm_min); + printk("ds1307_set_datetime: tm_sec = %d\n", dt->tm_sec); + } + + buf[0] = 0; /* register address on DS1307 */ + buf[1] = (BIN_TO_BCD(dt->tm_sec)); + buf[2] = (BIN_TO_BCD(dt->tm_min)); + buf[3] = (BIN_TO_BCD(dt->tm_hour)); + + if (datetoo) { + len = 8; + /* we skip buf[4] as we don't use day-of-week. */ + buf[5] = (BIN_TO_BCD(dt->tm_mday)); + buf[6] = (BIN_TO_BCD(dt->tm_mon + 1)); + /* The year only ranges from 0-99, we are being passed an offset from 1900, + * and the chip calulates leap years based on 2000, thus we adjust by 100. + */ + buf[7] = (BIN_TO_BCD(dt->tm_year - 100)); + } + ret = i2c_master_send(client, (char *)buf, len); + if (ret == len) + ret = 0; + else + printk("ds1307_set_datetime(), i2c_master_send() returned %d\n",ret); + + + return ret; +} + +static int +ds1307_get_ctrl(struct i2c_client *client, unsigned char *ctrl) +{ + *ctrl = DAT(client); + + return 0; +} + +static int +ds1307_set_ctrl(struct i2c_client *client, unsigned char *cinfo) +{ + unsigned char buf[2]; + int ret; + + + buf[0] = 7; /* control register address on DS1307 */ + buf[1] = *cinfo; + /* save the control reg info in the client data field so that get_ctrl + * function doesn't have to do an I2C transfer to get it. + */ + DAT(client) = buf[1]; + + ret = i2c_master_send(client, (char *)buf, 2); + + return ret; +} + +static int +ds1307_read_mem(struct i2c_client *client, struct rtc_mem *mem) +{ + unsigned char addr[1]; + struct i2c_msg msgs[2] = { + { client->addr, 0, 1, addr }, + { client->addr, I2C_M_RD, mem->nr, mem->data } + }; + + if ( (mem->loc < DS1307_RAM_ADDR_START) || + ((mem->loc + mem->nr -1) > DS1307_RAM_ADDR_END) ) + return -EINVAL; + + addr[0] = mem->loc; + + return i2c_transfer(client->adapter, msgs, 2) == 2 ? 0 : -EIO; +} + +static int +ds1307_write_mem(struct i2c_client *client, struct rtc_mem *mem) +{ + unsigned char addr[1]; + struct i2c_msg msgs[2] = { + { client->addr, 0, 1, addr }, + { client->addr, 0, mem->nr, mem->data } + }; + + if ( (mem->loc < DS1307_RAM_ADDR_START) || + ((mem->loc + mem->nr -1) > DS1307_RAM_ADDR_END) ) + return -EINVAL; + + addr[0] = mem->loc; + + return i2c_transfer(client->adapter, msgs, 2) == 2 ? 0 : -EIO; +} + +static int +ds1307_command(struct i2c_client *client, unsigned int cmd, void *arg) +{ + switch (cmd) { + case DS1307_GETDATETIME: + return ds1307_get_datetime(client, arg); + + case DS1307_SETTIME: + return ds1307_set_datetime(client, arg, 0); + + case DS1307_SETDATETIME: + return ds1307_set_datetime(client, arg, 1); + + case DS1307_GETCTRL: + return ds1307_get_ctrl(client, arg); + + case DS1307_SETCTRL: + return ds1307_set_ctrl(client, arg); + + case DS1307_MEM_READ: + return ds1307_read_mem(client, arg); + + case DS1307_MEM_WRITE: + return ds1307_write_mem(client, arg); + + default: + return -EINVAL; + } +} + +static int +ds1307_rtc_open(struct inode *inode, struct file *file) +{ + return 0; +} + +static int +ds1307_rtc_release(struct inode *inode, struct file *file) +{ + return 0; +} + +static int +ds1307_rtc_ioctl( struct inode *inode, struct file *file, + unsigned int cmd, unsigned long arg) +{ + unsigned long flags; + struct rtc_time wtime; + int status = 0; + + switch (cmd) { + default: + case RTC_UIE_ON: + case RTC_UIE_OFF: + case RTC_PIE_ON: + case RTC_PIE_OFF: + case RTC_AIE_ON: + case RTC_AIE_OFF: + case RTC_ALM_SET: + case RTC_ALM_READ: + case RTC_IRQP_READ: + case RTC_IRQP_SET: + case RTC_EPOCH_READ: + case RTC_EPOCH_SET: + case RTC_WKALM_SET: + case RTC_WKALM_RD: + status = -EINVAL; + break; + + case RTC_RD_TIME: + spin_lock_irqsave(&ds1307_rtc_lock, flags); + ds1307_command( ds1307_i2c_client, DS1307_GETDATETIME, &wtime); + spin_unlock_irqrestore(&ds1307_rtc_lock,flags); + + if( copy_to_user((void *)arg, &wtime, sizeof (struct rtc_time))) + status = -EFAULT; + break; + + case RTC_SET_TIME: + if (!capable(CAP_SYS_TIME)) + { + status = -EACCES; + break; + } + + if (copy_from_user(&wtime, (struct rtc_time *)arg, sizeof(struct rtc_time)) ) + { + status = -EFAULT; + break; + } + + spin_lock_irqsave(&ds1307_rtc_lock, flags); + ds1307_command( ds1307_i2c_client, DS1307_SETDATETIME, &wtime); + spin_unlock_irqrestore(&ds1307_rtc_lock,flags); + break; + } + + return status; +} + +static char * +ds1307_mon2str( unsigned int mon) +{ + char *mon2str[12] = { + "Jan", "Feb", "Mar", "Apr", "May", "Jun", + "Jul", "Aug", "Sep", "Oct", "Nov", "Dec" + }; + if( mon > 11) return "error"; + else return mon2str[ mon]; +} + +static int ds1307_rtc_proc_output( char *buf) +{ +#define CHECK(ctrl,bit) ((ctrl & bit) ? "yes" : "no") + unsigned char ram[DS1307_RAM_SIZE]; + int ret; + + char *p = buf; + + ret = ds1307_readram( ram, DS1307_RAM_SIZE); + if( ret > 0) + { + int i; + struct rtc_time dt; + char text[9]; + + p += sprintf(p, "DS1307 (64x8 Serial Real Time Clock)\n"); + + ds1307_convert_to_time( &dt, ram); + p += sprintf(p, "Date/Time : %02d-%s-%04d %02d:%02d:%02d\n", + dt.tm_mday, ds1307_mon2str(dt.tm_mon), dt.tm_year + 1900, + dt.tm_hour, dt.tm_min, dt.tm_sec); + + p += sprintf(p, "Clock halted : %s\n", CHECK(ram[0],0x80)); + p += sprintf(p, "24h mode : %s\n", CHECK(ram[2],0x40)); + p += sprintf(p, "Square wave enabled : %s\n", CHECK(ram[7],0x10)); + p += sprintf(p, "Freq : "); + + switch( ram[7] & 0x03) + { + case RATE_1HZ: + p += sprintf(p, "1Hz\n"); + break; + case RATE_4096HZ: + p += sprintf(p, "4.096kHz\n"); + break; + case RATE_8192HZ: + p += sprintf(p, "8.192kHz\n"); + break; + case RATE_32768HZ: + default: + p += sprintf(p, "32.768kHz\n"); + break; + + } + + p += sprintf(p, "RAM dump:\n"); + text[8]='\0'; + for( i=0; i126)) ram[i]='.'; + text[i%8] = ram[i]; + if( (i%8) == 7) p += sprintf(p, "%s\n",text); + } + p += sprintf(p, "\n"); + } + else + { + p += sprintf(p, "Failed to read RTC memory!\n"); + } + + return p - buf; +} + +static int ds1307_rtc_read_proc(char *page, char **start, off_t off, + int count, int *eof, void *data) +{ + int len = ds1307_rtc_proc_output (page); + if (len <= off+count) *eof = 1; + *start = page + off; + len -= off; + if (len>count) len = count; + if (len<0) len = 0; + return len; +} + +static __init int ds1307_init(void) +{ + int retval=0; + + if( slave_address != 0xffff) + { + normal_addr[0] = slave_address; + } + + if( normal_addr[0] == 0xffff) + { + printk(KERN_ERR"I2C: Invalid slave address for DS1307 RTC (%#x)\n", + normal_addr[0]); + return -EINVAL; + } + + retval = i2c_add_driver(&ds1307_driver); + + if (retval==0) + { + misc_register (&ds1307_rtc_miscdev); + create_proc_read_entry (PROC_DS1307_NAME, 0, 0, ds1307_rtc_read_proc, NULL); + printk("I2C: DS1307 RTC driver successfully loaded\n"); + + if( rtc_debug) ds1307_dumpram(); + } + return retval; +} + +static __exit void ds1307_exit(void) +{ + remove_proc_entry (PROC_DS1307_NAME, NULL); + misc_deregister(&ds1307_rtc_miscdev); + i2c_del_driver(&ds1307_driver); +} + +module_init(ds1307_init); +module_exit(ds1307_exit); + +MODULE_PARM (slave_address, "i"); +MODULE_PARM_DESC (slave_address, "I2C slave address for DS1307 RTC."); + +MODULE_AUTHOR ("Intrinsyc Software Inc."); +MODULE_LICENSE("GPL"); diff -urN orig/drivers/char/ds1307.h linux/drivers/char/ds1307.h --- orig/drivers/char/ds1307.h Thu Jan 1 01:00:00 1970 +++ linux/drivers/char/ds1307.h Mon Aug 5 22:25:06 2002 @@ -0,0 +1,58 @@ +/* + * ds1307.h + * + * Copyright (C) 2002 Intrinsyc Software Inc. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + * + */ +#ifndef DS1307_H +#define DS1307_H + +#if defined(CONFIG_PXA_EMERSON_SBC) || defined(CONFIG_PXA_CERF_BOARD) + #define DS1307_I2C_SLAVE_ADDR 0x68 +#else + #define DS1307_I2C_SLAVE_ADDR 0xffff +#endif + +#define DS1307_RAM_ADDR_START 0x08 +#define DS1307_RAM_ADDR_END 0x3F +#define DS1307_RAM_SIZE 0x40 + +#define PROC_DS1307_NAME "driver/ds1307" + +struct rtc_mem { + unsigned int loc; + unsigned int nr; + unsigned char *data; +}; + +#define DS1307_GETDATETIME 0 +#define DS1307_SETTIME 1 +#define DS1307_SETDATETIME 2 +#define DS1307_GETCTRL 3 +#define DS1307_SETCTRL 4 +#define DS1307_MEM_READ 5 +#define DS1307_MEM_WRITE 6 + +#define SQW_ENABLE 0x10 /* Square Wave Enable */ +#define SQW_DISABLE 0x00 /* Square Wave disable */ + +#define RATE_32768HZ 0x03 /* Rate Select 32.768KHz */ +#define RATE_8192HZ 0x02 /* Rate Select 8.192KHz */ +#define RATE_4096HZ 0x01 /* Rate Select 4.096KHz */ +#define RATE_1HZ 0x00 /* Rate Select 1Hz */ + +#define CLOCK_HALT 0x80 /* Clock Halt */ + +#define BCD_TO_BIN(val) (((val)&15) + ((val)>>4)*10) +#define BIN_TO_BCD(val) ((((val)/10)<<4) + (val)%10) + +#define TWELVE_HOUR_MODE(n) (((n)>>6)&1) +#define HOURS_AP(n) (((n)>>5)&1) +#define HOURS_12(n) BCD_TO_BIN((n)&0x1F) +#define HOURS_24(n) BCD_TO_BIN((n)&0x3F) + +#endif diff -urN orig/drivers/char/edb7211_keyb.c linux/drivers/char/edb7211_keyb.c --- orig/drivers/char/edb7211_keyb.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/char/edb7211_keyb.c Mon Sep 10 14:36:18 2001 @@ -0,0 +1,335 @@ +/* + * drivers/char/edb7211_keyb.c + * + * Copyright (C) 2000 Blue Mug, Inc. All Rights Reserved. + * + * EDB7211 Keyboard driver for ARM Linux. + * + * The EP7211 keyboard hardware only supports generating interrupts for 64 keys. + * The EBD7211's keyboard has 84 keys. Therefore we need to poll for keys, + * instead of waiting for interrupts. + * + * In a real-world hardware situation, this would be a bad thing. It would + * kill power management. + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include + +#include +#include + + +/* + * The number of jiffies between keyboard scans. + */ +#define KEYBOARD_SCAN_INTERVAL 5 + +/* + * Values for the keyboard column scan control register. + */ +#define KBSC_HI 0x0 /* All driven high */ +#define KBSC_LO 0x1 /* All driven low */ +#define KBSC_X 0x2 /* All high impedance */ +#define KBSC_COL0 0x8 /* Column 0 high, others high impedance */ +#define KBSC_COL1 0x9 /* Column 1 high, others high impedance */ +#define KBSC_COL2 0xa /* Column 2 high, others high impedance */ +#define KBSC_COL3 0xb /* Column 3 high, others high impedance */ +#define KBSC_COL4 0xc /* Column 4 high, others high impedance */ +#define KBSC_COL5 0xd /* Column 5 high, others high impedance */ +#define KBSC_COL6 0xe /* Column 6 high, others high impedance */ +#define KBSC_COL7 0xf /* Column 7 high, others high impedance */ + + +/* XXX: Figure out what these values should be... */ +/* Simple translation table for the SysRq keys */ +#ifdef CONFIG_MAGIC_SYSRQ +unsigned char edb7211_kbd_sysrq_xlate[128] = + "\000\0331234567890-=\177\t" /* 0x00 - 0x0f */ + "qwertyuiop[]\r\000as" /* 0x10 - 0x1f */ + "dfghjkl;'`\000\\zxcv" /* 0x20 - 0x2f */ + "bnm,./\000*\000 \000\201\202\203\204\205" /* 0x30 - 0x3f */ + "\206\207\210\211\212\000\000789-456+1" /* 0x40 - 0x4f */ + "230\177\000\000\213\214\000\000\000\000\000\000\000\000\000\000" /* 0x50 - 0x5f */ + "\r\000/"; /* 0x60 - 0x6f */ +#endif + +/* + * Row/column to scancode mappings. + * + * This table maps row/column keyboard matrix positions to XT scancodes. + * + * The port A rows come first, followed by the extended rows. + */ +static unsigned char colrow_2_scancode[128] = +{ +/* Column: + Row 0 1 2 3 4 5 6 7 */ +/* A0 */ 0x01, 0x3f, 0x3e, 0x3d, 0x3c, 0x3b, 0x40, 0x41, +/* A1 */ 0x02, 0x07, 0x06, 0x05, 0x04, 0x03, 0x08, 0x09, +/* A2 */ 0x0f, 0x14, 0x13, 0x12, 0x11, 0x10, 0x15, 0x16, +/* A3 */ 0x3a, 0x22, 0x21, 0x20, 0x1f, 0x1e, 0x23, 0x24, +/* A4 */ 0x29, 0x30, 0x2f, 0x2e, 0x2d, 0x2c, 0x31, 0x32, +/* A5 */ 0x39, 0x35, 0x6F, 0x52, 0x00, 0x6B, 0x34, 0x33, +/* A6 */ 0x6A, 0x27, 0x28, 0x00, 0x1c, 0x6D, 0x26, 0x25, +/* A7 */ 0x67, 0x19, 0x1a, 0x1b, 0x2b, 0x68, 0x18, 0x17, +/* E0 */ 0x6C, 0x0c, 0x0d, 0x0e, 0x00, 0x66, 0x0b, 0x0a, +/* E1 */ 0x69, 0x44, 0x45, 0x37, 0x46, 0x77, 0x43, 0x42, +/* E2 */ 0x2a, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, +/* E3 */ 0x1d, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, +/* E4 */ 0x55, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, +/* E5 */ 0x38, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, +/* E6 */ 0x64, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, +/* E7 */ 0x36, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 +}; + +/* + * A bitfield array which contains the state of the keyboard after the last + * scan. A bit set in this array corresponds to a key down. Only the lower + * 16 bits of each array element are used. + */ +static unsigned long previous_keys[8]; +static unsigned long keys[8]; + + +/* This will be set to a non-zero value if a key was found to be pressed + * in the last scan. */ +static int key_is_pressed; + +static struct tq_struct kbd_process_task; +static struct timer_list edb7211_kbd_timer; + +/* + * External methods. + */ +void edb7211_kbd_init_hw(void); + +/* + * Internal methods. + */ +static int edb7211_kbd_scan_matrix(u_long* keys); +static void edb7211_kbd_timeout(unsigned long data); +static void edb7211_kbd_process(void* data); + +/* + * Translate a raw keycode to an XT keyboard scancode. + */ +static int +edb7211_translate(unsigned char scancode, unsigned char *keycode, + char raw_mode) +{ + *keycode = colrow_2_scancode[scancode & 0x7f]; + return 1; +} + +/* + * Scan the keyboard matrix; for each key that is pressed, set the + * corresponding bit in the bitfield array. + * + * The parameter is expected to be an array of 8 32-bit values. Only the lower + * 16 bits of each value is used. Each value contains the row bits for the + * corresponding column. + */ +static int +edb7211_kbd_scan_matrix(u_long* keys) +{ + int column, row, key_pressed; + unsigned char port_a_data, ext_port_data; + + key_pressed = 0; + + /* Drive all the columns low. */ + clps_writel((clps_readl(SYSCON1) & ~SYSCON1_KBDSCANMASK) | KBSC_LO, + SYSCON1); + + for (column = 0; column < 8; column++) { + + /* Drive the column high. */ + clps_writel((clps_readl(SYSCON1) & ~SYSCON1_KBDSCANMASK) | + (KBSC_COL0 + column), SYSCON1); + + /* Read port A and the extended port. */ + port_a_data = clps_readb(PADR) & 0xff; + ext_port_data = __raw_readb(EP7211_VIRT_EXTKBD) & 0xff; + + /* Drive all columns tri-state. */ + clps_writel((clps_readl(SYSCON1) & ~SYSCON1_KBDSCANMASK) | KBSC_X, + SYSCON1); + + /* Look at each column in port A. */ + for (row=0; row < 8; row++) { + /* If the row's bit is set, set the bit in the bitfield. + * Otherwise, clear it. + */ + if (port_a_data & (1 << row)) { + keys[column] |= (1 << row); + key_pressed = 1; + } else { + keys[column] &= ~(1 << row); + } + } + + /* Look at each column in the extended port. */ + for (row=0; row < 8; row++) { + /* If the row's bit is set, set the bit in the bitfield. + * Otherwise, clear it. + */ + if (ext_port_data & (1 << row)) { + keys[column] |= (1 << (row + 8)); + key_pressed = 1; + } else { + keys[column] &= ~(1 << (row + 8)); + } + } + + /* + * Short delay: The example code for the EDB7211 runs an empty + * loop 256 times. At this rate, there were some spurious keys + * generated. I doubled the delay to let the column drives + * settle some. + */ + for (row=0; row < 512; row++) { } + } + + /* If we could use interrupts, we would drive all columns high so + * that interrupts will be generated on key presses. But we can't, + * so we leave all columns floating. + */ + clps_writel((clps_readl(SYSCON1) & ~SYSCON1_KBDSCANMASK) | KBSC_X, + SYSCON1); + + return key_pressed; +} + +/* + * XXX: This is really ugly; this needs to be reworked to have less levels of + * indentation. + */ +static void +edb7211_kbd_timeout(unsigned long data) +{ + /* Schedule the next timer event. */ + edb7211_kbd_timer.expires = jiffies + KEYBOARD_SCAN_INTERVAL; + add_timer(&edb7211_kbd_timer); + + if (edb7211_kbd_scan_matrix(keys) || key_is_pressed) { + queue_task(&kbd_process_task, &tq_timer); + } else { + key_is_pressed = 0; + } +} + +/* + * Process the keys that have been pressed. + */ +static void +edb7211_kbd_process(void* data) +{ + int i; + + /* First check if any keys have been released. */ + if (key_is_pressed) { + for (i=0; i < 8; i++) { + if (previous_keys[i]) { + int row; + + for (row=0; row < 16; row++) { + if ((previous_keys[i] & (1 << row)) && + !(keys[i] & (1 << row))) { + /* Generate the up event. */ + handle_scancode( + (row<<3)+i, 0); + } + } + } + } + } + + key_is_pressed = 0; + + /* Now scan the keys and send press events. */ + for (i=0; i < 8; i++) { + if (keys[i]) { + int row; + + for (row=0; row < 16; row++) { + if (keys[i] & (1 << row)) { + if (previous_keys[i] & (1 << row)) { + /* Generate the hold event. */ + handle_scancode((row<<3)+i, 1); + } else { + /* Generate the down event. */ + handle_scancode((row<<3)+i, 1); + } + + key_is_pressed = 1; + } + } + } + } + + /* Update the state variables. */ + memcpy(previous_keys, keys, 8 * sizeof(unsigned long)); +} + +static char edb7211_unexpected_up(unsigned char scancode) +{ + return 0200; +} + +static void edb7211_leds(unsigned char leds) +{ +} + +/* + * Initialize the keyboard hardware. Set the column drives low and + * start the timer. + */ +void __init +edb7211_kbd_init_hw(void) +{ + k_translate = edb7211_translate; + k_unexpected_up = edb7211_unexpected_up; + k_leds = edb7211_leds; + + /* + * If we had the ability to use interrupts, we would want to drive all + * columns high. But we have more keys than can generate interrupts, so + * we leave them floating. + */ + clps_writel((clps_readl(SYSCON1) & ~SYSCON1_KBDSCANMASK) | KBSC_X, + SYSCON1); + + /* Initialize the matrix processing task. */ + kbd_process_task.routine = edb7211_kbd_process; + kbd_process_task.data = NULL; + + /* Setup the timer to poll the keyboard. */ + init_timer(&edb7211_kbd_timer); + edb7211_kbd_timer.function = edb7211_kbd_timeout; + edb7211_kbd_timer.data = (unsigned long)NULL; + edb7211_kbd_timer.expires = jiffies + KEYBOARD_SCAN_INTERVAL; + add_timer(&edb7211_kbd_timer); +} + + diff -urN orig/drivers/char/gc_kbmap.h linux/drivers/char/gc_kbmap.h --- orig/drivers/char/gc_kbmap.h Thu Jan 1 01:00:00 1970 +++ linux/drivers/char/gc_kbmap.h Tue Aug 28 20:26:52 2001 @@ -0,0 +1,162 @@ + + +#define KK_NONE 0x7f +#define KK_ESC 0x00 +#define KK_F1 0x01 +#define KK_F2 0x02 +#define KK_F3 0x03 +#define KK_F4 0x04 +#define KK_F5 0x05 +#define KK_F6 0x06 +#define KK_F7 0x07 +#define KK_F8 0x08 +#define KK_F9 0x09 +#define KK_F10 0x0a +#define KK_F11 0x0b +#define KK_F12 0x0c +#define KK_PRNT 0x0d +#define KK_SCRL 0x0e +#define KK_BRK 0x0f +#define KK_AGR 0x10 +#define KK_1 0x11 +#define KK_2 0x12 +#define KK_3 0x13 +#define KK_4 0x14 +#define KK_5 0x15 +#define KK_6 0x16 +#define KK_7 0x17 +#define KK_8 0x18 +#define KK_9 0x19 +#define KK_0 0x1a +#define KK_MINS 0x1b +#define KK_EQLS 0x1c +#define KK_BKSP 0x1e +#define KK_INS 0x1f +#define KK_HOME 0x20 +#define KK_PGUP 0x21 +#define KK_NUML 0x22 +#define KP_SLH 0x23 +#define KP_STR 0x24 +#define KP_MNS 0x3a +#define KK_TAB 0x26 +#define KK_Q 0x27 +#define KK_W 0x28 +#define KK_E 0x29 +#define KK_R 0x2a +#define KK_T 0x2b +#define KK_Y 0x2c +#define KK_U 0x2d +#define KK_I 0x2e +#define KK_O 0x2f +#define KK_P 0x30 +#define KK_LSBK 0x31 +#define KK_RSBK 0x32 +#define KK_ENTR 0x47 +#define KK_DEL 0x34 +#define KK_END 0x35 +#define KK_PGDN 0x36 +#define KP_7 0x37 +#define KP_8 0x38 +#define KP_9 0x39 +#define KP_PLS 0x4b +#define KK_CAPS 0x5d +#define KK_A 0x3c +#define KK_S 0x3d +#define KK_D 0x3e +#define KK_F 0x3f +#define KK_G 0x40 +#define KK_H 0x41 +#define KK_J 0x42 +#define KK_K 0x43 +#define KK_L 0x44 +#define KK_SEMI 0x45 +#define KK_SQOT 0x46 +#define KK_HASH 0x1d +#define KP_4 0x48 +#define KP_5 0x49 +#define KP_6 0x4a +#define KK_LSFT 0x4c +#define KK_BSLH 0x33 +#define KK_Z 0x4e +#define KK_X 0x4f +#define KK_C 0x50 +#define KK_V 0x51 +#define KK_B 0x52 +#define KK_N 0x53 +#define KK_M 0x54 +#define KK_COMA 0x55 +#define KK_DOT 0x56 +#define KK_FSLH 0x57 +#define KK_RSFT 0x58 +#define KK_UP 0x59 +#define KP_1 0x5a +#define KP_2 0x5b +#define KP_3 0x5c +#define KP_ENT 0x67 +#define KK_LCTL 0x3b +#define KK_LALT 0x5e +#define KK_SPCE 0x5f +#define KK_RALT 0x60 +#define KK_RCTL 0x61 +#define KK_LEFT 0x62 +#define KK_DOWN 0x63 +#define KK_RGHT 0x64 +#define KP_0 0x65 +#define KP_DOT 0x66 + +static char kbmap[128] = { +KK_NONE, KK_LALT, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, +KK_NONE, KK_AGR, KK_BSLH, KK_TAB, KK_Z, KK_A, KK_X, KK_NONE, +KK_NONE, KK_NONE, KK_LSFT, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, +KK_NONE, KK_LCTL, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, +KK_NONE, 0x21, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, +KK_NONE, KK_ESC, KK_DEL, KK_Q, KK_CAPS, KK_S, KK_C, KK_3, +KK_NONE, KK_1, KK_NONE, KK_W, KK_NONE, KK_D, KK_V, KK_4, +KK_NONE, KK_2, KK_T, KK_E, KK_NONE, KK_F, KK_B, KK_5, +KK_NONE, KK_9, KK_Y, KK_R, KK_K, KK_G, KK_N, KK_6, +KK_NONE, KK_0, KK_U, KK_O, KK_L, KK_H, KK_M, KK_7, +KK_NONE, KK_MINS, KK_I, KK_P, KK_SEMI, KK_J, KK_COMA, KK_8, +KK_NONE, KK_EQLS, KK_ENTR, KK_LSBK, KK_BSLH, KK_FSLH, KK_DOT, KK_NONE, +KK_NONE, KK_NONE, KK_RSFT, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, +KK_NONE, KK_BKSP, KK_DOWN, KK_RSBK, KK_UP, KK_LEFT, KK_SPCE, KK_RGHT, +KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, +KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE}; + +static char kbmapFN[128] = { +KK_NONE, KK_LALT, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, +KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, +KK_NONE, KK_NONE, KK_LSFT, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, +KK_NONE, KK_LCTL, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, +KK_NONE, 0x21, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, +KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_F3, +KK_NONE, KK_F1, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_F4, +KK_NONE, KK_F2, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_F5, +KK_NONE, KK_F9, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_F6, +KK_NONE, KK_F10, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_F7, +KK_NONE, KK_NUML, KK_NONE, KK_INS, KK_PRNT, KK_NONE, KK_NONE, KK_F8, +KK_NONE, KK_BRK, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, +KK_NONE, KK_NONE, KK_RSFT, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, +KK_NONE, KK_NONE, KK_PGDN, KK_SCRL, KK_PGUP, KK_HOME, KK_NONE, KK_END, +KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, +KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE}; + +static char kbmapNL[128] = { +KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, +KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, +KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, +KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, +KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, +KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, +KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, +KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, +KK_NONE, KP_9, KK_NONE, KK_NONE, KP_2, KK_NONE, KK_NONE, KK_NONE, +KK_NONE, KP_STR, KP_4, KP_6, KP_3, KK_NONE, KP_0, KP_7, +KK_NONE, KK_NONE, KP_5, KP_MNS, KP_PLS, KP_1, KK_NONE, KP_8, +KK_NONE, KK_NONE, KP_ENT, KK_NONE, KK_NONE, KP_SLH, KP_DOT, KK_NONE, +KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, +KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, +KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, +KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE, KK_NONE}; + + + diff -urN orig/drivers/char/gc_keyb.c linux/drivers/char/gc_keyb.c --- orig/drivers/char/gc_keyb.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/char/gc_keyb.c Mon Sep 24 18:13:47 2001 @@ -0,0 +1,1145 @@ +/* + * linux/arch/arm/drivers/char/gc_keyb.c + * + * Copyright 2000 Applied Data Systems + * + * Keyboard & Smartio driver for GraphicsClient ARM Linux. + * Graphics Client is SA1110 based single board computer by + * Applied Data Systems (http://www.applieddata.net) + * + * Change log: + * 7-10/6/01 Thomas Thaele + * - Added Keyboard Sniffer on /dev/sio12 + * - First implementation of PC- compatible Scancodes (thanks to pc_keyb.c) + * 3/23/01 Woojung Huh + * Power Management added + * 12/01/00 Woojung Huh + * Bug fixed + * 11/16/00 Woojung Huh [whuh@applieddata.net] + * Added smartio device driver on it + */ + +/* + * Introduced setkeycode, ketkeycode for the GC+ by Thomas Thaele + * GC+ now performs like a real PC on the keyboard. + * Warning: this code is still beta! PrntScrn and Pause keys are not + * completely tested and implemented!!! Keyboard driver can be confused + * by hacking like crazy on the keyboard. (hardware problem on serial line?) + */ + +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include + +#define ADS_AVR_IRQ 63 + +#define SMARTIO_IOCTL_BASES 's' +#define SMARTIO_KPD_TIMEOUT _IOW(SMARTIO_IOCTL_BASES, 0, int) +#define SMARTIO_KPD_SETUP _IOW(SMARTIO_IOCTL_BASES, 1, short) +#define SMARTIO_BL_CONTROL _IOW(SMARTIO_IOCTL_BASES, 2, char) +#define SMARTIO_BL_CONTRAST _IOW(SMARTIO_IOCTL_BASES, 3, char) +#define SMARTIO_PORT_CONFIG _IOW(SMARTIO_IOCTL_BASES, 4, char) +#define SMARTIO_SNIFFER_TIMEOUT _IOW(SMARTIO_IOCTL_BASES, 5, long) + + +/* Simple translation table for the SysRq keys */ + +#ifdef CONFIG_MAGIC_SYSRQ +unsigned char pckbd_sysrq_xlate[128] = + "\000\0331234567890-=\177\t" /* 0x00 - 0x0f */ + "qwertyuiop[]\r\000as" /* 0x10 - 0x1f */ + "dfghjkl;'`\000\\zxcv" /* 0x20 - 0x2f */ + "bnm,./\000*\000 \000\201\202\203\204\205" /* 0x30 - 0x3f */ + "\206\207\210\211\212\000\000789-456+1" /* 0x40 - 0x4f */ + "230\177\000\000\213\214\000\000\000\000\000\000\000\000\000\000" /* 0x50 - 0x5f */ + "\r\000/"; /* 0x60 - 0x6f */ +#endif + +/* + * Translation of escaped scancodes to keycodes. + * This is now user-settable. + * The keycodes 1-88,96-111,119 are fairly standard, and + * should probably not be changed - changing might confuse X. + * X also interprets scancode 0x5d (KEY_Begin). + * + * For 1-88 keycode equals scancode. + */ + +#define E0_KPENTER 96 +#define E0_RCTRL 97 +#define E0_KPSLASH 98 +#define E0_PRSCR 99 +#define E0_RALT 100 +#define E0_BREAK 101 /* (control-pause) */ +#define E0_HOME 102 +#define E0_UP 103 +#define E0_PGUP 104 +#define E0_LEFT 105 +#define E0_RIGHT 106 +#define E0_END 107 +#define E0_DOWN 108 +#define E0_PGDN 109 +#define E0_INS 110 +#define E0_DEL 111 + +#define E1_PAUSE 119 + +/* + * The keycodes below are randomly located in 89-95,112-118,120-127. + * They could be thrown away (and all occurrences below replaced by 0), + * but that would force many users to use the `setkeycodes' utility, where + * they needed not before. It does not matter that there are duplicates, as + * long as no duplication occurs for any single keyboard. + */ +#define SC_LIM 89 + +#define FOCUS_PF1 85 /* actual code! */ +#define FOCUS_PF2 89 +#define FOCUS_PF3 90 +#define FOCUS_PF4 91 +#define FOCUS_PF5 92 +#define FOCUS_PF6 93 +#define FOCUS_PF7 94 +#define FOCUS_PF8 95 +#define FOCUS_PF9 120 +#define FOCUS_PF10 121 +#define FOCUS_PF11 122 +#define FOCUS_PF12 123 + +#define JAP_86 124 +/* tfj@olivia.ping.dk: + * The four keys are located over the numeric keypad, and are + * labelled A1-A4. It's an rc930 keyboard, from + * Regnecentralen/RC International, Now ICL. + * Scancodes: 59, 5a, 5b, 5c. + */ +#define RGN1 124 +#define RGN2 125 +#define RGN3 126 +#define RGN4 127 + +static unsigned char high_keys[128 - SC_LIM] = { + RGN1, RGN2, RGN3, RGN4, 0, 0, 0, /* 0x59-0x5f */ + 0, 0, 0, 0, 0, 0, 0, 0, /* 0x60-0x67 */ + 0, 0, 0, 0, 0, FOCUS_PF11, 0, FOCUS_PF12, /* 0x68-0x6f */ + 0, 0, 0, FOCUS_PF2, FOCUS_PF9, 0, 0, FOCUS_PF3, /* 0x70-0x77 */ + FOCUS_PF4, FOCUS_PF5, FOCUS_PF6, FOCUS_PF7, /* 0x78-0x7b */ + FOCUS_PF8, JAP_86, FOCUS_PF10, 0 /* 0x7c-0x7f */ +}; + +/* BTC */ +#define E0_MACRO 112 +/* LK450 */ +#define E0_F13 113 +#define E0_F14 114 +#define E0_HELP 115 +#define E0_DO 116 +#define E0_F17 117 +#define E0_KPMINPLUS 118 +/* + * My OmniKey generates e0 4c for the "OMNI" key and the + * right alt key does nada. [kkoller@nyx10.cs.du.edu] + */ +#define E0_OK 124 +/* + * New microsoft keyboard is rumoured to have + * e0 5b (left window button), e0 5c (right window button), + * e0 5d (menu button). [or: LBANNER, RBANNER, RMENU] + * [or: Windows_L, Windows_R, TaskMan] + */ +#define E0_MSLW 125 +#define E0_MSRW 126 +#define E0_MSTM 127 + +static unsigned char e0_keys[128] = { + 0, 0, 0, 0, 0, 0, 0, 0, /* 0x00-0x07 */ + 0, 0, 0, 0, 0, 0, 0, 0, /* 0x08-0x0f */ + 0, 0, 0, 0, 0, 0, 0, 0, /* 0x10-0x17 */ + 0, 0, 0, 0, E0_KPENTER, E0_RCTRL, 0, 0, /* 0x18-0x1f */ + 0, 0, 0, 0, 0, 0, 0, 0, /* 0x20-0x27 */ + 0, 0, 0, 0, 0, 0, 0, 0, /* 0x28-0x2f */ + 0, 0, 0, 0, 0, E0_KPSLASH, 0, E0_PRSCR, /* 0x30-0x37 */ + E0_RALT, 0, 0, 0, 0, E0_F13, E0_F14, E0_HELP, /* 0x38-0x3f */ + E0_DO, E0_F17, 0, 0, 0, 0, E0_BREAK, E0_HOME, /* 0x40-0x47 */ + E0_UP, E0_PGUP, 0, E0_LEFT, E0_OK, E0_RIGHT, E0_KPMINPLUS, E0_END,/* 0x48-0x4f */ + E0_DOWN, E0_PGDN, E0_INS, E0_DEL, 0, 0, 0, 0, /* 0x50-0x57 */ + 0, 0, 0, E0_MSLW, E0_MSRW, E0_MSTM, 0, 0, /* 0x58-0x5f */ + 0, 0, 0, 0, 0, 0, 0, 0, /* 0x60-0x67 */ + 0, 0, 0, 0, 0, 0, 0, E0_MACRO, /* 0x68-0x6f */ + 0, 0, 0, 0, 0, 0, 0, 0, /* 0x70-0x77 */ + 0, 0, 0, 0, 0, 0, 0, 0 /* 0x78-0x7f */ +}; + +int gc_kbd_setkeycode(unsigned int scancode, unsigned int keycode) +{ + if (scancode < SC_LIM || scancode > 255 || keycode > 127) + return -EINVAL; + if (scancode < 128) + high_keys[scancode - SC_LIM] = keycode; + else + e0_keys[scancode - 128] = keycode; + return 0; +} + +int gc_kbd_getkeycode(unsigned int scancode) +{ + return + (scancode < SC_LIM || scancode > 255) ? -EINVAL : + (scancode < 128) ? high_keys[scancode - SC_LIM] : + e0_keys[scancode - 128]; +} + +int gc_kbd_translate(unsigned char scancode, unsigned char *keycode, + char raw_mode) +{ + static int prev_scancode; + + /* special prefix scancodes.. */ + if (scancode == 0xe0 || scancode == 0xe1) { + prev_scancode = scancode; + return 0; + } + + /* 0xFF is sent by a few keyboards, ignore it. 0x00 is error */ + if (scancode == 0x00 || scancode == 0xff) { + prev_scancode = 0; + return 0; + } + + scancode &= 0x7f; + + if (prev_scancode) { + /* + * usually it will be 0xe0, but a Pause key generates + * e1 1d 45 e1 9d c5 when pressed, and nothing when released + */ + if (prev_scancode != 0xe0) { + if (prev_scancode == 0xe1 && scancode == 0x1d) { + prev_scancode = 0x100; + return 0; + } else if (prev_scancode == 0x100 && scancode == 0x45) { + *keycode = E1_PAUSE; + prev_scancode = 0; + } else { +#ifdef KBD_REPORT_UNKN + if (!raw_mode) + printk(KERN_INFO "keyboard: unknown e1 escape sequence\n"); +#endif + prev_scancode = 0; + return 0; + } + } else { + prev_scancode = 0; + /* + * The keyboard maintains its own internal caps lock and + * num lock statuses. In caps lock mode E0 AA precedes make + * code and E0 2A follows break code. In num lock mode, + * E0 2A precedes make code and E0 AA follows break code. + * We do our own book-keeping, so we will just ignore these. + */ + /* + * For my keyboard there is no caps lock mode, but there are + * both Shift-L and Shift-R modes. The former mode generates + * E0 2A / E0 AA pairs, the latter E0 B6 / E0 36 pairs. + * So, we should also ignore the latter. - aeb@cwi.nl + */ + if (scancode == 0x2a || scancode == 0x36) + return 0; + + if (e0_keys[scancode]) + *keycode = e0_keys[scancode]; + else { +#ifdef KBD_REPORT_UNKN + if (!raw_mode) + printk(KERN_INFO "keyboard: unknown scancode e0 %02x\n", + scancode); +#endif + return 0; + } + } + } else if (scancode >= SC_LIM) { + /* This happens with the FOCUS 9000 keyboard + Its keys PF1..PF12 are reported to generate + 55 73 77 78 79 7a 7b 7c 74 7e 6d 6f + Moreover, unless repeated, they do not generate + key-down events, so we have to zero up_flag below */ + /* Also, Japanese 86/106 keyboards are reported to + generate 0x73 and 0x7d for \ - and \ | respectively. */ + /* Also, some Brazilian keyboard is reported to produce + 0x73 and 0x7e for \ ? and KP-dot, respectively. */ + + *keycode = high_keys[scancode - SC_LIM]; + + if (!*keycode) { + if (!raw_mode) { +#ifdef KBD_REPORT_UNKN + printk(KERN_INFO "keyboard: unrecognized scancode (%02x)" + " - ignored\n", scancode); +#endif + } + return 0; + } + } else + *keycode = scancode; + return 1; +} + +// this table converts the hardware dependent codes of a MF-2 Keyboard to +// the codes normally comming out of a i8042. This table is 128 Bytes too +// big, but for stability reasons it should be kept like it is! +// There is no range checking in the code! +static int mf_two_kbdmap[256] = { + 00, 67, 65, 63, 61, 59, 60, 88, 00, 68, 66, 64, 62, 15, 41, 00, + 00, 56, 42, 00, 29, 16, 02, 00, 00, 00, 44, 31, 30, 17, 03, 00, + 00, 46, 45, 32, 18, 05, 04, 00, 00, 57, 47, 33, 20, 19, 06, 00, + 00, 49, 48, 35, 34, 21, 7, 00, 00, 00, 50, 36, 22, 8, 9, 00, + 00, 51, 37, 23, 24, 11, 10, 00, 00, 52, 53, 38, 39, 25, 12, 00, + 00, 00, 40, 00, 26, 13, 00, 00, 58, 54, 28, 27, 00, 43, 00, 00, + 00, 86, 00, 00, 00, 00, 14, 00, 00, 79, 00, 75, 71, 00, 00, 00, + 82, 83, 80, 76, 77, 72, 01, 69, 87, 78, 81, 74, 55, 73, 70, 00, + 00, 00, 00, 65, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, + 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, + 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, + 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, + 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, + 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, + 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, + 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00, 00 }; + + +// some texts displayed by the proc_file_system +static char *kbd_sniff[2] = { "off", "on" }; +static char *kbd_sniff_mode[2] = { "passive", "active" }; + +#define PASSIVE 0 +#define ACTIVE 1 + +// is the sniffer active (1) or inactive (0) +static int SNIFFER = 0; +// do we get a copy (SNIFFMODE = PASSIVE) or do we get the original data (SNIFFMODE = ACTIVE) +// and have to reinsert the data +static int SNIFFMODE = PASSIVE; + +// we allow only one process to sniff +static int sniffer_in_use = 0; + +// timeout for the keyboard sniffer -1 = blocking, otherwise timeout in msecs +static long sniffer_timeout = -1; + +// the value we sniffed from the keyboard +static int sniffed_value; + +static char *smartio_version = "1.02 MF-II compatibility patch "; +static char *smartio_date = "Aug-27-2001"; + +static int sio_reset_flag; +static int kbd_press_flag; + +static void send_SSP_msg(unchar *pBuf, int num) +{ + ushort tmp; + int i; + + for (i=0;i 7)) + return 0xFFFF; + + CONV_ADC_CMD.Opt[0] = (unchar) channel; + + lock_smartio(&flags); + send_SSP_msg((unchar *) &CONV_ADC_CMD, 3); + unlock_smartio(&flags); + + interruptible_sleep_on(&smartio_adc_queue); + + return adc_value & 0x3FF; +} + +static ushort read_sio_port(int port) +{ + unsigned long flags; + ushort ret; + + if ((port < SMARTIO_PORT_B) || (port > SMARTIO_PORT_D)) + return 0xFFFF; + + READ_PORT_CMD.Code = (unchar) port; + + lock_smartio(&flags); + send_SSP_msg((unchar *) &READ_PORT_CMD, 2); + ret = read_SSP_response(1); + unlock_smartio(&flags); + + return ret; +} + +static ushort read_sio_kpd(void) +{ + long timeout; + + // kpd_timeout is mSec order + // interrupt_sleep_on_timeout is based on 10msec timer tick + if (kpd_timeout == -1) { + interruptible_sleep_on(&smartio_kpd_queue); + } + else { + timeout = interruptible_sleep_on_timeout(&smartio_kpd_queue, + kpd_timeout/10); + if (timeout == 0) { + // timeout without keypad input + return 0xFFFF; + } + } + return kpd_value; +} + +static ushort read_sio_sniff(void) +{ + long timeout; + + // kpd_timeout is mSec order + // interrupt_sleep_on_timeout is based on 10msec timer tick + if (sniffer_timeout == -1) { + interruptible_sleep_on(&sniffer_queue); + } + else { + timeout = interruptible_sleep_on_timeout(&sniffer_queue, + sniffer_timeout/10); + if (timeout == 0) { + // timeout without keypad input + return -1; + } + } + return (ushort)sniffed_value; +} + +static struct sio_ver { + uint DevVer; + uint DevType; + uint FwLevel; +}; + +static ushort read_sio_version(struct sio_ver *ptr) +{ + unsigned long flags; + ushort ret; + + // Read Device Version + lock_smartio(&flags); + send_SSP_msg((unchar *) &READ_DEVVER_CMD, 2); + ret = read_SSP_response(1); + unlock_smartio(&flags); + ptr->DevVer = (uint)ret; + // Read Device Type + lock_smartio(&flags); + send_SSP_msg((unchar *) &READ_DEVTYPE_CMD, 2); + ret = read_SSP_response(2); + unlock_smartio(&flags); + // swap MSB & LSB + ret = ((ret & 0xFF) << 8) | ((ret & 0xFF00) >> 8); + ptr->DevType = (uint)ret; + // Read Firmware Level + lock_smartio(&flags); + send_SSP_msg((unchar *) &READ_FWLEVEL_CMD, 2); + ret = read_SSP_response(2); + unlock_smartio(&flags); + // swap MSB & LSB + ret = ((ret & 0xFF) << 8) | ((ret & 0xFF00) >> 8); + ptr->FwLevel = (uint)ret; + + return 0; +} + +static ssize_t sio_read(struct file *file, char *buf, size_t count, loff_t *ppos) +{ + struct inode *inode = file->f_dentry->d_inode; + unsigned int minor = MINOR(inode->i_rdev); + ushort *ret = (ushort *)buf; + + switch (minor) { + case SMARTIO_ADC: + if ((*ret = read_sio_adc(buf[0])) != 0xFFFF) + return sizeof(ushort); // 2 bytes + case SMARTIO_PORT_B: + case SMARTIO_PORT_C: + case SMARTIO_PORT_D: + if ((*ret = read_sio_port(minor)) != 0xFFFF) + return sizeof(ushort); + case SMARTIO_VERSION: + if ((read_sio_version((struct sio_ver *)buf)) != 0xFFFF) + return sizeof(struct sio_ver); + case SMARTIO_KEYPAD: + if ((*ret = read_sio_kpd()) != 0xFFFF) + return sizeof(ushort); + case SMARTIO_KBD_SNIFFER: + if ((*ret = read_sio_sniff()) != (ushort)-1) + return 1; + default : + return -ENXIO; + } +} + +static SMARTIO_CMD WRITE_PORT_CMD = { 0x81, 0x00, { 0x00, 0x00 } }; +static SMARTIO_CMD SELECT_OPT_CMD = { 0x80, 0x00, { 0x00, 0x00 } }; +static SMARTIO_CMD CONTROL_BL_CMD = { 0x80, 0x00, { 0x00, 0x00 } }; +static SMARTIO_CMD CONTRAST_BL_CMD = { 0x80, 0x21, { 0x00, 0x00 } }; +static SMARTIO_CMD CONTROL_KPD_CMD = { 0x80, 0x27, { 0x00, 0x00 } }; +static SMARTIO_CMD CONTROL_VEE_CMD = { 0x80, 0x22, { 0x00, 0x00 } }; + +static ushort write_sio_port(int port, unchar value) +{ + unsigned long flags; + + if ((port < SMARTIO_PORT_B) || (port > SMARTIO_PORT_D)) + return 0xFFFF; + + WRITE_PORT_CMD.Code = (unchar) port; + WRITE_PORT_CMD.Opt[0] = (unchar) value; + + lock_smartio(&flags); + send_SSP_msg((unchar *) &WRITE_PORT_CMD, 3); + unlock_smartio(&flags); + + return 0; +} + +static ushort write_sio_select(unchar select) +{ + unsigned long flags; + + if ((select < 1) || (select > 2)) + return 0xFFFF; + + SELECT_OPT_CMD.Code = (unchar) (select + 0x28); + + lock_smartio(&flags); + send_SSP_msg((unchar *) &SELECT_OPT_CMD, 2); + unlock_smartio(&flags); + + return 0; +} + +static ushort control_sio_backlite(int cmd, int value) +{ + unsigned long flags; + + if (cmd == SMARTIO_BL_CONTRAST) { + value &= 0xFF; + CONTRAST_BL_CMD.Opt[0] = (unchar) value; + + lock_smartio(&flags); + send_SSP_msg((unchar *) &CONTRAST_BL_CMD, 3); + unlock_smartio(&flags); + } + else if (cmd == SMARTIO_BL_CONTROL) { + if (value == 0x00) { + // Backlite OFF + CONTROL_BL_CMD.Code = 0x24; + } + else { + // Backlite ON + CONTROL_BL_CMD.Code = 0x23; + } + lock_smartio(&flags); + send_SSP_msg((unchar *) &CONTROL_BL_CMD, 2); + unlock_smartio(&flags); + } + else + return 0xFFFF; + + return 0; +} + +static ushort control_sio_keypad(int x, int y) +{ + unsigned long flags; + + if ( (x<1) || (x>8) || (y<1) || (y>8)) { + return 0xFFFF; + } + + CONTROL_KPD_CMD.Opt[0] = (unchar) x; + CONTROL_KPD_CMD.Opt[1] = (unchar) y; + + lock_smartio(&flags); + send_SSP_msg((unchar *) &CONTROL_KPD_CMD, 4); + unlock_smartio(&flags); + + return 0; +} + +static ushort control_sio_vee(int value) +{ + unsigned long flags; + + value &= 0xFF; + CONTROL_VEE_CMD.Opt[0] = (unchar) value; + + lock_smartio(&flags); + send_SSP_msg((unchar *) &CONTROL_VEE_CMD, 3); + unlock_smartio(&flags); + + return 0; +} + +static ssize_t sio_write(struct file *file, const char *buf, size_t cont, loff_t *ppos) +{ + struct inode *inode = file->f_dentry->d_inode; + unsigned int minor = MINOR(inode->i_rdev); + + switch (minor) { + case SMARTIO_PORT_B: + case SMARTIO_PORT_C: + case SMARTIO_PORT_D: + if (write_sio_port(minor, buf[0]) != 0xFFFF) + return 1; + case SMARTIO_SELECT_OPTION: + if (write_sio_select(buf[0]) != 0xFFFF) + return 1; + case SMARTIO_BACKLITE: + if (control_sio_backlite(SMARTIO_BL_CONTROL, buf[0]) != 0xFFFF) + return 1; + case SMARTIO_KEYPAD: + if (control_sio_keypad(buf[0], buf[1]) != 0xFFFF) + return 2; + case SMARTIO_VEE_PWM: + if (control_sio_vee(buf[0]) != 0xFFFF) + return 1; + case SMARTIO_KBD_SNIFFER: + // here are the scancodes injected + handle_scancode((unchar)buf[0], (buf[0] & 0x80) ? 0 : 1); + wake_up_interruptible(&keyboard_done_queue); + // give some time to process! File IO is a bit faster than manual typing ;-) + udelay(10000); + return 1; + default: + return -ENXIO; + } +} + +static unsigned int sio_poll(struct file *file, struct poll_table_struct *wait) +{ + return 0; +} + +static SMARTIO_CMD IOCTL_PORT_CMD = { 0x81, 0x00, { 0x00, 0x00 } }; + +static ushort ioctl_sio_port(int port, unchar value) +{ + unsigned long flags; + + if ((port < SMARTIO_PORT_B) || (port > SMARTIO_PORT_D)) + return 0xFFFF; + + IOCTL_PORT_CMD.Code = (unchar) port + 0x04; // 0x05 ~ 0x08 + if (port == SMARTIO_PORT_B) { + // Port B has 4 bits only + IOCTL_PORT_CMD.Opt[0] = (unchar) value & 0x0F; + } + else + IOCTL_PORT_CMD.Opt[0] = (unchar) value; + + lock_smartio(&flags); + send_SSP_msg((unchar *) &IOCTL_PORT_CMD, 3); + unlock_smartio(&flags); + + return 0; +} + +static int sio_ioctl(struct inode *inode, struct file *file, unsigned int cmd, unsigned long arg) +{ + unsigned int minor = MINOR(inode->i_rdev); + unchar *buf = (unchar *)arg; + + switch (minor) { + case SMARTIO_PORT_B: + case SMARTIO_PORT_C: + case SMARTIO_PORT_D: + if (cmd == SMARTIO_PORT_CONFIG) { + if (ioctl_sio_port(minor, buf[0]) != 0xFFFF) + return 0; + } + return -EINVAL; + case SMARTIO_SELECT_OPTION: + if (write_sio_select(buf[0]) != 0xFFFF) return 0; + return -EINVAL; + case SMARTIO_BACKLITE: + if (cmd == SMARTIO_BL_CONTROL) { + if (control_sio_backlite(SMARTIO_BL_CONTROL, buf[0]) != 0xFFFF) return 0; + } + else if (cmd == SMARTIO_BL_CONTRAST) { + if (control_sio_backlite(SMARTIO_BL_CONTRAST, buf[0]) != 0xFFFF) return 0; + } + else return -EINVAL; + case SMARTIO_KEYPAD: + if (cmd == SMARTIO_KPD_TIMEOUT) { + kpd_timeout = *(long*)buf; + return 0; + } + else if (cmd == SMARTIO_KPD_SETUP) { + if (control_sio_keypad(buf[0], buf[1]) != 0xFFFF) return 0; + } + return -EINVAL; + case SMARTIO_VEE_PWM: + if (control_sio_vee(buf[0]) != 0xFFFF) return 0; + return -EINVAL; + case SMARTIO_KBD_SNIFFER: + if (cmd == SMARTIO_SNIFFER_TIMEOUT) { + sniffer_timeout = *(long*)buf; + if (sniffer_timeout < 0) sniffer_timeout = -1; + // the value will be devided by 10 later on + if (!sniffer_timeout) sniffer_timeout = 10; + return 0; + } + return -EINVAL; + default: + return -ENXIO; + } +} + +static int sio_open(struct inode *inode, struct file *file) +{ + unsigned int minor = MINOR(inode->i_rdev); + + // we open all by default. we only have a special handler for the kbd sniffer + switch (minor) { + case SMARTIO_KBD_SNIFFER: + if (sniffer_in_use) return -EBUSY; + sniffer_in_use = 1; + SNIFFER = 1; + // sniff in active or passive mode + if ((file->f_flags & O_RDWR) == O_RDWR) SNIFFMODE = 1; else SNIFFMODE = 0; + // do we have a blocking or non blocking sniffer? + if ((file->f_flags & O_NONBLOCK) == O_NONBLOCK) sniffer_timeout = 100; else sniffer_timeout = -1; + break; + default: + break; + } + return 0; +} + +static int sio_close(struct inode *inode, struct file *file) +{ + unsigned int minor = MINOR(inode->i_rdev); + + switch (minor) { + case SMARTIO_KBD_SNIFFER: + SNIFFER = 0; + SNIFFMODE = 0; + sniffer_in_use = 0; + break; + default: + break; + } + return 0; +} + +static struct file_operations sio_fops = { + read: sio_read, + write: sio_write, + poll: sio_poll, + ioctl: sio_ioctl, + open: sio_open, + release: sio_close, +}; + +static struct proc_dir_entry *sio_dir, *parent_dir = NULL; + +#define SMARTIO_MAJOR 58 +#define MAJOR_NR SMARTIO_MAJOR + +#define PROC_NAME "sio" + +static int sio_read_proc(char *buf, char **start, off_t pos, int count, int *eof, void *data) +{ + char *p = buf; + + p += sprintf(p, "ADS SMARTIO Status: \n"); + p += sprintf(p, "\t Keyboard Interrupt : %lu\n", kbd_int); + p += sprintf(p, "\t Keypad Interrupt : %lu\n", kpd_int); + p += sprintf(p, "\t ADC Interrupt : %lu\n", adc_int); + p += sprintf(p, "\t Keyboard Sniffer : %s mode : %s\n", kbd_sniff[ SNIFFER ], kbd_sniff_mode [ SNIFFMODE ]); + + return (p-buf); +} + +#ifdef CONFIG_PM +static int pm_smartio_callback(struct pm_dev *dev, pm_request_t rqst, void *data) +{ + switch (rqst) { + case PM_RESUME: + gc_sio_init(); + break; + case PM_SUSPEND: + // 4/5/01 Woojung + // It checks Keybard received pair of press/release code. + // System can sleep before receiving release code + if (kbd_press_flag) { + interruptible_sleep_on(&keyboard_done_queue); + } + break; + } + + return 0; +} +#endif + +void __init sio_init(void) +{ + if (register_chrdev(MAJOR_NR, "sio", &sio_fops)) { + printk("smartio : unable to get major %d\n", MAJOR_NR); + return; + } + else { + printk("smartio driver initialized. version %s, date:%s\n", + smartio_version, smartio_date); + + if (sio_reset_flag != 1) { + gc_sio_init(); + if (request_irq(ADS_AVR_IRQ, gc_sio_interrupt,0,"sio",NULL) != 0){ + printk("smartio : Could not allocate IRQ!\n"); + return; + } + } + + if ((sio_dir = create_proc_entry(PROC_NAME, 0, parent_dir)) == NULL) { + printk("smartio : Unable to create /proc entry\n"); + return; + } + else { + sio_dir->read_proc = sio_read_proc; +#ifdef CONFIG_PM + pm_register(PM_SYS_DEV, PM_SYS_KBC, pm_smartio_callback); +#endif + } + } +} diff -urN orig/drivers/char/gckeymap.c linux/drivers/char/gckeymap.c --- orig/drivers/char/gckeymap.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/char/gckeymap.c Tue Aug 28 20:26:52 2001 @@ -0,0 +1,262 @@ +/* Do not edit this file! It was automatically generated by */ +/* loadkeys --mktable defkeymap.map > defkeymap.c */ + +#include +#include +#include + +u_short plain_map[NR_KEYS] = { + 0xf200, 0xf01b, 0xf031, 0xf032, 0xf033, 0xf034, 0xf035, 0xf036, + 0xf037, 0xf038, 0xf039, 0xf030, 0xf02d, 0xf03d, 0xf07f, 0xf009, + 0xfb71, 0xfb77, 0xfb65, 0xfb72, 0xfb74, 0xfb79, 0xfb75, 0xfb69, + 0xfb6f, 0xfb70, 0xf05b, 0xf05d, 0xf201, 0xf702, 0xfb61, 0xfb73, + 0xfb64, 0xfb66, 0xfb67, 0xfb68, 0xfb6a, 0xfb6b, 0xfb6c, 0xf03b, + 0xf027, 0xf060, 0xf700, 0xf05c, 0xfb7a, 0xfb78, 0xfb63, 0xfb76, + 0xfb62, 0xfb6e, 0xfb6d, 0xf02c, 0xf02e, 0xf02f, 0xf700, 0xf30c, + 0xf703, 0xf020, 0xf207, 0xf100, 0xf101, 0xf102, 0xf103, 0xf104, + 0xf105, 0xf106, 0xf107, 0xf108, 0xf109, 0xf208, 0xf209, 0xf307, + 0xf308, 0xf309, 0xf30b, 0xf304, 0xf305, 0xf306, 0xf30a, 0xf301, + 0xf302, 0xf303, 0xf300, 0xf310, 0xf206, 0xf200, 0xf03c, 0xf10a, + 0xf10b, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, + 0xf30e, 0xf702, 0xf30d, 0xf01c, 0xf701, 0xf205, 0xf114, 0xf603, + 0xf118, 0xf601, 0xf602, 0xf117, 0xf600, 0xf119, 0xf115, 0xf116, + 0xf11a, 0xf10c, 0xf10d, 0xf11b, 0xf11c, 0xf110, 0xf311, 0xf11d, + 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, +}; + +u_short shift_map[NR_KEYS] = { + 0xf200, 0xf01b, 0xf021, 0xf040, 0xf023, 0xf024, 0xf025, 0xf05e, + 0xf026, 0xf02a, 0xf028, 0xf029, 0xf05f, 0xf02b, 0xf07f, 0xf009, + 0xfb51, 0xfb57, 0xfb45, 0xfb52, 0xfb54, 0xfb59, 0xfb55, 0xfb49, + 0xfb4f, 0xfb50, 0xf07b, 0xf07d, 0xf201, 0xf702, 0xfb41, 0xfb53, + 0xfb44, 0xfb46, 0xfb47, 0xfb48, 0xfb4a, 0xfb4b, 0xfb4c, 0xf03a, + 0xf022, 0xf07e, 0xf700, 0xf07c, 0xfb5a, 0xfb58, 0xfb43, 0xfb56, + 0xfb42, 0xfb4e, 0xfb4d, 0xf03c, 0xf03e, 0xf03f, 0xf700, 0xf30c, + 0xf703, 0xf020, 0xf207, 0xf10a, 0xf10b, 0xf10c, 0xf10d, 0xf10e, + 0xf10f, 0xf110, 0xf111, 0xf112, 0xf113, 0xf213, 0xf203, 0xf307, + 0xf308, 0xf309, 0xf30b, 0xf304, 0xf305, 0xf306, 0xf30a, 0xf301, + 0xf302, 0xf303, 0xf300, 0xf310, 0xf206, 0xf200, 0xf03e, 0xf10a, + 0xf10b, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, + 0xf30e, 0xf702, 0xf30d, 0xf200, 0xf701, 0xf205, 0xf114, 0xf603, + 0xf20b, 0xf601, 0xf602, 0xf117, 0xf600, 0xf20a, 0xf115, 0xf116, + 0xf11a, 0xf10c, 0xf10d, 0xf11b, 0xf11c, 0xf110, 0xf311, 0xf11d, + 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, +}; + +u_short altgr_map[NR_KEYS] = { + 0xf200, 0xf200, 0xf200, 0xf040, 0xf200, 0xf024, 0xf200, 0xf200, + 0xf07b, 0xf05b, 0xf05d, 0xf07d, 0xf05c, 0xf200, 0xf200, 0xf200, + 0xfb71, 0xfb77, 0xf918, 0xfb72, 0xfb74, 0xfb79, 0xfb75, 0xfb69, + 0xfb6f, 0xfb70, 0xf200, 0xf07e, 0xf201, 0xf702, 0xf914, 0xfb73, + 0xf917, 0xf919, 0xfb67, 0xfb68, 0xfb6a, 0xfb6b, 0xfb6c, 0xf200, + 0xf200, 0xf200, 0xf700, 0xf200, 0xfb7a, 0xfb78, 0xf916, 0xfb76, + 0xf915, 0xfb6e, 0xfb6d, 0xf200, 0xf200, 0xf200, 0xf700, 0xf30c, + 0xf703, 0xf200, 0xf207, 0xf50c, 0xf50d, 0xf50e, 0xf50f, 0xf510, + 0xf511, 0xf512, 0xf513, 0xf514, 0xf515, 0xf208, 0xf202, 0xf911, + 0xf912, 0xf913, 0xf30b, 0xf90e, 0xf90f, 0xf910, 0xf30a, 0xf90b, + 0xf90c, 0xf90d, 0xf90a, 0xf310, 0xf206, 0xf200, 0xf07c, 0xf516, + 0xf517, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, + 0xf30e, 0xf702, 0xf30d, 0xf200, 0xf701, 0xf205, 0xf114, 0xf603, + 0xf118, 0xf601, 0xf602, 0xf117, 0xf600, 0xf119, 0xf115, 0xf116, + 0xf11a, 0xf10c, 0xf10d, 0xf11b, 0xf11c, 0xf110, 0xf311, 0xf11d, + 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, +}; + +u_short ctrl_map[NR_KEYS] = { + 0xf200, 0xf200, 0xf200, 0xf000, 0xf01b, 0xf01c, 0xf01d, 0xf01e, + 0xf01f, 0xf07f, 0xf200, 0xf200, 0xf01f, 0xf200, 0xf008, 0xf200, + 0xf011, 0xf017, 0xf005, 0xf012, 0xf014, 0xf019, 0xf015, 0xf009, + 0xf00f, 0xf010, 0xf01b, 0xf01d, 0xf201, 0xf702, 0xf001, 0xf013, + 0xf004, 0xf006, 0xf007, 0xf008, 0xf00a, 0xf00b, 0xf00c, 0xf200, + 0xf007, 0xf000, 0xf700, 0xf01c, 0xf01a, 0xf018, 0xf003, 0xf016, + 0xf002, 0xf00e, 0xf00d, 0xf200, 0xf20e, 0xf07f, 0xf700, 0xf30c, + 0xf703, 0xf000, 0xf207, 0xf100, 0xf101, 0xf102, 0xf103, 0xf104, + 0xf105, 0xf106, 0xf107, 0xf108, 0xf109, 0xf208, 0xf204, 0xf307, + 0xf308, 0xf309, 0xf30b, 0xf304, 0xf305, 0xf306, 0xf30a, 0xf301, + 0xf302, 0xf303, 0xf300, 0xf310, 0xf206, 0xf200, 0xf200, 0xf10a, + 0xf10b, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, + 0xf30e, 0xf702, 0xf30d, 0xf01c, 0xf701, 0xf205, 0xf114, 0xf603, + 0xf118, 0xf601, 0xf602, 0xf117, 0xf600, 0xf119, 0xf115, 0xf116, + 0xf11a, 0xf10c, 0xf10d, 0xf11b, 0xf11c, 0xf110, 0xf311, 0xf11d, + 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, +}; + +u_short shift_ctrl_map[NR_KEYS] = { + 0xf200, 0xf200, 0xf200, 0xf000, 0xf200, 0xf200, 0xf200, 0xf200, + 0xf200, 0xf200, 0xf200, 0xf200, 0xf01f, 0xf200, 0xf200, 0xf200, + 0xf011, 0xf017, 0xf005, 0xf012, 0xf014, 0xf019, 0xf015, 0xf009, + 0xf00f, 0xf010, 0xf200, 0xf200, 0xf201, 0xf702, 0xf001, 0xf013, + 0xf004, 0xf006, 0xf007, 0xf008, 0xf00a, 0xf00b, 0xf00c, 0xf200, + 0xf200, 0xf200, 0xf700, 0xf200, 0xf01a, 0xf018, 0xf003, 0xf016, + 0xf002, 0xf00e, 0xf00d, 0xf200, 0xf200, 0xf200, 0xf700, 0xf30c, + 0xf703, 0xf200, 0xf207, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, + 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf208, 0xf200, 0xf307, + 0xf308, 0xf309, 0xf30b, 0xf304, 0xf305, 0xf306, 0xf30a, 0xf301, + 0xf302, 0xf303, 0xf300, 0xf310, 0xf206, 0xf200, 0xf200, 0xf200, + 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, + 0xf30e, 0xf702, 0xf30d, 0xf200, 0xf701, 0xf205, 0xf114, 0xf603, + 0xf118, 0xf601, 0xf602, 0xf117, 0xf600, 0xf119, 0xf115, 0xf116, + 0xf11a, 0xf10c, 0xf10d, 0xf11b, 0xf11c, 0xf110, 0xf311, 0xf11d, + 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, +}; + +u_short alt_map[NR_KEYS] = { + 0xf200, 0xf81b, 0xf831, 0xf832, 0xf833, 0xf834, 0xf835, 0xf836, + 0xf837, 0xf838, 0xf839, 0xf830, 0xf82d, 0xf83d, 0xf87f, 0xf809, + 0xf871, 0xf877, 0xf865, 0xf872, 0xf874, 0xf879, 0xf875, 0xf869, + 0xf86f, 0xf870, 0xf85b, 0xf85d, 0xf80d, 0xf702, 0xf861, 0xf873, + 0xf864, 0xf866, 0xf867, 0xf868, 0xf86a, 0xf86b, 0xf86c, 0xf83b, + 0xf827, 0xf860, 0xf700, 0xf85c, 0xf87a, 0xf878, 0xf863, 0xf876, + 0xf862, 0xf86e, 0xf86d, 0xf82c, 0xf82e, 0xf82f, 0xf700, 0xf30c, + 0xf703, 0xf820, 0xf207, 0xf500, 0xf501, 0xf502, 0xf503, 0xf504, + 0xf505, 0xf506, 0xf507, 0xf508, 0xf509, 0xf208, 0xf209, 0xf907, + 0xf908, 0xf909, 0xf30b, 0xf904, 0xf905, 0xf906, 0xf30a, 0xf901, + 0xf902, 0xf903, 0xf900, 0xf310, 0xf206, 0xf200, 0xf83c, 0xf50a, + 0xf50b, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, + 0xf30e, 0xf702, 0xf30d, 0xf01c, 0xf701, 0xf205, 0xf114, 0xf603, + 0xf118, 0xf210, 0xf211, 0xf117, 0xf600, 0xf119, 0xf115, 0xf116, + 0xf11a, 0xf10c, 0xf10d, 0xf11b, 0xf11c, 0xf110, 0xf311, 0xf11d, + 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, +}; + +u_short ctrl_alt_map[NR_KEYS] = { + 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, + 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, + 0xf811, 0xf817, 0xf805, 0xf812, 0xf814, 0xf819, 0xf815, 0xf809, + 0xf80f, 0xf810, 0xf200, 0xf200, 0xf201, 0xf702, 0xf801, 0xf813, + 0xf804, 0xf806, 0xf807, 0xf808, 0xf80a, 0xf80b, 0xf80c, 0xf200, + 0xf200, 0xf200, 0xf700, 0xf200, 0xf81a, 0xf818, 0xf803, 0xf816, + 0xf802, 0xf80e, 0xf80d, 0xf200, 0xf200, 0xf200, 0xf700, 0xf30c, + 0xf703, 0xf200, 0xf207, 0xf500, 0xf501, 0xf502, 0xf503, 0xf504, + 0xf505, 0xf506, 0xf507, 0xf508, 0xf509, 0xf208, 0xf200, 0xf307, + 0xf308, 0xf309, 0xf30b, 0xf304, 0xf305, 0xf306, 0xf30a, 0xf301, + 0xf302, 0xf303, 0xf300, 0xf20c, 0xf206, 0xf200, 0xf200, 0xf50a, + 0xf50b, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, + 0xf30e, 0xf702, 0xf30d, 0xf200, 0xf701, 0xf205, 0xf114, 0xf603, + 0xf118, 0xf601, 0xf602, 0xf117, 0xf600, 0xf119, 0xf115, 0xf20c, + 0xf11a, 0xf10c, 0xf10d, 0xf11b, 0xf11c, 0xf110, 0xf311, 0xf11d, + 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, 0xf200, +}; + +ushort *key_maps[MAX_NR_KEYMAPS] = { + plain_map, shift_map, altgr_map, 0, + ctrl_map, shift_ctrl_map, 0, 0, + alt_map, 0, 0, 0, + ctrl_alt_map, 0 +}; + +unsigned int keymap_count = 7; + +/* + * Philosophy: most people do not define more strings, but they who do + * often want quite a lot of string space. So, we statically allocate + * the default and allocate dynamically in chunks of 512 bytes. + */ + +char func_buf[] = { + '\033', '[', '[', 'A', 0, + '\033', '[', '[', 'B', 0, + '\033', '[', '[', 'C', 0, + '\033', '[', '[', 'D', 0, + '\033', '[', '[', 'E', 0, + '\033', '[', '1', '7', '~', 0, + '\033', '[', '1', '8', '~', 0, + '\033', '[', '1', '9', '~', 0, + '\033', '[', '2', '0', '~', 0, + '\033', '[', '2', '1', '~', 0, + '\033', '[', '2', '3', '~', 0, + '\033', '[', '2', '4', '~', 0, + '\033', '[', '2', '5', '~', 0, + '\033', '[', '2', '6', '~', 0, + '\033', '[', '2', '8', '~', 0, + '\033', '[', '2', '9', '~', 0, + '\033', '[', '3', '1', '~', 0, + '\033', '[', '3', '2', '~', 0, + '\033', '[', '3', '3', '~', 0, + '\033', '[', '3', '4', '~', 0, + '\033', '[', '1', '~', 0, + '\033', '[', '2', '~', 0, + '\033', '[', '3', '~', 0, + '\033', '[', '4', '~', 0, + '\033', '[', '5', '~', 0, + '\033', '[', '6', '~', 0, + '\033', '[', 'M', 0, + '\033', '[', 'P', 0, +}; + +char *funcbufptr = func_buf; +int funcbufsize = sizeof(func_buf); +int funcbufleft = 0; /* space left */ + +char *func_table[MAX_NR_FUNC] = { + func_buf + 0, + func_buf + 5, + func_buf + 10, + func_buf + 15, + func_buf + 20, + func_buf + 25, + func_buf + 31, + func_buf + 37, + func_buf + 43, + func_buf + 49, + func_buf + 55, + func_buf + 61, + func_buf + 67, + func_buf + 73, + func_buf + 79, + func_buf + 85, + func_buf + 91, + func_buf + 97, + func_buf + 103, + func_buf + 109, + func_buf + 115, + func_buf + 120, + func_buf + 125, + func_buf + 130, + func_buf + 135, + func_buf + 140, + func_buf + 145, + 0, + 0, + func_buf + 149, + 0, +}; + +struct kbdiacr accent_table[MAX_DIACR] = { + {'`', 'A', '\300'}, {'`', 'a', '\340'}, + {'\'', 'A', '\301'}, {'\'', 'a', '\341'}, + {'^', 'A', '\302'}, {'^', 'a', '\342'}, + {'~', 'A', '\303'}, {'~', 'a', '\343'}, + {'"', 'A', '\304'}, {'"', 'a', '\344'}, + {'O', 'A', '\305'}, {'o', 'a', '\345'}, + {'0', 'A', '\305'}, {'0', 'a', '\345'}, + {'A', 'A', '\305'}, {'a', 'a', '\345'}, + {'A', 'E', '\306'}, {'a', 'e', '\346'}, + {',', 'C', '\307'}, {',', 'c', '\347'}, + {'`', 'E', '\310'}, {'`', 'e', '\350'}, + {'\'', 'E', '\311'}, {'\'', 'e', '\351'}, + {'^', 'E', '\312'}, {'^', 'e', '\352'}, + {'"', 'E', '\313'}, {'"', 'e', '\353'}, + {'`', 'I', '\314'}, {'`', 'i', '\354'}, + {'\'', 'I', '\315'}, {'\'', 'i', '\355'}, + {'^', 'I', '\316'}, {'^', 'i', '\356'}, + {'"', 'I', '\317'}, {'"', 'i', '\357'}, + {'-', 'D', '\320'}, {'-', 'd', '\360'}, + {'~', 'N', '\321'}, {'~', 'n', '\361'}, + {'`', 'O', '\322'}, {'`', 'o', '\362'}, + {'\'', 'O', '\323'}, {'\'', 'o', '\363'}, + {'^', 'O', '\324'}, {'^', 'o', '\364'}, + {'~', 'O', '\325'}, {'~', 'o', '\365'}, + {'"', 'O', '\326'}, {'"', 'o', '\366'}, + {'/', 'O', '\330'}, {'/', 'o', '\370'}, + {'`', 'U', '\331'}, {'`', 'u', '\371'}, + {'\'', 'U', '\332'}, {'\'', 'u', '\372'}, + {'^', 'U', '\333'}, {'^', 'u', '\373'}, + {'"', 'U', '\334'}, {'"', 'u', '\374'}, + {'\'', 'Y', '\335'}, {'\'', 'y', '\375'}, + {'T', 'H', '\336'}, {'t', 'h', '\376'}, + {'s', 's', '\337'}, {'"', 'y', '\377'}, + {'s', 'z', '\337'}, {'i', 'j', '\377'}, +}; + +unsigned int accent_table_size = 68; diff -urN orig/drivers/char/gckeymap.map linux/drivers/char/gckeymap.map --- orig/drivers/char/gckeymap.map Thu Jan 1 01:00:00 1970 +++ linux/drivers/char/gckeymap.map Tue Aug 28 20:26:52 2001 @@ -0,0 +1,357 @@ +# Default kernel keymap. This uses 7 modifier combinations. +keymaps 0-2,4-5,8,12 +# Change the above line into +# keymaps 0-2,4-6,8,12 +# in case you want the entries +# altgr control keycode 83 = Boot +# altgr control keycode 111 = Boot +# below. +# +# In fact AltGr is used very little, and one more keymap can +# be saved by mapping AltGr to Alt (and adapting a few entries): +# keycode 100 = Alt +# +keycode 1 = Escape Escape + alt keycode 1 = Meta_Escape +keycode 2 = one exclam + alt keycode 2 = Meta_one +keycode 3 = two at at + control keycode 3 = nul + shift control keycode 3 = nul + alt keycode 3 = Meta_two +keycode 4 = three numbersign + control keycode 4 = Escape + alt keycode 4 = Meta_three +keycode 5 = four dollar dollar + control keycode 5 = Control_backslash + alt keycode 5 = Meta_four +keycode 6 = five percent + control keycode 6 = Control_bracketright + alt keycode 6 = Meta_five +keycode 7 = six asciicircum + control keycode 7 = Control_asciicircum + alt keycode 7 = Meta_six +keycode 8 = seven ampersand braceleft + control keycode 8 = Control_underscore + alt keycode 8 = Meta_seven +keycode 9 = eight asterisk bracketleft + control keycode 9 = Delete + alt keycode 9 = Meta_eight +keycode 10 = nine parenleft bracketright + alt keycode 10 = Meta_nine +keycode 11 = zero parenright braceright + alt keycode 11 = Meta_zero +keycode 12 = minus underscore backslash + control keycode 12 = Control_underscore + shift control keycode 12 = Control_underscore + alt keycode 12 = Meta_minus +keycode 13 = equal plus + alt keycode 13 = Meta_equal +keycode 14 = Delete Delete + control keycode 14 = BackSpace + alt keycode 14 = Meta_Delete +keycode 15 = Tab Tab + alt keycode 15 = Meta_Tab +keycode 16 = q +keycode 17 = w +keycode 18 = e + altgr keycode 18 = Hex_E +keycode 19 = r +keycode 20 = t +keycode 21 = y +keycode 22 = u +keycode 23 = i +keycode 24 = o +keycode 25 = p +keycode 26 = bracketleft braceleft + control keycode 26 = Escape + alt keycode 26 = Meta_bracketleft +keycode 27 = bracketright braceright asciitilde + control keycode 27 = Control_bracketright + alt keycode 27 = Meta_bracketright +keycode 28 = Return + alt keycode 28 = Meta_Control_m +keycode 29 = Control +keycode 30 = a + altgr keycode 30 = Hex_A +keycode 31 = s +keycode 32 = d + altgr keycode 32 = Hex_D +keycode 33 = f + altgr keycode 33 = Hex_F +keycode 34 = g +keycode 35 = h +keycode 36 = j +keycode 37 = k +keycode 38 = l +keycode 39 = semicolon colon + alt keycode 39 = Meta_semicolon +keycode 40 = apostrophe quotedbl + control keycode 40 = Control_g + alt keycode 40 = Meta_apostrophe +keycode 41 = grave asciitilde + control keycode 41 = nul + alt keycode 41 = Meta_grave +keycode 42 = Shift +keycode 43 = backslash bar + control keycode 43 = Control_backslash + alt keycode 43 = Meta_backslash +keycode 44 = z +keycode 45 = x +keycode 46 = c + altgr keycode 46 = Hex_C +keycode 47 = v +keycode 48 = b + altgr keycode 48 = Hex_B +keycode 49 = n +keycode 50 = m +keycode 51 = comma less + alt keycode 51 = Meta_comma +keycode 52 = period greater + control keycode 52 = Compose + alt keycode 52 = Meta_period +keycode 53 = slash question + control keycode 53 = Delete + alt keycode 53 = Meta_slash +keycode 54 = Shift +keycode 55 = KP_Multiply +keycode 56 = Alt +keycode 57 = space space + control keycode 57 = nul + alt keycode 57 = Meta_space +keycode 58 = Caps_Lock +keycode 59 = F1 F11 Console_13 + control keycode 59 = F1 + alt keycode 59 = Console_1 + control alt keycode 59 = Console_1 +keycode 60 = F2 F12 Console_14 + control keycode 60 = F2 + alt keycode 60 = Console_2 + control alt keycode 60 = Console_2 +keycode 61 = F3 F13 Console_15 + control keycode 61 = F3 + alt keycode 61 = Console_3 + control alt keycode 61 = Console_3 +keycode 62 = F4 F14 Console_16 + control keycode 62 = F4 + alt keycode 62 = Console_4 + control alt keycode 62 = Console_4 +keycode 63 = F5 F15 Console_17 + control keycode 63 = F5 + alt keycode 63 = Console_5 + control alt keycode 63 = Console_5 +keycode 64 = F6 F16 Console_18 + control keycode 64 = F6 + alt keycode 64 = Console_6 + control alt keycode 64 = Console_6 +keycode 65 = F7 F17 Console_19 + control keycode 65 = F7 + alt keycode 65 = Console_7 + control alt keycode 65 = Console_7 +keycode 66 = F8 F18 Console_20 + control keycode 66 = F8 + alt keycode 66 = Console_8 + control alt keycode 66 = Console_8 +keycode 67 = F9 F19 Console_21 + control keycode 67 = F9 + alt keycode 67 = Console_9 + control alt keycode 67 = Console_9 +keycode 68 = F10 F20 Console_22 + control keycode 68 = F10 + alt keycode 68 = Console_10 + control alt keycode 68 = Console_10 +keycode 69 = Num_Lock + shift keycode 69 = Bare_Num_Lock +keycode 70 = Scroll_Lock Show_Memory Show_Registers + control keycode 70 = Show_State + alt keycode 70 = Scroll_Lock +keycode 71 = KP_7 + alt keycode 71 = Ascii_7 + altgr keycode 71 = Hex_7 +keycode 72 = KP_8 + alt keycode 72 = Ascii_8 + altgr keycode 72 = Hex_8 +keycode 73 = KP_9 + alt keycode 73 = Ascii_9 + altgr keycode 73 = Hex_9 +keycode 74 = KP_Subtract +keycode 75 = KP_4 + alt keycode 75 = Ascii_4 + altgr keycode 75 = Hex_4 +keycode 76 = KP_5 + alt keycode 76 = Ascii_5 + altgr keycode 76 = Hex_5 +keycode 77 = KP_6 + alt keycode 77 = Ascii_6 + altgr keycode 77 = Hex_6 +keycode 78 = KP_Add +keycode 79 = KP_1 + alt keycode 79 = Ascii_1 + altgr keycode 79 = Hex_1 +keycode 80 = KP_2 + alt keycode 80 = Ascii_2 + altgr keycode 80 = Hex_2 +keycode 81 = KP_3 + alt keycode 81 = Ascii_3 + altgr keycode 81 = Hex_3 +keycode 82 = KP_0 + alt keycode 82 = Ascii_0 + altgr keycode 82 = Hex_0 +keycode 83 = KP_Period +# altgr control keycode 83 = Boot + control alt keycode 83 = Boot +keycode 84 = Last_Console +keycode 85 = +keycode 86 = less greater bar + alt keycode 86 = Meta_less +keycode 87 = F11 F11 Console_23 + control keycode 87 = F11 + alt keycode 87 = Console_11 + control alt keycode 87 = Console_11 +keycode 88 = F12 F12 Console_24 + control keycode 88 = F12 + alt keycode 88 = Console_12 + control alt keycode 88 = Console_12 +keycode 89 = +keycode 90 = +keycode 91 = +keycode 92 = +keycode 93 = +keycode 94 = +keycode 95 = +keycode 96 = KP_Enter +keycode 97 = Control +keycode 98 = KP_Divide +keycode 99 = Control_backslash + control keycode 99 = Control_backslash + alt keycode 99 = Control_backslash +keycode 100 = AltGr +keycode 101 = Break +keycode 102 = Find +keycode 103 = Up +keycode 104 = Prior + shift keycode 104 = Scroll_Backward +keycode 105 = Left + alt keycode 105 = Decr_Console +keycode 106 = Right + alt keycode 106 = Incr_Console +keycode 107 = Select +keycode 108 = Down +keycode 109 = Next + shift keycode 109 = Scroll_Forward +keycode 110 = Insert +keycode 111 = Remove +# altgr control keycode 111 = Boot + control alt keycode 111 = Boot +keycode 112 = Macro +keycode 113 = F13 +keycode 114 = F14 +keycode 115 = Help +keycode 116 = Do +keycode 117 = F17 +keycode 118 = KP_MinPlus +keycode 119 = Pause +keycode 120 = +keycode 121 = +keycode 122 = +keycode 123 = +keycode 124 = +keycode 125 = +keycode 126 = +keycode 127 = +string F1 = "\033[[A" +string F2 = "\033[[B" +string F3 = "\033[[C" +string F4 = "\033[[D" +string F5 = "\033[[E" +string F6 = "\033[17~" +string F7 = "\033[18~" +string F8 = "\033[19~" +string F9 = "\033[20~" +string F10 = "\033[21~" +string F11 = "\033[23~" +string F12 = "\033[24~" +string F13 = "\033[25~" +string F14 = "\033[26~" +string F15 = "\033[28~" +string F16 = "\033[29~" +string F17 = "\033[31~" +string F18 = "\033[32~" +string F19 = "\033[33~" +string F20 = "\033[34~" +string Find = "\033[1~" +string Insert = "\033[2~" +string Remove = "\033[3~" +string Select = "\033[4~" +string Prior = "\033[5~" +string Next = "\033[6~" +string Macro = "\033[M" +string Pause = "\033[P" +compose '`' 'A' to 'Ā' +compose '`' 'a' to 'ā' +compose '\'' 'A' to 'Á' +compose '\'' 'a' to 'á' +compose '^' 'A' to 'Â' +compose '^' 'a' to 'â' +compose '~' 'A' to 'Ã' +compose '~' 'a' to 'ã' +compose '"' 'A' to 'Ä' +compose '"' 'a' to 'ä' +compose 'O' 'A' to 'Å' +compose 'o' 'a' to 'å' +compose '0' 'A' to 'Å' +compose '0' 'a' to 'å' +compose 'A' 'A' to 'Å' +compose 'a' 'a' to 'å' +compose 'A' 'E' to 'Æ' +compose 'a' 'e' to 'æ' +compose ',' 'C' to 'Į' +compose ',' 'c' to 'į' +compose '`' 'E' to 'Č' +compose '`' 'e' to 'č' +compose '\'' 'E' to 'É' +compose '\'' 'e' to 'é' +compose '^' 'E' to 'Ę' +compose '^' 'e' to 'ę' +compose '"' 'E' to 'Ë' +compose '"' 'e' to 'ë' +compose '`' 'I' to 'Ė' +compose '`' 'i' to 'ė' +compose '\'' 'I' to 'Í' +compose '\'' 'i' to 'í' +compose '^' 'I' to 'Î' +compose '^' 'i' to 'î' +compose '"' 'I' to 'Ï' +compose '"' 'i' to 'ï' +compose '-' 'D' to 'Ð' +compose '-' 'd' to 'ð' +compose '~' 'N' to 'Ņ' +compose '~' 'n' to 'ņ' +compose '`' 'O' to 'Ō' +compose '`' 'o' to 'ō' +compose '\'' 'O' to 'Ó' +compose '\'' 'o' to 'ó' +compose '^' 'O' to 'Ô' +compose '^' 'o' to 'ô' +compose '~' 'O' to 'Õ' +compose '~' 'o' to 'õ' +compose '"' 'O' to 'Ö' +compose '"' 'o' to 'ö' +compose '/' 'O' to 'Ø' +compose '/' 'o' to 'ø' +compose '`' 'U' to 'Ų' +compose '`' 'u' to 'ų' +compose '\'' 'U' to 'Ú' +compose '\'' 'u' to 'ú' +compose '^' 'U' to 'Û' +compose '^' 'u' to 'û' +compose '"' 'U' to 'Ü' +compose '"' 'u' to 'ü' +compose '\'' 'Y' to 'Ý' +compose '\'' 'y' to 'ý' +compose 'T' 'H' to 'Þ' +compose 't' 'h' to 'þ' +compose 's' 's' to 'ß' +compose '"' 'y' to 'ĸ' +compose 's' 'z' to 'ß' +compose 'i' 'j' to 'ĸ' diff -urN orig/drivers/char/generic_serial.c linux/drivers/char/generic_serial.c --- orig/drivers/char/generic_serial.c Wed Feb 27 14:24:58 2002 +++ linux/drivers/char/generic_serial.c Tue Feb 26 17:38:50 2002 @@ -883,6 +883,9 @@ if(!memcmp(tiosp->c_cc, old_termios->c_cc, NCC)) printk("c_cc changed\n"); } + /* + * should be using tty_get_baud_rate() here -- rmk + */ baudrate = tiosp->c_cflag & CBAUD; if (baudrate & CBAUDEX) { baudrate &= ~CBAUDEX; @@ -957,6 +960,11 @@ unsigned long flags; unsigned long page; + /* + * Do we expect to allocate tmp_buf from an interrupt routine? + * If not, then save_flags() cli() and restore_flags() are + * redundant here and should be replaced by a semaphore. -- rmk + */ save_flags (flags); if (!tmp_buf) { page = get_free_page(GFP_KERNEL); @@ -976,6 +984,11 @@ if (port->flags & ASYNC_INITIALIZED) return 0; + /* + * Do we expect to allocate xmit_buf from an interrupt routine? + * If not, then save_flags() cli() and restore_flags() are + * redundant here and should be replaced by a semaphore. -- rmk + */ if (!port->xmit_buf) { /* We may sleep in get_free_page() */ unsigned long tmp; @@ -1016,7 +1029,7 @@ struct serial_struct sio; if (copy_from_user(&sio, sp, sizeof(struct serial_struct))) - return(-EFAULT); + return -EFAULT; if (!capable(CAP_SYS_ADMIN)) { if ((sio.baud_base != port->baud_base) || diff -urN orig/drivers/char/keyboard.c linux/drivers/char/keyboard.c --- orig/drivers/char/keyboard.c Mon Aug 5 13:30:46 2002 +++ linux/drivers/char/keyboard.c Mon Aug 5 14:55:21 2002 @@ -21,6 +21,10 @@ * * 27-05-97: Added support for the Magic SysRq Key (Martin Mares) * 30-07-98: Dead keys redone, aeb@cwi.nl. + * + * 04-04-1998: Added keyboard autorepeat support (some keyboards don't + * autorepeat, and some keyboard changers interfere with keyboard + * autorepeat settings). - Russell King (rmk@arm.linux.org.uk) */ #include @@ -30,6 +34,7 @@ #include #include #include +#include #include #include @@ -61,6 +66,14 @@ #define KBD_DEFLOCK 0 #endif +/* + * Default autorepeat settings. + * DEFAULT_REPEAT_TIMEOUT is the timeout from the keypress to the first repeat + * DEFAULT_REPEAT_INTERVAL is the timeout between successive repeats + */ +#define DEFAULT_REPEAT_TIMEOUT HZ*300/1000 +#define DEFAULT_REPEAT_INTERVAL HZ*30/1000 + void (*kbd_ledfunc)(unsigned int led); EXPORT_SYMBOL(handle_scancode); EXPORT_SYMBOL(kbd_ledfunc); @@ -90,11 +103,15 @@ static int npadch = -1; /* -1 or number assembled on pad */ static unsigned char diacr; static char rep; /* flag telling character repeat */ +static int kbd_repeatkeycode= -1; +static int kbd_repeattimeout = DEFAULT_REPEAT_TIMEOUT; +static int kbd_repeatinterval= DEFAULT_REPEAT_INTERVAL; struct kbd_struct kbd_table[MAX_NR_CONSOLES]; static struct tty_struct **ttytab; static struct kbd_struct * kbd = kbd_table; static struct tty_struct * tty; +static void kbd_processkeycode(unsigned char scancode, char up_flag, int autorepeat); void compute_shiftstate(void); typedef void (*k_hand)(unsigned char value, char up_flag); @@ -161,7 +178,8 @@ * string, and in both cases we might assume that it is * in utf-8 already. */ -void to_utf8(ushort c) { +void to_utf8(ushort c) +{ if (c < 0x80) put_queue(c); /* 0******* */ else if (c < 0x800) { @@ -180,17 +198,23 @@ * Translation of escaped scancodes to keycodes. * This is now user-settable (for machines were it makes sense). */ - int setkeycode(unsigned int scancode, unsigned int keycode) { - return kbd_setkeycode(scancode, keycode); + return kbd_setkeycode(scancode, keycode); } int getkeycode(unsigned int scancode) { - return kbd_getkeycode(scancode); + return kbd_getkeycode(scancode); } +static void key_callback(unsigned long nr); + +static struct timer_list key_autorepeat_timer = +{ + function: key_callback +}; + void handle_scancode(unsigned char scancode, int down) { unsigned char keycode; @@ -198,6 +222,7 @@ char raw_mode; pm_access(pm_kbd); + add_keyboard_randomness(scancode | up_flag); tty = ttytab? ttytab[fg_console]: NULL; @@ -232,12 +257,35 @@ * return the keycode if in MEDIUMRAW mode. */ + kbd_processkeycode(keycode, up_flag, 0); + +out: + do_poke_blanked_console = 1; + schedule_console_callback(); +} + +static void +kbd_processkeycode(unsigned char keycode, char up_flag, int autorepeat) +{ + char raw_mode = (kbd->kbdmode == VC_RAW); + if (up_flag) { rep = 0; if(!test_and_clear_bit(keycode, key_down)) up_flag = kbd_unexpected_up(keycode); - } else + } else { rep = test_and_set_bit(keycode, key_down); + /* If the keyboard autorepeated for us, ignore it. + * We do our own autorepeat processing. + */ + if (rep && !autorepeat) + return; + } + + if (kbd_repeatkeycode == keycode || !up_flag || raw_mode) { + kbd_repeatkeycode = -1; + del_timer(&key_autorepeat_timer); + } #ifdef CONFIG_MAGIC_SYSRQ /* Handle the SysRq Hack */ if (keycode == SYSRQ_KEY) { @@ -251,6 +299,23 @@ } #endif + /* + * Calculate the next time when we have to do some autorepeat + * processing. Note that we do not do autorepeat processing + * while in raw mode but we do do autorepeat processing in + * medium raw mode. + */ + if (!up_flag && !raw_mode) { + kbd_repeatkeycode = keycode; + if (vc_kbd_mode(kbd, VC_REPEAT)) { + if (rep) + key_autorepeat_timer.expires = jiffies + kbd_repeatinterval; + else + key_autorepeat_timer.expires = jiffies + kbd_repeattimeout; + add_timer(&key_autorepeat_timer); + } + } + if (kbd->kbdmode == VC_MEDIUMRAW) { /* soon keycodes will require more than one byte */ put_queue(keycode + up_flag); @@ -319,9 +384,24 @@ #endif } } + rep = 0; out: - do_poke_blanked_console = 1; - schedule_console_callback(); +} + +/* + * This clears the key down arrays when the keyboard is reset. On + * keyboard reset, this must be called before any keycodes are + * received. + */ +void kbd_reset_kdown(void) +{ + int i; + + for (i = 0; i < NR_SHIFT; i++) + k_down[i] = 0; + for (i = 0; i < SIZE(key_down); i++) + key_down[i] = 0; + shift_state = 0; } #ifdef CONFIG_FORWARD_KEYBOARD @@ -523,7 +603,7 @@ { } -static void do_null() +static void do_null(void) { compute_shiftstate(); } @@ -638,8 +718,8 @@ static void do_pad(unsigned char value, char up_flag) { - static const char *pad_chars = "0123456789+-*/\015,.?()"; - static const char *app_map = "pqrstuvwxylSRQMnnmPQ"; + static const char *pad_chars = "0123456789+-*/\015,.?()#"; + static const char *app_map = "pqrstuvwxylSRQMnnmPQS"; if (up_flag) return; /* no action, if this is a key release */ @@ -740,9 +820,10 @@ } } -/* called after returning from RAW mode or when changing consoles - - recompute k_down[] and shift_state from key_down[] */ -/* maybe called when keymap is undefined, so that shiftkey release is seen */ +/* Called after returning from RAW mode or when changing consoles - + * recompute k_down[] and shift_state from key_down[] + * Maybe called when keymap is undefined so that shift key release is seen + */ void compute_shiftstate(void) { int i, j, k, sym, val; @@ -821,19 +902,22 @@ } /* - * The leds display either (i) the status of NumLock, CapsLock, ScrollLock, - * or (ii) whatever pattern of lights people want to show using KDSETLED, - * or (iii) specified bits of specified words in kernel memory. + * The leds display either + * (i) the status of NumLock, CapsLock, ScrollLock, or + * (ii) whatever pattern of lights people want to show using KDSETLED, or + * (iii) specified bits of specified words in kernel memory. */ static unsigned char ledstate = 0xff; /* undefined */ static unsigned char ledioctl; -unsigned char getledstate(void) { +unsigned char getledstate(void) +{ return ledstate; } -void setledstate(struct kbd_struct *kbd, unsigned int led) { +void setledstate(struct kbd_struct *kbd, unsigned int led) +{ if (!(led & ~7)) { ledioctl = led; kbd->ledmode = LED_SHOW_IOCTL; @@ -849,7 +933,8 @@ } ledptrs[3]; void register_leds(int console, unsigned int led, - unsigned int *addr, unsigned int mask) { + unsigned int *addr, unsigned int mask) +{ struct kbd_struct *kbd = kbd_table + console; if (led < 3) { ledptrs[led].addr = addr; @@ -860,7 +945,8 @@ kbd->ledmode = LED_SHOW_FLAGS; } -static inline unsigned char getleds(void){ +static inline unsigned char getleds(void) +{ struct kbd_struct *kbd = kbd_table + fg_console; unsigned char leds; @@ -907,6 +993,19 @@ { unsigned char leds = getleds(); + if (rep && kbd_repeatkeycode != -1) { + tty = ttytab? ttytab[fg_console]: NULL; + kbd = kbd_table + fg_console; + + /* This prevents the kbd_key routine from being called + * twice, once by this BH, and once by the interrupt + * routine. + */ + kbd_disable_irq(); + kbd_processkeycode(kbd_repeatkeycode, 0, 1); + kbd_enable_irq(); + } + if (leds != ledstate) { ledstate = leds; kbd_leds(leds); @@ -917,6 +1016,12 @@ EXPORT_SYMBOL(keyboard_tasklet); DECLARE_TASKLET_DISABLED(keyboard_tasklet, kbd_bh, 0); +static void key_callback(unsigned long nr) +{ + rep = 1; + tasklet_schedule(&keyboard_tasklet); +} + typedef void (pm_kbd_func) (void); pm_callback pm_kbd_request_override = NULL; @@ -943,7 +1048,7 @@ tasklet_enable(&keyboard_tasklet); tasklet_schedule(&keyboard_tasklet); - + pm_kbd = pm_register(PM_SYS_DEV, PM_SYS_KBC, pm_kbd_request_override); return 0; diff -urN orig/drivers/char/mem.c linux/drivers/char/mem.c --- orig/drivers/char/mem.c Mon Aug 5 13:30:46 2002 +++ linux/drivers/char/mem.c Mon Aug 5 13:43:22 2002 @@ -26,9 +26,6 @@ #include #include -#ifdef CONFIG_I2C -extern int i2c_init_all(void); -#endif #ifdef CONFIG_FB extern void fbmem_init(void); #endif @@ -648,9 +645,6 @@ printk("unable to get major %d for memory devs\n", MEM_MAJOR); memory_devfs_register(); rand_initialize(); -#ifdef CONFIG_I2C - i2c_init_all(); -#endif #if defined (CONFIG_FB) fbmem_init(); #endif diff -urN orig/drivers/char/omaha-rtc.c linux/drivers/char/omaha-rtc.c --- orig/drivers/char/omaha-rtc.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/char/omaha-rtc.c Thu Oct 24 14:58:03 2002 @@ -0,0 +1,566 @@ +/* + * (C) ARM Limited 2002. + * + * Real Time Clock interface for Linux on Omaha + * + * Based on sa1100-rtc.c + * + * Copyright (c) 2000 Nils Faerber + * + * Based on rtc.c by Paul Gortmaker + * Date/time conversion routines taken from arch/arm/kernel/time.c + * by Linus Torvalds and Russell King + * and the GNU C Library + * ( ... I love the GPL ... just take what you need! ;) + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation; either version + * 2 of the License, or (at your option) any later version. + * + * 1.00 2001-06-08 Nicolas Pitre + * - added periodic timer capability using OSMR1 + * - flag compatibility with other RTC chips + * - permission checks for ioctls + * - major cleanup, partial rewrite + * + * 0.03 2001-03-07 CIH + * - Modify the bug setups RTC clock. + * + * 0.02 2001-02-27 Nils Faerber + * - removed mktime(), added alarm irq clear + * + * 0.01 2000-10-01 Nils Faerber + * - initial release + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#define DRIVER_VERSION "1.00" + +#define epoch 1970 + +#define TIMER_FREQ 3686400 + +#define RTC_DEF_DIVIDER 32768 - 1 +#define RTC_DEF_TRIM 0 + +/* Those are the bits from a classic RTC we want to mimic */ +#define RTC_IRQF 0x80 /* any of the following 3 is active */ +#define RTC_PF 0x40 +#define RTC_AF 0x20 +#define RTC_UF 0x10 + +// bitdefs for rtc registers +#define TICNT_ENABLE 0x80 // Enable tick interrupt +#define TICNT_PERIOD 0x7F // Divisor required for 1Hz tick +#define RTC_ENABLE 0x1 // Enable bit for RTC + +static unsigned long rtc_status; +static unsigned long rtc_irq_data; +static unsigned long rtc_freq = 1024; + +static struct fasync_struct *rtc_async_queue; +static DECLARE_WAIT_QUEUE_HEAD(rtc_wait); + +extern spinlock_t rtc_lock; + +static const unsigned char days_in_mo[] = + {31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31}; + +#define is_leap(year) \ + ((year) % 4 == 0 && ((year) % 100 != 0 || (year) % 400 == 0)) + +// all the alarm and rtc registers +static volatile unsigned int almsec = IO_ADDRESS(PLAT_PERIPHERAL_BASE+OMAHA_ALMSEC); +static volatile unsigned int almmin = IO_ADDRESS(PLAT_PERIPHERAL_BASE+OMAHA_ALMMIN); +static volatile unsigned int almhour = IO_ADDRESS(PLAT_PERIPHERAL_BASE+OMAHA_ALMHOUR); +static volatile unsigned int almday = IO_ADDRESS(PLAT_PERIPHERAL_BASE+OMAHA_ALMDAY); +static volatile unsigned int almmon = IO_ADDRESS(PLAT_PERIPHERAL_BASE+OMAHA_ALMMON); +static volatile unsigned int almyear = IO_ADDRESS(PLAT_PERIPHERAL_BASE+OMAHA_ALMYEAR); + +static volatile unsigned int bcdsec = IO_ADDRESS(PLAT_PERIPHERAL_BASE+OMAHA_BCDSEC); +static volatile unsigned int bcdmin = IO_ADDRESS(PLAT_PERIPHERAL_BASE+OMAHA_BCDMIN); +static volatile unsigned int bcdhour = IO_ADDRESS(PLAT_PERIPHERAL_BASE+OMAHA_BCDHOUR); +static volatile unsigned int bcdday = IO_ADDRESS(PLAT_PERIPHERAL_BASE+OMAHA_BCDDAY); +static volatile unsigned int bcddate = IO_ADDRESS(PLAT_PERIPHERAL_BASE+OMAHA_BCDDATE); +static volatile unsigned int bcdmon = IO_ADDRESS(PLAT_PERIPHERAL_BASE+OMAHA_BCDMON); +static volatile unsigned int bcdyear = IO_ADDRESS(PLAT_PERIPHERAL_BASE+OMAHA_BCDYEAR); + +/* + * Converts seconds since 1970-01-01 00:00:00 to Gregorian date. + */ + +static void decodetime (unsigned long t, struct rtc_time *tval) +{ + long days, month, year, rem; + + days = t / 86400; + rem = t % 86400; + tval->tm_hour = rem / 3600; + rem %= 3600; + tval->tm_min = rem / 60; + tval->tm_sec = rem % 60; + tval->tm_wday = (4 + days) % 7; + +#define LEAPS_THRU_END_OF(y) ((y)/4 - (y)/100 + (y)/400) + + year = epoch; + while (days >= (365 + is_leap(year))) { + unsigned long yg = year + days / 365; + days -= ((yg - year) * 365 + + LEAPS_THRU_END_OF (yg - 1) + - LEAPS_THRU_END_OF (year - 1)); + year = yg; + } + tval->tm_year = year - 1900; + tval->tm_yday = days + 1; + + month = 0; + if (days >= 31) { + days -= 31; + month++; + if (days >= (28 + is_leap(year))) { + days -= (28 + is_leap(year)); + month++; + while (days >= days_in_mo[month]) { + days -= days_in_mo[month]; + month++; + } + } + } + tval->tm_mon = month; + tval->tm_mday = days + 1; +} + +// Get alarm time in seconds +static unsigned long get_alarm_time(void) +{ + int sec, min,hour,date,mon,year; + + // Read data from h/w + year = __raw_readb(almyear); + mon = __raw_readb(almmon); + date = __raw_readb(almday); + hour = __raw_readb(almhour); + min = __raw_readb(almmin); + sec = __raw_readb(almsec); + + // convert all the data into binary + year = BCD_TO_BIN(year); + mon = BCD_TO_BIN(mon); + date = BCD_TO_BIN(date); + hour = BCD_TO_BIN(hour); + min = BCD_TO_BIN(min); + sec = BCD_TO_BIN(sec); + + // convert year to 19xx or 20xx as appropriate + if (year > 69) + year += 1900; + else + year += 2000; + + // Now calculate number of seconds since time began... + return mktime(year,mon,date,hour,min,sec); +} + +// Get rtc time in seconds +static unsigned long get_rtc_time(void) +{ + int sec,min,hour,day,date,mon,year; + + // Read data from h/w + year = __raw_readb(bcdyear); + mon = __raw_readb(bcdmon); + date = __raw_readb(bcdday); + day = __raw_readb(bcddate); + hour = __raw_readb(bcdhour); + min = __raw_readb(bcdmin); + sec = __raw_readb(bcdsec); + + // convert all the data into binary + year = BCD_TO_BIN(year); + mon = BCD_TO_BIN(mon); + date = BCD_TO_BIN(date); + day = BCD_TO_BIN(day); + hour = BCD_TO_BIN(hour); + min = BCD_TO_BIN(min); + sec = BCD_TO_BIN(sec); + + // convert year to 19xx or 20xx as appropriate + if (year > 69) + year += 1900; + else + year += 2000; + + // Now calculate number of seconds since time began... + return mktime(year,mon,date,hour,min,sec); +} + +/* Sets time of alarm */ +static void set_alarm_time(struct rtc_time *tval) +{ + + int sec,min,hour,day,mon,year; + + // Convert data from binary to 8-bit bcd + sec = BIN_TO_BCD(tval->tm_sec); + min = BIN_TO_BCD(tval->tm_min); + hour = BIN_TO_BCD(tval->tm_hour); + day = BIN_TO_BCD(tval->tm_mday); + mon = BIN_TO_BCD(tval->tm_mon); + + // Year is special + year = tval->tm_year; + if(year > 1999) + year -=2000; + else + year -=1900; + + year = BIN_TO_BCD(year); + + // Write all the registers + __raw_writeb(year,almyear); + __raw_writeb(mon,almmon); + __raw_writeb(day,almday); + __raw_writeb(hour,almhour); + __raw_writeb(min,almmin); + __raw_writeb(sec,almsec); +} + +/* Sets time of alarm */ +static void set_rtc_time(struct rtc_time *tval) +{ + + int sec,min,hour,day,date,mon,year; + + // Convert data from binary to 8-bit bcd + sec = BIN_TO_BCD(tval->tm_sec); + min = BIN_TO_BCD(tval->tm_min); + hour = BIN_TO_BCD(tval->tm_hour); + day = BIN_TO_BCD(tval->tm_mday); + date = BIN_TO_BCD(tval->tm_wday); + mon = BIN_TO_BCD(tval->tm_mon); + + // Year is special + year = tval->tm_year; + if(year > 1999) + year -=2000; + else + year -=1900; + + year = BIN_TO_BCD(year); + + // Write all the registers + __raw_writeb(year,bcdyear); + __raw_writeb(mon,bcdmon); + __raw_writeb(date,bcddate); + __raw_writeb(day,bcdday); + __raw_writeb(hour,bcdhour); + __raw_writeb(min,bcdmin); + __raw_writeb(sec,bcdsec); +} + +static void rtc_interrupt(int irq, void *dev_id, struct pt_regs *regs) +{ + /* update irq data & counter */ + rtc_irq_data += 0x100; + + /* wake up waiting process */ + wake_up_interruptible(&rtc_wait); + kill_fasync (&rtc_async_queue, SIGIO, POLL_IN); +} + +static int rtc_open(struct inode *inode, struct file *file) +{ + if (test_and_set_bit (1, &rtc_status)) + return -EBUSY; + MOD_INC_USE_COUNT; + rtc_irq_data = 0; + return 0; +} + +static int rtc_release(struct inode *inode, struct file *file) +{ + rtc_status = 0; + MOD_DEC_USE_COUNT; + return 0; +} + +static int rtc_fasync (int fd, struct file *filp, int on) +{ + return fasync_helper (fd, filp, on, &rtc_async_queue); +} + +static unsigned int rtc_poll(struct file *file, poll_table *wait) +{ + poll_wait (file, &rtc_wait, wait); + return (rtc_irq_data) ? 0 : POLLIN | POLLRDNORM; +} + +static loff_t rtc_llseek(struct file *file, loff_t offset, int origin) +{ + return -ESPIPE; +} + +ssize_t rtc_read(struct file *file, char *buf, size_t count, loff_t *ppos) +{ + DECLARE_WAITQUEUE(wait, current); + unsigned long data; + ssize_t retval; + + if (count < sizeof(unsigned long)) + return -EINVAL; + + add_wait_queue(&rtc_wait, &wait); + set_current_state(TASK_INTERRUPTIBLE); + for (;;) { + spin_lock_irq (&rtc_lock); + data = rtc_irq_data; + if (data != 0) { + rtc_irq_data = 0; + break; + } + spin_unlock_irq (&rtc_lock); + + if (file->f_flags & O_NONBLOCK) { + retval = -EAGAIN; + goto out; + } + + if (signal_pending(current)) { + retval = -ERESTARTSYS; + goto out; + } + + schedule(); + } + + spin_unlock_irq (&rtc_lock); + + data -= 0x100; /* the first IRQ wasn't actually missed */ + + retval = put_user(data, (unsigned long *)buf); + if (!retval) + retval = sizeof(unsigned long); + +out: + set_current_state(TASK_RUNNING); + remove_wait_queue(&rtc_wait, &wait); + return retval; +} + +static int rtc_ioctl(struct inode *inode, struct file *file, + unsigned int cmd, unsigned long arg) +{ + volatile unsigned int rtcalm = IO_ADDRESS(PLAT_PERIPHERAL_BASE+OMAHA_RTCALM); + volatile unsigned int ticnt = IO_ADDRESS(PLAT_PERIPHERAL_BASE+OMAHA_TICINT); + + struct rtc_time tm, tm2; + switch (cmd) { + case RTC_AIE_OFF: + spin_lock_irq(&rtc_lock); + __raw_writel(0,rtcalm); + rtc_irq_data = 0; + spin_unlock_irq(&rtc_lock); + return 0; + case RTC_AIE_ON: + spin_lock_irq(&rtc_lock); + __raw_writel(0x7F,rtcalm); + rtc_irq_data = 0; + spin_unlock_irq(&rtc_lock); + return 0; + case RTC_UIE_OFF: + spin_lock_irq(&rtc_lock); + __raw_writel(~TICNT_ENABLE,ticnt); + rtc_irq_data = 0; + spin_unlock_irq(&rtc_lock); + return 0; + case RTC_UIE_ON: + spin_lock_irq(&rtc_lock); + __raw_writel(TICNT_ENABLE|TICNT_PERIOD,ticnt); + rtc_irq_data = 0; + spin_unlock_irq(&rtc_lock); + return 0; + case RTC_PIE_OFF: + spin_lock_irq(&rtc_lock); + // Periodic int not available + rtc_irq_data = 0; + spin_unlock_irq(&rtc_lock); + return 0; + case RTC_PIE_ON: + spin_lock_irq(&rtc_lock); + // Periodic int not available + rtc_irq_data = 0; + spin_unlock_irq(&rtc_lock); + return 0; + case RTC_ALM_READ: + decodetime(get_alarm_time(),&tm); + break; + case RTC_ALM_SET: + if (copy_from_user (&tm2, (struct rtc_time*)arg, sizeof (tm2))) + return -EFAULT; + decodetime(get_rtc_time(),&tm); + if ((unsigned)tm2.tm_hour < 24) + tm.tm_hour = tm2.tm_hour; + if ((unsigned)tm2.tm_min < 60) + tm.tm_min = tm2.tm_min; + if ((unsigned)tm2.tm_sec < 60) + tm.tm_sec = tm2.tm_sec; + + // Munge (as per sa1100) + tm.tm_year+=1900; + tm.tm_mon+=1; + + // Set the alarm + set_alarm_time(&tm); + return 0; + case RTC_RD_TIME: + decodetime (get_rtc_time(), &tm); + break; + case RTC_SET_TIME: + if (!capable(CAP_SYS_TIME)) + return -EACCES; + if (copy_from_user (&tm, (struct rtc_time*)arg, sizeof (tm))) + return -EFAULT; + tm.tm_year += 1900; + if (tm.tm_year < epoch || (unsigned)tm.tm_mon >= 12 || + tm.tm_mday < 1 || tm.tm_mday > (days_in_mo[tm.tm_mon] + + (tm.tm_mon == 1 && is_leap(tm.tm_year))) || + (unsigned)tm.tm_hour >= 24 || + (unsigned)tm.tm_min >= 60 || + (unsigned)tm.tm_sec >= 60) + return -EINVAL; + tm.tm_mon +=1; // wierd: same as sa1100 though (gets month wrong otherwise!) + set_rtc_time(&tm); + return 0; + case RTC_IRQP_READ: + return put_user(rtc_freq, (unsigned long *)arg); + case RTC_IRQP_SET: + if (arg < 1 || arg > TIMER_FREQ) + return -EINVAL; + if ((arg > 64) && (!capable(CAP_SYS_RESOURCE))) + return -EACCES; + rtc_freq = arg; + return 0; + case RTC_EPOCH_READ: + return put_user (epoch, (unsigned long *)arg); + default: + return -EINVAL; + } + return copy_to_user ((void *)arg, &tm, sizeof (tm)) ? -EFAULT : 0; +} + +static struct file_operations rtc_fops = { + .owner = THIS_MODULE, + .llseek = rtc_llseek, + .read = rtc_read, + .poll = rtc_poll, + .ioctl = rtc_ioctl, + .open = rtc_open, + .release = rtc_release, + .fasync = rtc_fasync, +}; + +static struct miscdevice omahartc_miscdev = { + .minor = RTC_MINOR, + .name = "rtc", + .fops = &rtc_fops, +}; + +static int rtc_read_proc(char *page, char **start, off_t off, int count, int *eof, void *data) +{ + char *p = page; + int len; + struct rtc_time tm; + + decodetime (get_rtc_time(), &tm); + p += sprintf(p, "rtc_time\t: %02d:%02d:%02d\n" + "rtc_date\t: %04d-%02d-%02d\n" + "rtc_epoch\t: %04d\n", + tm.tm_hour, tm.tm_min, tm.tm_sec, + tm.tm_year + 1900, tm.tm_mon + 1, tm.tm_mday, epoch); + decodetime (get_alarm_time(), &tm); + p += sprintf(p, "alrm_time\t: %02d:%02d:%02d\n" + "alrm_date\t: %04d-%02d-%02d\n", + tm.tm_hour, tm.tm_min, tm.tm_sec, + tm.tm_year + 1900, tm.tm_mon + 1, tm.tm_mday); + p += sprintf(p, "periodic_freq\t: %ld\n", rtc_freq); + + len = (p - page) - off; + if (len < 0) + len = 0; + + *eof = (len <= count) ? 1 : 0; + *start = page + off; + + return len; +} + +static int __init rtc_init(void) +{ + volatile unsigned int ticnt = IO_ADDRESS(PLAT_PERIPHERAL_BASE+OMAHA_TICINT); + volatile unsigned int rtccon = IO_ADDRESS(PLAT_PERIPHERAL_BASE+OMAHA_RTCCON); + int ret; + + misc_register (&omahartc_miscdev); + create_proc_read_entry ("driver/rtc", 0, 0, rtc_read_proc, NULL); + + // Enable RTC + __raw_writel(RTC_ENABLE,rtccon); + + // Acquire 1Hz timer + ret = request_irq (OMAHA_INT_TICK, rtc_interrupt, SA_INTERRUPT, "rtc 1Hz", NULL); + if (ret) { + printk (KERN_ERR "rtc: IRQ %d already in use.\n", OMAHA_INT_TICK); + goto IRQ_TICK_failed; + } + + // Acquire RTC (Alarm interrupt) + ret = request_irq (OMAHA_INT_RTC, rtc_interrupt, SA_INTERRUPT, "rtc Alrm", NULL); + if (ret) { + printk (KERN_ERR "rtc: IRQ %d already in use.\n", OMAHA_INT_RTC); + goto IRQ_RTC_failed; + } + + printk (KERN_INFO "Omaha Real Time Clock driver v" DRIVER_VERSION "\n"); + + // Program tick interrupt divisor to generate real 1Hz clock and enable the interrupt + __raw_writeb(TICNT_ENABLE|TICNT_PERIOD,ticnt); + + return 0; + +IRQ_TICK_failed: + free_irq (OMAHA_INT_TICK, NULL); +IRQ_RTC_failed: + free_irq(OMAHA_INT_RTC, NULL); + remove_proc_entry ("driver/rtc", NULL); + misc_deregister (&omahartc_miscdev); + return ret; +} + +static void __exit rtc_exit(void) +{ + free_irq (OMAHA_INT_TICK, NULL); + remove_proc_entry ("driver/rtc", NULL); + misc_deregister (&omahartc_miscdev); +} + +module_init(rtc_init); +module_exit(rtc_exit); + +MODULE_AUTHOR("ARM Limited "); +MODULE_DESCRIPTION("Omaha Realtime Clock Driver (RTC)"); +EXPORT_NO_SYMBOLS; diff -urN orig/drivers/char/omaha_wdt.c linux/drivers/char/omaha_wdt.c --- orig/drivers/char/omaha_wdt.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/char/omaha_wdt.c Thu Oct 24 14:58:03 2002 @@ -0,0 +1,193 @@ +/* + * Watchdog driver for the Omaha + * (C) ARM Limited 2002. + * + * Based on sa1100_wdt.c + * + * (c) Copyright 2000 Oleg Drokin + * Based on SoftDog driver by Alan Cox + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation; either version + * 2 of the License, or (at your option) any later version. + * + * Neither Oleg Drokin nor iXcelerator.com admit liability nor provide + * warranty for any of this software. This material is provided + * "AS-IS" and at no charge. + * + * (c) Copyright 2000 Oleg Drokin + * + * 27/11/2000 Initial release + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#define TIMER_MARGIN 8 /* (secs) Default is 1 minute */ +#define WT_TPS 7812 /* Watchdog ticks per second. */ +#define WT_ENABLE 0x21 // Enable bits for watchdog +#define WT_CLKSEL_128 0x18 // Select 1/128 divider + +static int omaha_margin = TIMER_MARGIN; /* in seconds */ +static int omahawdt_users; +static int pre_margin; +#ifdef MODULE +MODULE_PARM(omaha_margin,"i"); +#endif + +/* + * Allow only one person to hold it open + */ + +static int omahadog_open(struct inode *inode, struct file *file) +{ + volatile unsigned int wtcon = IO_ADDRESS(PLAT_PERIPHERAL_BASE+OMAHA_WTCON); + volatile unsigned int wtdat = IO_ADDRESS(PLAT_PERIPHERAL_BASE+OMAHA_WTDAT); + volatile unsigned int wtcnt = IO_ADDRESS(PLAT_PERIPHERAL_BASE+OMAHA_WTCNT); + unsigned int tmp; + + if(test_and_set_bit(1,&omahawdt_users)) + return -EBUSY; + MOD_INC_USE_COUNT; + /* Activate omaha Watchdog timer */ + + /* Assume that uhal has set up pre-scaler according + * to the system clock frequency (don't change it!) + * + * Ie. all counting occurs at 1MHz / 128 = 7812.5Hz + * + * Since we have 16-bit counter, maximum period is + * 65536/7812.5 = 8.338608 seconds! + */ + + pre_margin = WT_TPS * omaha_margin; + + // Set count to the maximum + __raw_writel(pre_margin,wtcnt); + + // Set the clock division factor + tmp = __raw_readl(wtcon); + tmp |= WT_CLKSEL_128; + __raw_writel(tmp,wtcon); + + // Program an initial count into WTDAT + __raw_writel(0x8000,wtdat); + + // enable the watchdog + tmp = __raw_readl(wtcon); + tmp |= WT_ENABLE; + + __raw_writel(tmp,wtcon); + + return 0; +} + +static int omahadog_release(struct inode *inode, struct file *file) +{ + volatile unsigned int wtcon = IO_ADDRESS(PLAT_PERIPHERAL_BASE+OMAHA_WTCON); + unsigned int tmp; + + /* + * Shut off the timer. + * Lock it in if it's a module and we defined ...NOWAYOUT + */ +#ifndef CONFIG_WATCHDOG_NOWAYOUT + tmp = __raw_readl(wtcon); + tmp &= ~WT_ENABLE; + __raw_writel(tmp,wtcon); +#endif + omahawdt_users = 0; + MOD_DEC_USE_COUNT; + return 0; +} + +static ssize_t omahadog_write(struct file *file, const char *data, size_t len, loff_t *ppos) +{ + volatile unsigned int wtcnt = IO_ADDRESS(PLAT_PERIPHERAL_BASE+OMAHA_WTCNT); + + /* Can't seek (pwrite) on this device */ + if (ppos != &file->f_pos) + return -ESPIPE; + + /* Refresh timer. */ + if(len) { + __raw_writel(pre_margin,wtcnt); + return 1; + } + return 0; +} + +static int omahadog_ioctl(struct inode *inode, struct file *file, + unsigned int cmd, unsigned long arg) +{ + volatile unsigned int wtdat = IO_ADDRESS(PLAT_PERIPHERAL_BASE+OMAHA_WTDAT); + static struct watchdog_info ident = { + identity: "omaha Watchdog", + }; + + switch(cmd){ + default: + return -ENOIOCTLCMD; + case WDIOC_GETSUPPORT: + return copy_to_user((struct watchdog_info *)arg, &ident, sizeof(ident)); + case WDIOC_GETSTATUS: + return put_user(0,(int *)arg); + case WDIOC_GETBOOTSTATUS: + return 0; + case WDIOC_KEEPALIVE: + __raw_writel(pre_margin,wtdat); + return 0; + } +} + +static struct file_operations omahadog_fops= +{ + .owner = THIS_MODULE, + .write = omahadog_write, + .ioctl = omahadog_ioctl, + .open = omahadog_open, + .release = omahadog_release, +}; + +static struct miscdevice omahadog_miscdev= +{ + .minor = WATCHDOG_MINOR, + .name = "omaha watchdog", + .fops = &omahadog_fops +}; + +static int __init omahadog_init(void) +{ + int ret; + + ret = misc_register(&omahadog_miscdev); + + if (ret) + return ret; + + printk("Omaha Watchdog Timer: timer margin %d sec\n", omaha_margin); + + return 0; +} + +static void __exit omahadog_exit(void) +{ + misc_deregister(&omahadog_miscdev); +} + +module_init(omahadog_init); +module_exit(omahadog_exit); diff -urN orig/drivers/char/pcmcia/Config.in linux/drivers/char/pcmcia/Config.in --- orig/drivers/char/pcmcia/Config.in Sat Mar 31 23:47:20 2001 +++ linux/drivers/char/pcmcia/Config.in Thu Jul 19 15:42:45 2001 @@ -5,7 +5,13 @@ mainmenu_option next_comment comment 'PCMCIA character devices' -dep_tristate 'PCMCIA serial device support' CONFIG_PCMCIA_SERIAL_CS $CONFIG_SERIAL +if [ "$CONFIG_SERIAL" = "y" -o "$CONFIG_SERIAL_8250" = "y" ]; then + dep_tristate 'PCMCIA serial device support' CONFIG_PCMCIA_SERIAL_CS y +else + if [ "$CONFIG_SERIAL" = "m" -o "$CONFIG_SERIAL_8250" = "m" ]; then + dep_tristate 'PCMCIA serial device support' CONFIG_PCMCIA_SERIAL_CS m + fi +fi if [ "$CONFIG_PCMCIA_SERIAL_CS" = "y" ]; then define_bool CONFIG_PCMCIA_CHRDEV y fi diff -urN orig/drivers/char/pcmcia/memory_cs.c linux/drivers/char/pcmcia/memory_cs.c --- orig/drivers/char/pcmcia/memory_cs.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/char/pcmcia/memory_cs.c Thu Sep 13 13:31:26 2001 @@ -0,0 +1,214 @@ +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include + +#include +#include + +static dev_info_t dev_info = "memory_cs"; +static dev_link_t *dev_list; + +struct memcs_dev { + dev_link_t link; + struct map_info map; +}; + +static __u8 mem_cs_read8(struct map_info *map, unsigned long ofs) +{ + return readb(map->map_priv_1 + ofs); +} + +static __u16 mem_cs_read16(struct map_info *map, unsigned long ofs) +{ + return readw(map->map_priv_1 + ofs); +} + +static __u32 mem_cs_read32(struct map_info *map, unsigned long ofs) +{ + return readl(map->map_priv_1 + ofs); +} + +static void mem_cs_copy_from(struct map_info *map, void *to, unsigned long ofs, ssize_t size) +{ + memcpy_fromio(to, map->map_priv_1 + ofs, size); +} + +static void mem_cs_write8(struct map_info *map, __u8 val, unsigned long ofs) +{ + writeb(val, map->map_priv_1 + ofs); +} + +static void mem_cs_write16(struct map_info *map, __u16 val, unsigned long ofs) +{ + writew(val, map->map_priv_1 + ofs); +} + +static void mem_cs_write32(struct map_info *map, __u32 val, unsigned long ofs) +{ + writel(val, map->map_priv_1 + ofs); +} + +static void mem_cs_copy_to(struct map_info *map, unsigned long ofs, const void *to, ssize_t size) +{ + memcpy_toio(map->map_priv_1 + ofs, from, size); +} + +static void mem_cs_release(u_long arg); + +static void mem_cs_detach(dev_link_t *link) +{ + del_timer(&link->release); + if (link->state & DEV_CONFIG) { + mem_cs_release((u_long)link); + if (link->state & DEV_STALE_CONFIG) { + link->state |= DEV_STALE_LINK; + return; + } + } + + if (link->handle) + CardServices(DeregisterClient, link->handle); + + kfree(link); +} + +static void mem_cs_release(u_long arg) +{ + dev_link_t *link = (dev_link_t *)arg; + + link->dev = NULL; + if (link->win) { + CardServices(ReleaseWindow, link->win); + } + link->state &= ~DEV_CONFIG; + + if (link->state & DEV_STALE_LINK) + mem_cs_detach(link); +} + +static void mem_cs_config(dev_link_t *link) +{ + struct memcs_dev *dev = link->priv; + cs_status_t status; + win_req_t req; + + link->state |= DEV_CONFIG; + + req.Attributes = word_width ? WIN_DATA_WIDTH_16 : WIN_DATA_WIDTH_8; + req.Base = 0; + req.Size = 0; + req.AccessSpeed = mem_speed; + + link->win = (window_handle_t)link->handle; + + CS_CHECK(RequestWindow, &link->win, &req); + + CS_CHECK(GetStatus, link->handle, &status); + + dev->map.buswidth = word_width ? 2 : 1; + dev->map.size = req.Size; + dev->map.map_priv_1 = ioremap(req.Base, req.Size); +} + +static int +mem_cs_event(event_t event, int priority, event_callback_args_t *args) +{ + dev_link_t *link = args->client_data; + + switch (event) { + case CS_EVENT_CARD_REMOVAL: + link->state &= ~DEV_PRESENT; + if (link->state & DEV_CONFIG) + mod_timer(&link->release, jiffies + HZ/20); + break; + + case CS_EVENT_CARD_INSERTION: + link->state |= DEV_PRESENT | DEV_CONFIG_PENDING; + mem_cs_config(link); + break; + } + return 0; +} + +static dev_link_t *mem_cs_attach(void) +{ + struct memcs_dev *dev; + client_reg_t clnt; + + dev = kmalloc(sizeof(*dev), GFP_KERNEL); + if (dev) { + memset(dev, 0, sizeof(*dev)); + + dev->map.read8 = mem_cs_read8; + dev->map.read16 = mem_cs_read16; + dev->map.read32 = mem_cs_read32; + dev->map.copy_from = mem_cs_copy_from; + dev->map.write8 = mem_cs_write8; + dev->map.write16 = mem_cs_write16; + dev->map.write32 = mem_cs_write32; + dev->map.copy_to = mem_cs_copy_to; + + dev->link.release.function = &mem_cs_release; + dev->link.release.data = (u_long)link; + dev->link.priv = dev; + + dev->link.next = dev_list; + dev_list = &dev->link; + + clnt.dev_info = &dev_info; + clnt.Attributes = INOF_IO_CLIENT | INFO_CARD_SHARE; + clnt.EventMask = + CS_EVENT_WRITE_PROTECT | CS_EVENT_CARD_INSERTION | + CS_EVENT_CARD_REMOVAL | CS_EVENT_BATTERY_DEAD | + CS_EVENT_BATTERY_LOW; + + clnt.event_handler = &mem_cs_event; + clnt.Version = 0x0210; + clnt.event_callback_args.client_data = &dev->link; + + ret = CardServices(RegisterClient, &dev->link.handle, &clnt); + if (ret != CS_SUCCESS) { + error_info_t err = { RegisterClient, ret }; + CardServices(ReportError, dev->link.handle, &err); + mem_cs_detach(&dev->link); + dev = NULL; + } + } + + return &dev->link; +} + +static int __init mem_cs_init(void) +{ + servinfo_t serv; + + CardServices(GetCardServicesInfo, &serv); + if (serv.Revision != CS_RELEASE_CODE) { + printk(KERN_NOTICE "memory_cs: Card services release " + "does not match\n"); + return -ENXIO; + } + register_pccard_driver(&dev_info, mem_cs_attach, mem_cs_detach); + return 0; +} + +static void __exit mem_cs_exit(void) +{ + unregister_pccard_driver(&dev_info); + while (dev_list != NULL) + memory_detach(dev_list); +} + +module_init(mem_cs_init); +module_exit(mem_cs_exit); + +MODULE_LICENSE("GPL"); diff -urN orig/drivers/char/sa1100-rtc.c linux/drivers/char/sa1100-rtc.c --- orig/drivers/char/sa1100-rtc.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/char/sa1100-rtc.c Thu Jan 3 22:06:31 2002 @@ -0,0 +1,474 @@ +/* + * Real Time Clock interface for Linux on StrongARM SA1100 + * + * Copyright (c) 2000 Nils Faerber + * + * Based on rtc.c by Paul Gortmaker + * Date/time conversion routines taken from arch/arm/kernel/time.c + * by Linus Torvalds and Russel King + * and the GNU C Library + * ( ... I love the GPL ... just take what you need! ;) + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation; either version + * 2 of the License, or (at your option) any later version. + * + * 1.00 2001-06-08 Nicolas Pitre + * - added periodic timer capability using OSMR1 + * - flag compatibility with other RTC chips + * - permission checks for ioctls + * - major cleanup, partial rewrite + * + * 0.03 2001-03-07 CIH + * - Modify the bug setups RTC clock. + * + * 0.02 2001-02-27 Nils Faerber + * - removed mktime(), added alarm irq clear + * + * 0.01 2000-10-01 Nils Faerber + * - initial release + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#define DRIVER_VERSION "1.00" + +#define TIMER_FREQ 3686400 + +#define RTC_DEF_DIVIDER 32768 - 1 +#define RTC_DEF_TRIM 0 + +/* Those are the bits from a classic RTC we want to mimic */ +#define RTC_IRQF 0x80 /* any of the following 3 is active */ +#define RTC_PF 0x40 +#define RTC_AF 0x20 +#define RTC_UF 0x10 + +static unsigned long rtc_status; +static unsigned long rtc_irq_data; +static unsigned long rtc_freq = 1024; + +static struct fasync_struct *rtc_async_queue; +static DECLARE_WAIT_QUEUE_HEAD(rtc_wait); + +extern spinlock_t rtc_lock; + +static const unsigned char days_in_mo[] = + {31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31}; + +#define is_leap(year) \ + ((year) % 4 == 0 && ((year) % 100 != 0 || (year) % 400 == 0)) + +/* + * Converts seconds since 1970-01-01 00:00:00 to Gregorian date. + */ + +static void decodetime (unsigned long t, struct rtc_time *tval) +{ + long days, month, year, rem; + + days = t / 86400; + rem = t % 86400; + tval->tm_hour = rem / 3600; + rem %= 3600; + tval->tm_min = rem / 60; + tval->tm_sec = rem % 60; + tval->tm_wday = (4 + days) % 7; + +#define LEAPS_THRU_END_OF(y) ((y)/4 - (y)/100 + (y)/400) + + year = 1970 + days / 365; + days -= ((year - 1970) * 365 + + LEAPS_THRU_END_OF (year - 1) + - LEAPS_THRU_END_OF (1970 - 1)); + if (days < 0) { + year -= 1; + days += 365 + is_leap(year); + } + tval->tm_year = year - 1900; + tval->tm_yday = days + 1; + + month = 0; + if (days >= 31) { + days -= 31; + month++; + if (days >= (28 + is_leap(year))) { + days -= (28 + is_leap(year)); + month++; + while (days >= days_in_mo[month]) { + days -= days_in_mo[month]; + month++; + } + } + } + tval->tm_mon = month; + tval->tm_mday = days + 1; +} + +static void rtc_interrupt(int irq, void *dev_id, struct pt_regs *regs) +{ + unsigned int rtsr = RTSR; + + /* clear interrupt sources */ + RTSR = 0; + RTSR = (RTSR_AL|RTSR_HZ); + + /* clear alarm interrupt if it has occurred */ + if (rtsr & RTSR_AL) + rtsr &= ~RTSR_ALE; + RTSR = rtsr & (RTSR_ALE|RTSR_HZE); + + /* update irq data & counter */ + if (rtsr & RTSR_AL) + rtc_irq_data |= (RTC_AF|RTC_IRQF); + if (rtsr & RTSR_HZ) + rtc_irq_data |= (RTC_UF|RTC_IRQF); + rtc_irq_data += 0x100; + + /* wake up waiting process */ + wake_up_interruptible(&rtc_wait); + kill_fasync (&rtc_async_queue, SIGIO, POLL_IN); +} + +static void timer1_interrupt(int irq, void *dev_id, struct pt_regs *regs) +{ + /* + * If we match for the first time, the periodic interrupt flag won't + * be set. If it is, then we did wrap around (very unlikely but + * still possible) and compute the amount of missed periods. + * The match reg is updated only when the data is actually retrieved + * to avoid unnecessary interrupts. + */ + OSSR = OSSR_M1; /* clear match on timer1 */ + if (rtc_irq_data & RTC_PF) { + rtc_irq_data += (rtc_freq * ((1<<30)/(TIMER_FREQ>>2))) << 8; + } else { + rtc_irq_data += (0x100|RTC_PF|RTC_IRQF); + } + + wake_up_interruptible(&rtc_wait); + kill_fasync (&rtc_async_queue, SIGIO, POLL_IN); +} + +static int rtc_open(struct inode *inode, struct file *file) +{ + if (test_and_set_bit (1, &rtc_status)) + return -EBUSY; + rtc_irq_data = 0; + return 0; +} + +static int rtc_release(struct inode *inode, struct file *file) +{ + spin_lock_irq (&rtc_lock); + RTSR = 0; + RTSR = (RTSR_AL|RTSR_HZ); + OIER &= ~OIER_E1; + OSSR = OSSR_M1; + spin_unlock_irq (&rtc_lock); + rtc_status = 0; + return 0; +} + +static int rtc_fasync (int fd, struct file *filp, int on) +{ + return fasync_helper (fd, filp, on, &rtc_async_queue); +} + +static unsigned int rtc_poll(struct file *file, poll_table *wait) +{ + poll_wait (file, &rtc_wait, wait); + return (rtc_irq_data) ? 0 : POLLIN | POLLRDNORM; +} + +static loff_t rtc_llseek(struct file *file, loff_t offset, int origin) +{ + return -ESPIPE; +} + +ssize_t rtc_read(struct file *file, char *buf, size_t count, loff_t *ppos) +{ + DECLARE_WAITQUEUE(wait, current); + unsigned long data; + ssize_t retval; + + if (count < sizeof(unsigned long)) + return -EINVAL; + + add_wait_queue(&rtc_wait, &wait); + set_current_state(TASK_INTERRUPTIBLE); + for (;;) { + spin_lock_irq (&rtc_lock); + data = rtc_irq_data; + if (data != 0) { + rtc_irq_data = 0; + break; + } + spin_unlock_irq (&rtc_lock); + + if (file->f_flags & O_NONBLOCK) { + retval = -EAGAIN; + goto out; + } + + if (signal_pending(current)) { + retval = -ERESTARTSYS; + goto out; + } + + schedule(); + } + + if (data & RTC_PF) { + /* interpolate missed periods and set match for the next one */ + unsigned long period = TIMER_FREQ/rtc_freq; + unsigned long oscr = OSCR; + unsigned long osmr1 = OSMR1; + unsigned long missed = (oscr - osmr1)/period; + data += missed << 8; + OSSR = OSSR_M1; /* clear match on timer 1 */ + OSMR1 = osmr1 + (missed + 1)*period; + /* ensure we didn't miss another match in the mean time */ + while( (signed long)((osmr1 = OSMR1) - OSCR) <= 0 ) { + data += 0x100; + OSSR = OSSR_M1; /* clear match on timer 1 */ + OSMR1 = osmr1 + period; + } + } + spin_unlock_irq (&rtc_lock); + + data -= 0x100; /* the first IRQ wasn't actually missed */ + + retval = put_user(data, (unsigned long *)buf); + if (!retval) + retval = sizeof(unsigned long); + +out: + set_current_state(TASK_RUNNING); + remove_wait_queue(&rtc_wait, &wait); + return retval; +} + +static int rtc_ioctl(struct inode *inode, struct file *file, + unsigned int cmd, unsigned long arg) +{ + struct rtc_time tm, tm2; + + switch (cmd) { + case RTC_AIE_OFF: + spin_lock_irq(&rtc_lock); + RTSR &= ~RTSR_ALE; + rtc_irq_data = 0; + spin_unlock_irq(&rtc_lock); + return 0; + case RTC_AIE_ON: + spin_lock_irq(&rtc_lock); + RTSR |= RTSR_ALE; + rtc_irq_data = 0; + spin_unlock_irq(&rtc_lock); + return 0; + case RTC_UIE_OFF: + spin_lock_irq(&rtc_lock); + RTSR &= ~RTSR_HZE; + rtc_irq_data = 0; + spin_unlock_irq(&rtc_lock); + return 0; + case RTC_UIE_ON: + spin_lock_irq(&rtc_lock); + RTSR |= RTSR_HZE; + rtc_irq_data = 0; + spin_unlock_irq(&rtc_lock); + return 0; + case RTC_PIE_OFF: + spin_lock_irq(&rtc_lock); + OIER &= ~OIER_E1; + rtc_irq_data = 0; + spin_unlock_irq(&rtc_lock); + return 0; + case RTC_PIE_ON: + if ((rtc_freq > 64) && !capable(CAP_SYS_RESOURCE)) + return -EACCES; + spin_lock_irq(&rtc_lock); + OSMR1 = TIMER_FREQ/rtc_freq + OSCR; + OIER |= OIER_E1; + rtc_irq_data = 0; + spin_unlock_irq(&rtc_lock); + return 0; + case RTC_ALM_READ: + decodetime (RTAR, &tm); + break; + case RTC_ALM_SET: + if (copy_from_user (&tm2, (struct rtc_time*)arg, sizeof (tm2))) + return -EFAULT; + decodetime (RCNR, &tm); + if ((unsigned)tm2.tm_hour < 24) + tm.tm_hour = tm2.tm_hour; + if ((unsigned)tm2.tm_min < 60) + tm.tm_min = tm2.tm_min; + if ((unsigned)tm2.tm_sec < 60) + tm.tm_sec = tm2.tm_sec; + RTAR = mktime ( tm.tm_year + 1900, tm.tm_mon + 1, tm.tm_mday, + tm.tm_hour, tm.tm_min, tm.tm_sec); + return 0; + case RTC_RD_TIME: + decodetime (RCNR, &tm); + break; + case RTC_SET_TIME: + if (!capable(CAP_SYS_TIME)) + return -EACCES; + if (copy_from_user (&tm, (struct rtc_time*)arg, sizeof (tm))) + return -EFAULT; + tm.tm_year += 1900; + if (tm.tm_year < 1970 || (unsigned)tm.tm_mon >= 12 || + tm.tm_mday < 1 || tm.tm_mday > (days_in_mo[tm.tm_mon] + + (tm.tm_mon == 1 && is_leap(tm.tm_year))) || + (unsigned)tm.tm_hour >= 24 || + (unsigned)tm.tm_min >= 60 || + (unsigned)tm.tm_sec >= 60) + return -EINVAL; + RCNR = mktime ( tm.tm_year, tm.tm_mon + 1, tm.tm_mday, + tm.tm_hour, tm.tm_min, tm.tm_sec); + return 0; + case RTC_IRQP_READ: + return put_user(rtc_freq, (unsigned long *)arg); + case RTC_IRQP_SET: + if (arg < 1 || arg > TIMER_FREQ) + return -EINVAL; + if ((arg > 64) && (!capable(CAP_SYS_RESOURCE))) + return -EACCES; + rtc_freq = arg; + return 0; + case RTC_EPOCH_READ: + return put_user (1970, (unsigned long *)arg); + default: + return -EINVAL; + } + return copy_to_user ((void *)arg, &tm, sizeof (tm)) ? -EFAULT : 0; +} + +static struct file_operations rtc_fops = { + owner: THIS_MODULE, + llseek: rtc_llseek, + read: rtc_read, + poll: rtc_poll, + ioctl: rtc_ioctl, + open: rtc_open, + release: rtc_release, + fasync: rtc_fasync, +}; + +static struct miscdevice sa1100rtc_miscdev = { + RTC_MINOR, + "rtc", + &rtc_fops +}; + +static int rtc_read_proc(char *page, char **start, off_t off, int count, int *eof, void *data) +{ + char *p = page; + int len; + struct rtc_time tm; + + decodetime (RCNR, &tm); + p += sprintf(p, "rtc_time\t: %02d:%02d:%02d\n" + "rtc_date\t: %04d-%02d-%02d\n" + "rtc_epoch\t: %04d\n", + tm.tm_hour, tm.tm_min, tm.tm_sec, + tm.tm_year + 1900, tm.tm_mon + 1, tm.tm_mday, 1970); + decodetime (RTAR, &tm); + p += sprintf(p, "alrm_time\t: %02d:%02d:%02d\n" + "alrm_date\t: %04d-%02d-%02d\n", + tm.tm_hour, tm.tm_min, tm.tm_sec, + tm.tm_year + 1900, tm.tm_mon + 1, tm.tm_mday); + p += sprintf(p, "trim/divider\t: 0x%08x\n", RTTR); + p += sprintf(p, "alarm_IRQ\t: %s\n", (RTSR & RTSR_ALE) ? "yes" : "no" ); + p += sprintf(p, "update_IRQ\t: %s\n", (RTSR & RTSR_HZE) ? "yes" : "no"); + p += sprintf(p, "periodic_IRQ\t: %s\n", (OIER & OIER_E1) ? "yes" : "no"); + p += sprintf(p, "periodic_freq\t: %ld\n", rtc_freq); + + len = (p - page) - off; + if (len < 0) + len = 0; + + *eof = (len <= count) ? 1 : 0; + *start = page + off; + + return len; +} + +static int __init rtc_init(void) +{ + int ret; + + misc_register (&sa1100rtc_miscdev); + create_proc_read_entry ("driver/rtc", 0, 0, rtc_read_proc, NULL); + ret = request_irq (IRQ_RTC1Hz, rtc_interrupt, SA_INTERRUPT, "rtc 1Hz", NULL); + if (ret) { + printk (KERN_ERR "rtc: IRQ %d already in use.\n", IRQ_RTC1Hz); + goto IRQ_RTC1Hz_failed; + } + ret = request_irq (IRQ_RTCAlrm, rtc_interrupt, SA_INTERRUPT, "rtc Alrm", NULL); + if (ret) { + printk(KERN_ERR "rtc: IRQ %d already in use.\n", IRQ_RTCAlrm); + goto IRQ_RTCAlrm_failed; + } + ret = request_irq (IRQ_OST1, timer1_interrupt, SA_INTERRUPT, "rtc timer", NULL); + if (ret) { + printk(KERN_ERR "rtc: IRQ %d already in use.\n", IRQ_OST1); + goto IRQ_OST1_failed; + } + + printk (KERN_INFO "SA1100 Real Time Clock driver v" DRIVER_VERSION "\n"); + + /* + * According to the manual we should be able to let RTTR be zero + * and then a default diviser for a 32.768KHz clock is used. + * Apparently this doesn't work, at least for my SA1110 rev 5. + * If the clock divider is uninitialized then reset it to the + * default value to get the 1Hz clock. + */ + if (RTTR == 0) { + RTTR = RTC_DEF_DIVIDER + (RTC_DEF_TRIM << 16); + printk (KERN_WARNING "rtc: warning: initializing default clock divider/trim value\n"); + /* The current RTC value probably doesn't make sense either */ + RCNR = 0; + } + + return 0; + +IRQ_OST1_failed: + free_irq (IRQ_RTCAlrm, NULL); +IRQ_RTCAlrm_failed: + free_irq (IRQ_RTC1Hz, NULL); +IRQ_RTC1Hz_failed: + remove_proc_entry ("driver/rtc", NULL); + misc_deregister (&sa1100rtc_miscdev); + return ret; +} + +static void __exit rtc_exit(void) +{ + free_irq (IRQ_OST1, NULL); + free_irq (IRQ_RTCAlrm, NULL); + free_irq (IRQ_RTC1Hz, NULL); + remove_proc_entry ("driver/rtc", NULL); + misc_deregister (&sa1100rtc_miscdev); +} + +module_init(rtc_init); +module_exit(rtc_exit); + +MODULE_AUTHOR("Nils Faerber "); +MODULE_DESCRIPTION("SA1100 Realtime Clock Driver (RTC)"); +EXPORT_NO_SYMBOLS; diff -urN orig/drivers/char/sa1100_wdt.c linux/drivers/char/sa1100_wdt.c --- orig/drivers/char/sa1100_wdt.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/char/sa1100_wdt.c Sat Jun 23 11:37:42 2001 @@ -0,0 +1,150 @@ +/* + * Watchdog driver for the SA11x0 + * + * (c) Copyright 2000 Oleg Drokin + * Based on SoftDog driver by Alan Cox + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation; either version + * 2 of the License, or (at your option) any later version. + * + * Neither Oleg Drokin nor iXcelerator.com admit liability nor provide + * warranty for any of this software. This material is provided + * "AS-IS" and at no charge. + * + * (c) Copyright 2000 Oleg Drokin + * + * 27/11/2000 Initial release + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#define TIMER_MARGIN 60 /* (secs) Default is 1 minute */ + +static int sa1100_margin = TIMER_MARGIN; /* in seconds */ +static int sa1100wdt_users; +static int pre_margin; +#ifdef MODULE +MODULE_PARM(sa1100_margin,"i"); +#endif + +/* + * Allow only one person to hold it open + */ + +static int sa1100dog_open(struct inode *inode, struct file *file) +{ + if(test_and_set_bit(1,&sa1100wdt_users)) + return -EBUSY; + MOD_INC_USE_COUNT; + /* Activate SA1100 Watchdog timer */ + pre_margin=3686400 * sa1100_margin; + OSMR3 = OSCR + pre_margin; + OSSR = OSSR_M3; + OWER = OWER_WME; + OIER |= OIER_E3; + return 0; +} + +static int sa1100dog_release(struct inode *inode, struct file *file) +{ + /* + * Shut off the timer. + * Lock it in if it's a module and we defined ...NOWAYOUT + */ + OSMR3 = OSCR + pre_margin; +#ifndef CONFIG_WATCHDOG_NOWAYOUT + OIER &= ~OIER_E3; +#endif + sa1100wdt_users = 0; + MOD_DEC_USE_COUNT; + return 0; +} + +static ssize_t sa1100dog_write(struct file *file, const char *data, size_t len, loff_t *ppos) +{ + /* Can't seek (pwrite) on this device */ + if (ppos != &file->f_pos) + return -ESPIPE; + + /* Refresh OSMR3 timer. */ + if(len) { + OSMR3 = OSCR + pre_margin; + return 1; + } + return 0; +} + +static int sa1100dog_ioctl(struct inode *inode, struct file *file, + unsigned int cmd, unsigned long arg) +{ + static struct watchdog_info ident = { + identity: "SA1100 Watchdog", + }; + + switch(cmd){ + default: + return -ENOIOCTLCMD; + case WDIOC_GETSUPPORT: + return copy_to_user((struct watchdog_info *)arg, &ident, sizeof(ident)); + case WDIOC_GETSTATUS: + return put_user(0,(int *)arg); + case WDIOC_GETBOOTSTATUS: + return put_user((RCSR & RCSR_WDR) ? WDIOF_CARDRESET : 0, (int *)arg); + case WDIOC_KEEPALIVE: + OSMR3 = OSCR + pre_margin; + return 0; + } +} + +static struct file_operations sa1100dog_fops= +{ + owner: THIS_MODULE, + write: sa1100dog_write, + ioctl: sa1100dog_ioctl, + open: sa1100dog_open, + release: sa1100dog_release, +}; + +static struct miscdevice sa1100dog_miscdev= +{ + WATCHDOG_MINOR, + "SA1100 watchdog", + &sa1100dog_fops +}; + +static int __init sa1100dog_init(void) +{ + int ret; + + ret = misc_register(&sa1100dog_miscdev); + + if (ret) + return ret; + + printk("SA1100 Watchdog Timer: timer margin %d sec\n", sa1100_margin); + + return 0; +} + +static void __exit sa1100dog_exit(void) +{ + misc_deregister(&sa1100dog_miscdev); +} + +module_init(sa1100dog_init); +module_exit(sa1100dog_exit); diff -urN orig/drivers/char/sa1111_keyb.c linux/drivers/char/sa1111_keyb.c --- orig/drivers/char/sa1111_keyb.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/char/sa1111_keyb.c Thu Jan 24 10:50:20 2002 @@ -0,0 +1,1153 @@ +/* + * SA1111 PS/2 keyboard/mouse driver + * + * 2000 by VASARA RESEARCH INC. + * + * Changelog: + * Jun.xx,2000: Kunihiko IMAI + * Port to 2.4.0test1-ac19-rmk1-np1 + * Apr.17,2000: Takafumi Kawana + * Internal Release for XP860 + * + * + * This driver is based on linux/drivers/char/pc_keyb.c + * Original declaration follows: + + * + * linux/drivers/char/pc_keyb.c + * + * Separation of the PC low-level part by Geert Uytterhoeven, May 1997 + * See keyboard.c for the whole history. + * + * Major cleanup by Martin Mares, May 1997 + * + * Combined the keyboard and PS/2 mouse handling into one file, + * because they share the same hardware. + * Johan Myreen 1998-10-08. + * + * Code fixes to handle mouse ACKs properly. + * C. Scott Ananian 1999-01-29. + * + */ +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include + +#include + +/* Some configuration switches are present in the include file... */ + +#include +#include +#include + +#define KBD_STAT_RXB (1<<4) +#define KBD_STAT_RXF (1<<5) +#define KBD_STAT_TXB (1<<6) +#define KBD_STAT_TXE (1<<7) +#define KBD_STAT_STP (1<<8) + +#define MSE_STAT_RXB (1<<4) +#define MSE_STAT_RXF (1<<5) +#define MSE_STAT_TXB (1<<6) +#define MSE_STAT_TXE (1<<7) +#define MSE_STAT_STP (1<<8) + +/* Simple translation table for the SysRq keys */ + +#ifdef CONFIG_MAGIC_SYSRQ +unsigned char sa1111_sysrq_xlate[128] = "\000\0331234567890-=\177\t" /* 0x00 - 0x0f */ + "qwertyuiop[]\r\000as" /* 0x10 - 0x1f */ + "dfghjkl;'`\000\\zxcv" /* 0x20 - 0x2f */ + "bnm,./\000*\000 \000\201\202\203\204\205" /* 0x30 - 0x3f */ + "\206\207\210\211\212\000\000789-456+1" /* 0x40 - 0x4f */ + "230\177\000\000\213\214\000\000\000\000\000\000\000\000\000\000" /* 0x50 - 0x5f */ + "\r\000/"; /* 0x60 - 0x6f */ +#endif + +// static void kbd_write_command_w(int data); +static void kbd_write_output_w(int data); + +spinlock_t kbd_controller_lock = SPIN_LOCK_UNLOCKED; +static void handle_kbd_event(void); + +/* used only by send_data - set by keyboard_interrupt */ +static volatile unsigned char reply_expected = 0; +static volatile unsigned char acknowledge = 0; +static volatile unsigned char resend = 0; + + +#if defined CONFIG_PSMOUSE +/* + * PS/2 Auxiliary Device + */ + +static int __init psaux_init(void); + +static struct aux_queue *queue; /* Mouse data buffer. */ +static int aux_count = 0; +/* used when we send commands to the mouse that expect an ACK. */ +static unsigned char mouse_reply_expected = 0; + +#define AUX_INTS_OFF (KBD_MODE_KCC | KBD_MODE_DISABLE_MOUSE | KBD_MODE_SYS | KBD_MODE_KBD_INT) +#define AUX_INTS_ON (KBD_MODE_KCC | KBD_MODE_SYS | KBD_MODE_MOUSE_INT | KBD_MODE_KBD_INT) + +#define MAX_RETRIES 60 /* some aux operations take long time */ +#endif /* CONFIG_PSMOUSE */ + +/* + * Wait for keyboard controller input buffer to drain. + * + * Don't use 'jiffies' so that we don't depend on + * interrupts.. + * + * Quote from PS/2 System Reference Manual: + * + * "Address hex 0060 and address hex 0064 should be written only when + * the input-buffer-full bit and output-buffer-full bit in the + * Controller Status register are set 0." + */ + +static void kb_wait(void) +{ + unsigned long timeout = KBC_TIMEOUT; + + do { + /* + * "handle_kbd_event()" will handle any incoming events + * while we wait - keypresses or mouse movement. + */ + handle_kbd_event(); + if (KBDSTAT & KBD_STAT_TXE) + return; + mdelay(1); + timeout--; + } + while (timeout); +#ifdef KBD_REPORT_TIMEOUTS + printk(KERN_WARNING "Keyboard timed out[1]\n"); +#endif +} + +/* + * Translation of escaped scancodes to keycodes. + * This is now user-settable. + * The keycodes 1-88,96-111,119 are fairly standard, and + * should probably not be changed - changing might confuse X. + * X also interprets scancode 0x5d (KEY_Begin). + * + * For 1-88 keycode equals scancode. + */ + +#define E0_KPENTER 96 +#define E0_RCTRL 97 +#define E0_KPSLASH 98 +#define E0_PRSCR 99 +#define E0_RALT 100 +#define E0_BREAK 101 /* (control-pause) */ +#define E0_HOME 102 +#define E0_UP 103 +#define E0_PGUP 104 +#define E0_LEFT 105 +#define E0_RIGHT 106 +#define E0_END 107 +#define E0_DOWN 108 +#define E0_PGDN 109 +#define E0_INS 110 +#define E0_DEL 111 + +/* for USB 106 keyboard */ +#define E0_YEN 124 +#define E0_BACKSLASH 89 + + +#define E1_PAUSE 119 + +/* + * The keycodes below are randomly located in 89-95,112-118,120-127. + * They could be thrown away (and all occurrences below replaced by 0), + * but that would force many users to use the `setkeycodes' utility, where + * they needed not before. It does not matter that there are duplicates, as + * long as no duplication occurs for any single keyboard. + */ +#define SC_LIM 89 + +#define FOCUS_PF1 85 /* actual code! */ +#define FOCUS_PF2 89 +#define FOCUS_PF3 90 +#define FOCUS_PF4 91 +#define FOCUS_PF5 92 +#define FOCUS_PF6 93 +#define FOCUS_PF7 94 +#define FOCUS_PF8 95 +#define FOCUS_PF9 120 +#define FOCUS_PF10 121 +#define FOCUS_PF11 122 +#define FOCUS_PF12 123 + +#define JAP_86 124 +/* tfj@olivia.ping.dk: + * The four keys are located over the numeric keypad, and are + * labelled A1-A4. It's an rc930 keyboard, from + * Regnecentralen/RC International, Now ICL. + * Scancodes: 59, 5a, 5b, 5c. + */ +#define RGN1 124 +#define RGN2 125 +#define RGN3 126 +#define RGN4 127 + +static unsigned char high_keys[128 - SC_LIM] = { + RGN1, RGN2, RGN3, RGN4, 0, 0, 0, /* 0x59-0x5f */ + 0, 0, 0, 0, 0, 0, 0, 0, /* 0x60-0x67 */ + 0, 0, 0, 0, 0, FOCUS_PF11, 0, FOCUS_PF12, /* 0x68-0x6f */ + 0, 0, 0, FOCUS_PF2, FOCUS_PF9, 0, 0, FOCUS_PF3, /* 0x70-0x77 */ + FOCUS_PF4, FOCUS_PF5, FOCUS_PF6, FOCUS_PF7, /* 0x78-0x7b */ + FOCUS_PF8, JAP_86, FOCUS_PF10, 0 /* 0x7c-0x7f */ +}; + +/* BTC */ +#define E0_MACRO 112 +/* LK450 */ +#define E0_F13 113 +#define E0_F14 114 +#define E0_HELP 115 +#define E0_DO 116 +#define E0_F17 117 +#define E0_KPMINPLUS 118 +/* + * My OmniKey generates e0 4c for the "OMNI" key and the + * right alt key does nada. [kkoller@nyx10.cs.du.edu] + */ +#define E0_OK 124 +/* + * New microsoft keyboard is rumoured to have + * e0 5b (left window button), e0 5c (right window button), + * e0 5d (menu button). [or: LBANNER, RBANNER, RMENU] + * [or: Windows_L, Windows_R, TaskMan] + */ +#define E0_MSLW 125 +#define E0_MSRW 126 +#define E0_MSTM 127 + +static unsigned char e0_keys[128] = { + 0, 0, 0, 0, 0, 0, 0, 0, /* 0x00-0x07 */ + 0, 0, 0, 0, 0, 0, 0, 0, /* 0x08-0x0f */ + 0, 0, 0, 0, 0, 0, 0, 0, /* 0x10-0x17 */ + 0, 0, 0, 0, E0_KPENTER, E0_RCTRL, 0, 0, /* 0x18-0x1f */ + 0, 0, 0, 0, 0, 0, 0, 0, /* 0x20-0x27 */ + 0, 0, 0, 0, 0, 0, 0, 0, /* 0x28-0x2f */ + 0, 0, 0, 0, 0, E0_KPSLASH, 0, E0_PRSCR, /* 0x30-0x37 */ + E0_RALT, 0, 0, 0, 0, E0_F13, E0_F14, E0_HELP, /* 0x38-0x3f */ + E0_DO, E0_F17, 0, 0, 0, 0, E0_BREAK, E0_HOME, /* 0x40-0x47 */ + E0_UP, E0_PGUP, 0, E0_LEFT, E0_OK, E0_RIGHT, E0_KPMINPLUS, E0_END, /* 0x48-0x4f */ + E0_DOWN, E0_PGDN, E0_INS, E0_DEL, 0, 0, 0, 0, /* 0x50-0x57 */ + 0, 0, 0, E0_MSLW, E0_MSRW, E0_MSTM, 0, 0, /* 0x58-0x5f */ + 0, 0, 0, 0, 0, 0, 0, 0, /* 0x60-0x67 */ + 0, 0, 0, 0, 0, 0, 0, E0_MACRO, /* 0x68-0x6f */ + //0, 0, 0, 0, 0, 0, 0, 0, /* 0x70-0x77 */ + 0, 0, 0, 0, 0, E0_BACKSLASH, 0, 0, /* 0x70-0x77 */ + 0, 0, 0, E0_YEN, 0, 0, 0, 0 /* 0x78-0x7f */ +}; + +int sa1111_setkeycode(unsigned int scancode, unsigned int keycode) +{ + if (scancode < SC_LIM || scancode > 255 || keycode > 127) + return -EINVAL; + if (scancode < 128) + high_keys[scancode - SC_LIM] = keycode; + else + e0_keys[scancode - 128] = keycode; + return 0; +} + +int sa1111_getkeycode(unsigned int scancode) +{ + return + (scancode < SC_LIM || scancode > 255) ? -EINVAL : + (scancode < + 128) ? high_keys[scancode - SC_LIM] : e0_keys[scancode - 128]; +} + +static int do_acknowledge(unsigned char scancode) +{ + if (reply_expected) { + /* Unfortunately, we must recognise these codes only if we know they + * are known to be valid (i.e., after sending a command), because there + * are some brain-damaged keyboards (yes, FOCUS 9000 again) which have + * keys with such codes :( + */ + if (scancode == KBD_REPLY_ACK) { + acknowledge = 1; + reply_expected = 0; + return 0; + } else if (scancode == KBD_REPLY_RESEND) { + resend = 1; + reply_expected = 0; + return 0; + } + /* Should not happen... */ +#if 0 + printk(KERN_DEBUG "keyboard reply expected - got %02x\n", + scancode); +#endif + } + return 1; +} + +int +sa1111_translate(unsigned char scancode, unsigned char *keycode, + char raw_mode) +{ + static int prev_scancode = 0; + + /* special prefix scancodes.. */ + if (scancode == 0xe0 || scancode == 0xe1) { + prev_scancode = scancode; + return 0; + } + + /* 0xFF is sent by a few keyboards, ignore it. 0x00 is error */ + if (scancode == 0x00 || scancode == 0xff) { + prev_scancode = 0; + return 0; + } + + scancode &= 0x7f; + + if (prev_scancode) { + /* + * usually it will be 0xe0, but a Pause key generates + * e1 1d 45 e1 9d c5 when pressed, and nothing when released + */ + if (prev_scancode != 0xe0) { + if (prev_scancode == 0xe1 && scancode == 0x1d) { + prev_scancode = 0x100; + return 0; + } + else if (prev_scancode == 0x100 + && scancode == 0x45) { + *keycode = E1_PAUSE; + prev_scancode = 0; + } else { +#ifdef KBD_REPORT_UNKN + if (!raw_mode) + printk(KERN_INFO + "keyboard: unknown e1 escape sequence\n"); +#endif + prev_scancode = 0; + return 0; + } + } else { + prev_scancode = 0; + /* + * The keyboard maintains its own internal caps lock and + * num lock statuses. In caps lock mode E0 AA precedes make + * code and E0 2A follows break code. In num lock mode, + * E0 2A precedes make code and E0 AA follows break code. + * We do our own book-keeping, so we will just ignore these. + */ + /* + * For my keyboard there is no caps lock mode, but there are + * both Shift-L and Shift-R modes. The former mode generates + * E0 2A / E0 AA pairs, the latter E0 B6 / E0 36 pairs. + * So, we should also ignore the latter. - aeb@cwi.nl + */ + if (scancode == 0x2a || scancode == 0x36) + return 0; + + if (e0_keys[scancode]) + *keycode = e0_keys[scancode]; + else { +#ifdef KBD_REPORT_UNKN + if (!raw_mode) + printk(KERN_INFO + "keyboard: unknown scancode e0 %02x\n", + scancode); +#endif + return 0; + } + } + } else if (scancode >= SC_LIM) { + /* This happens with the FOCUS 9000 keyboard + Its keys PF1..PF12 are reported to generate + 55 73 77 78 79 7a 7b 7c 74 7e 6d 6f + Moreover, unless repeated, they do not generate + key-down events, so we have to zero up_flag below */ + /* Also, Japanese 86/106 keyboards are reported to + generate 0x73 and 0x7d for \ - and \ | respectively. */ + /* Also, some Brazilian keyboard is reported to produce + 0x73 and 0x7e for \ ? and KP-dot, respectively. */ + + *keycode = high_keys[scancode - SC_LIM]; + + if (!*keycode) { + if (!raw_mode) { +#ifdef KBD_REPORT_UNKN + printk(KERN_INFO + "keyboard: unrecognized scancode (%02x)" + " - ignored\n", scancode); +#endif + } + return 0; + } + } else + *keycode = scancode; + return 1; +} + +char sa1111_unexpected_up(unsigned char keycode) +{ + /* unexpected, but this can happen: maybe this was a key release for a + FOCUS 9000 PF key; if we want to see it, we have to clear up_flag */ + if (keycode >= SC_LIM || keycode == 85) + return 0; + else + return 0200; +} + +static unsigned char kbd_exists = 1; + +static inline void handle_keyboard_event(unsigned char scancode) +{ +#ifdef CONFIG_VT + kbd_exists = 1; + if (do_acknowledge(scancode)) + handle_scancode(scancode, !(scancode & 0x80)); +#endif + tasklet_schedule(&keyboard_tasklet); +} + +/* + * This reads the keyboard status port, and does the + * appropriate action. + * + * It requires that we hold the keyboard controller + * spinlock. + */ +static void handle_kbd_event(void) +{ + unsigned int status = KBDSTAT; + unsigned int work = 10000; + unsigned char scancode; + + while (status & KBD_STAT_RXF) { + while (status & KBD_STAT_RXF) { + scancode = KBDDATA & 0xff; + if (!(status & KBD_STAT_STP)) + handle_keyboard_event(scancode); + if (!--work) { + printk(KERN_ERR + "pc_keyb: keyboard controller jammed (0x%02X).\n", + status); + return; + } + status = KBDSTAT; + } + work = 10000; + } + + if (status & KBD_STAT_STP) + KBDSTAT = KBD_STAT_STP; +} + +static void keyboard_interrupt(int irq, void *dev_id, struct pt_regs *regs) +{ + unsigned long flags; + +#ifdef CONFIG_VT + kbd_pt_regs = regs; +#endif + spin_lock_irqsave(&kbd_controller_lock, flags); + handle_kbd_event(); + spin_unlock_irqrestore(&kbd_controller_lock, flags); +} + +/* + * send_data sends a character to the keyboard and waits + * for an acknowledge, possibly retrying if asked to. Returns + * the success status. + * + * Don't use 'jiffies', so that we don't depend on interrupts + */ +static int send_data(unsigned char data) +{ + int retries = 3; + + do { + unsigned long timeout = KBD_TIMEOUT; + + acknowledge = 0; /* Set by interrupt routine on receipt of ACK. */ + resend = 0; + reply_expected = 1; + kbd_write_output_w(data); + for (;;) { + if (acknowledge) + return 1; + if (resend) + break; + mdelay(1); + if (!--timeout) { +#ifdef KBD_REPORT_TIMEOUTS + printk(KERN_WARNING + "keyboard: Timeout - AT keyboard not present?\n"); +#endif + return 0; + } + } + } + while (retries-- > 0); +#ifdef KBD_REPORT_TIMEOUTS + printk(KERN_WARNING + "keyboard: Too many NACKs -- noisy kbd cable?\n"); +#endif + return 0; +} + +void sa1111_leds(unsigned char leds) +{ + if (kbd_exists + && (!send_data(KBD_CMD_SET_LEDS) || !send_data(leds))) { + send_data(KBD_CMD_ENABLE); /* re-enable kbd if any errors */ + kbd_exists = 0; + } +} + +#define KBD_NO_DATA (-1) /* No data */ +#define KBD_BAD_DATA (-2) /* Parity or other error */ + +static int __init kbd_read_data(void) +{ + int retval = KBD_NO_DATA; + unsigned int status; + + status = KBDSTAT; + if (status & KBD_STAT_RXF) { + unsigned char data = KBDDATA; + + retval = data; + if (status & KBD_STAT_STP) + retval = KBD_BAD_DATA; + } + return retval; +} + +static void __init kbd_clear_input(void) +{ + int maxread = 100; /* Random number */ + + do { + if (kbd_read_data() == KBD_NO_DATA) + break; + } + while (--maxread); +} + +static int __init kbd_wait_for_input(void) +{ + long timeout = KBD_INIT_TIMEOUT; + + do { + int retval = kbd_read_data(); + if (retval >= 0) + return retval; + mdelay(1); + } + while (--timeout); + return -1; +} + +#if 0 +static void kbd_write_command_w(int data) +{ + unsigned long flags; + + spin_lock_irqsave(&kbd_controller_lock, flags); + kb_wait(); + kbd_write_command(data); + spin_unlock_irqrestore(&kbd_controller_lock, flags); +} +#endif + +static void kbd_write_output_w(int data) +{ + unsigned long flags; + + spin_lock_irqsave(&kbd_controller_lock, flags); + kb_wait(); + KBDDATA = data & 0xff; + spin_unlock_irqrestore(&kbd_controller_lock, flags); +} + +/* + * Test the keyboard interface. We basically check to make sure that + * we can drive each line to the keyboard independently of each other. + */ +static int kbdif_test(void) +{ + int ret = 0; + + KBDCR = KBDCR_ENA | KBDCR_FKC; + udelay(2); + if ((KBDSTAT & (KBDSTAT_KBC | KBDSTAT_KBD)) != KBDSTAT_KBD) { + printk("Keyboard interface test failed[1]: %02x\n", + KBDSTAT); + ret = -ENODEV; + } + + KBDCR = KBDCR_ENA; + udelay(2); + if ((KBDSTAT & (KBDSTAT_KBC | KBDSTAT_KBD)) != (KBDSTAT_KBC | KBDSTAT_KBD)) { + printk("Keyboard interface test failed[2]: %02x\n", + KBDSTAT); + ret = -ENODEV; + } + + KBDCR = KBDCR_ENA | KBDCR_FKD; + udelay(2); + if ((KBDSTAT & (KBDSTAT_KBC | KBDSTAT_KBD)) != KBDSTAT_KBC) { + printk("Keyboard interface test failed[3]: %02x\n", + KBDSTAT); + ret = -ENODEV; + } + + return ret; +} + +static char *__init initialize_kbd(void) +{ + int status; + + /* + * Test the keyboard interface. + */ + kbdif_test(); + + /* + * Ok, drop the force low bits, and wait a while, + * and clear the stop bit error flag. + */ + KBDCR = KBDCR_ENA; + udelay(4); + KBDSTAT = KBD_STAT_STP; + + /* + * Ok, we're now ready to talk to the keyboard. Reset + * it, just to make sure we're starting in a sane state. + * + * Set up to try again if the keyboard asks for RESEND. + */ + do { + KBDDATA = KBD_CMD_RESET; + status = kbd_wait_for_input(); + if (status == KBD_REPLY_ACK) + break; + if (status != KBD_REPLY_RESEND) + return "Keyboard reset failed, no ACK"; + } while (1); + + if (kbd_wait_for_input() != KBD_REPLY_POR) + return "Keyboard reset failed, no POR"; + + /* + * Set keyboard controller mode. During this, the keyboard should be + * in the disabled state. + * + * Set up to try again if the keyboard asks for RESEND. + */ + do { + kbd_write_output_w(KBD_CMD_DISABLE); + status = kbd_wait_for_input(); + if (status == KBD_REPLY_ACK) + break; + if (status != KBD_REPLY_RESEND) + return "Disable keyboard: no ACK"; + } while (1); + +#if 0 /*@@@ */ + kbd_write_command_w(KBD_CCMD_WRITE_MODE); + kbd_write_output_w(KBD_MODE_KBD_INT + | KBD_MODE_SYS | KBD_MODE_DISABLE_MOUSE | + KBD_MODE_KCC); + + /* ibm powerpc portables need this to use scan-code set 1 -- Cort */ + kbd_write_command_w(KBD_CCMD_READ_MODE); + if (!(kbd_wait_for_input() & KBD_MODE_KCC)) { + /* + * If the controller does not support conversion, + * Set the keyboard to scan-code set 1. + */ + kbd_write_output_w(0xF0); + kbd_wait_for_input(); + kbd_write_output_w(0x01); + kbd_wait_for_input(); + } +#else + kbd_write_output_w(0xf0); + kbd_wait_for_input(); + kbd_write_output_w(0x01); + kbd_wait_for_input(); +#endif + + + kbd_write_output_w(KBD_CMD_ENABLE); + if (kbd_wait_for_input() != KBD_REPLY_ACK) + return "Enable keyboard: no ACK"; + + /* + * Finally, set the typematic rate to maximum. + */ + kbd_write_output_w(KBD_CMD_SET_RATE); + if (kbd_wait_for_input() != KBD_REPLY_ACK) + return "Set rate: no ACK"; + kbd_write_output_w(0x00); + if (kbd_wait_for_input() != KBD_REPLY_ACK) + return "Set rate: no ACK"; + + return NULL; +} + +int __init sa1111_kbd_init_hw(void) +{ + char *msg; + int ret; + + if (!request_mem_region(_KBDCR, 512, "keyboard")) + return -EBUSY; + + SKPCR |= SKPCR_PTCLKEN; + KBDCLKDIV = 0; + KBDPRECNT = 127; + + /* Flush any pending input. */ + kbd_clear_input(); + + msg = initialize_kbd(); + if (msg) + printk(KERN_WARNING "initialize_kbd: %s\n", msg); + +#if defined CONFIG_PSMOUSE + psaux_init(); +#endif + + k_setkeycode = sa1111_setkeycode; + k_getkeycode = sa1111_getkeycode; + k_translate = sa1111_translate; + k_unexpected_up = sa1111_unexpected_up; + k_leds = sa1111_leds; +#ifdef CONFIG_MAGIC_SYSRQ + k_sysrq_xlate = sa1111_sysrq_xlate; + k_sysrq_key = 0x54; +#endif + + /* Ok, finally allocate the IRQ, and off we go.. */ + ret = request_irq(IRQ_TPRXINT, keyboard_interrupt, 0, "keyboard", NULL); + if (ret) + release_mem_region(_KBDCR, 512); + + return ret; +} + +#if defined CONFIG_PSMOUSE + +static inline void handle_mouse_event(unsigned char scancode) +{ + if (mouse_reply_expected) { + if (scancode == AUX_ACK) { + mouse_reply_expected--; + return; + } + mouse_reply_expected = 0; + } + + add_mouse_randomness(scancode); + if (aux_count) { + int head = queue->head; + + queue->buf[head] = scancode; + head = (head + 1) & (AUX_BUF_SIZE - 1); + if (head != queue->tail) { + queue->head = head; + if (queue->fasync) + kill_fasync(&queue->fasync, SIGIO, + POLL_IN); + wake_up_interruptible(&queue->proc_list); + } + } +} + +static void handle_mse_event(void) +{ + unsigned int msests = MSESTAT; + unsigned int work = 10000; + unsigned char scancode; + + while (msests & MSE_STAT_RXF) { + while (msests & MSE_STAT_RXF) { + scancode = MSEDATA & 0xff; + if (!(msests & MSE_STAT_STP)) + handle_mouse_event(scancode); + if (!--work) { + printk(KERN_ERR + "pc_keyb: mouse controller jammed (0x%02X).\n", + msests); + return; + /*XXX*/} + msests = MSESTAT; + } + work = 10000; + } +} + +static void ms_wait(void) +{ + unsigned long timeout = KBC_TIMEOUT; + + do { + /* + * "handle_kbd_event()" will handle any incoming events + * while we wait - keypresses or mouse movement. + */ + handle_mse_event(); + if (MSESTAT & MSE_STAT_TXE) + return; + mdelay(1); + timeout--; + } + while (timeout); +#ifdef KBD_REPORT_TIMEOUTS + printk(KERN_WARNING "Mouse timed out[1]\n"); +#endif +} + +static void mouse_interrupt(int irq, void *dev_id, struct pt_regs *regs) +{ + unsigned long flags; + + spin_lock_irqsave(&kbd_controller_lock, flags); + handle_mse_event(); + spin_unlock_irqrestore(&kbd_controller_lock, flags); +} + +/* + * Check if this is a dual port controller. + */ +static int __init detect_auxiliary_port(void) +{ + unsigned long flags; + int loops = 10; + int retval = 0; + + /* Check if the BIOS detected a device on the auxiliary port. */ + if (aux_device_present == 0xaa) + return 1; + + spin_lock_irqsave(&kbd_controller_lock, flags); + + /* Put the value 0x5A in the output buffer using the "Write + * Auxiliary Device Output Buffer" command (0xD3). Poll the + * Status Register for a while to see if the value really + * turns up in the Data Register. If the KBD_STAT_MOUSE_OBF + * bit is also set to 1 in the Status Register, we assume this + * controller has an Auxiliary Port (a.k.a. Mouse Port). + */ + // kb_wait(); + // kbd_write_command(KBD_CCMD_WRITE_AUX_OBUF); + + SKPCR |= SKPCR_PMCLKEN; + + MSECLKDIV = 0; + MSEPRECNT = 127; + MSECR = MSECR_ENA; + mdelay(50); + MSEDATA = 0xf4; + mdelay(50); + + do { + unsigned int msests = MSESTAT; + + if (msests & MSE_STAT_RXF) { + do { + msests = MSEDATA; /* dummy read */ + mdelay(50); + msests = MSESTAT; + } + while (msests & MSE_STAT_RXF); + printk(KERN_INFO "Detected PS/2 Mouse Port.\n"); + retval = 1; + break; + } + mdelay(1); + } + while (--loops); + spin_unlock_irqrestore(&kbd_controller_lock, flags); + + return retval; +} + +/* + * Send a byte to the mouse. + */ +static void aux_write_dev(int val) +{ + unsigned long flags; + + spin_lock_irqsave(&kbd_controller_lock, flags); + // kb_wait(); + // kbd_write_command(KBD_CCMD_WRITE_MOUSE); + ms_wait(); + MSEDATA = val; + spin_unlock_irqrestore(&kbd_controller_lock, flags); +} + +/* + * Send a byte to the mouse & handle returned ack + */ +static void aux_write_ack(int val) +{ + unsigned long flags; + + spin_lock_irqsave(&kbd_controller_lock, flags); + // kb_wait(); + // kbd_write_command(KBD_CCMD_WRITE_MOUSE); + ms_wait(); + MSEDATA = val; + /* we expect an ACK in response. */ + mouse_reply_expected++; + ms_wait(); + spin_unlock_irqrestore(&kbd_controller_lock, flags); +} + +static unsigned char get_from_queue(void) +{ + unsigned char result; + unsigned long flags; + + spin_lock_irqsave(&kbd_controller_lock, flags); + result = queue->buf[queue->tail]; + queue->tail = (queue->tail + 1) & (AUX_BUF_SIZE - 1); + spin_unlock_irqrestore(&kbd_controller_lock, flags); + return result; +} + + +static inline int queue_empty(void) +{ + return queue->head == queue->tail; +} + +static int fasync_aux(int fd, struct file *filp, int on) +{ + int retval; + + retval = fasync_helper(fd, filp, on, &queue->fasync); + if (retval < 0) + return retval; + return 0; +} + + +/* + * Random magic cookie for the aux device + */ +#define AUX_DEV ((void *)queue) + +static int release_aux(struct inode *inode, struct file *file) +{ + fasync_aux(-1, file, 0); + if (--aux_count) + return 0; + // kbd_write_cmd(AUX_INTS_OFF); /* Disable controller ints */ + // kbd_write_command_w(KBD_CCMD_MOUSE_DISABLE); + aux_write_ack(AUX_DISABLE_DEV); /* Disable aux device */ + MSECR &= ~MSECR_ENA; + free_irq(IRQ_MSRXINT, AUX_DEV); + return 0; +} + +/* + * Install interrupt handler. + * Enable auxiliary device. + */ + +static int open_aux(struct inode *inode, struct file *file) +{ + if (aux_count++) { + return 0; + } + queue->head = queue->tail = 0; /* Flush input queue */ + /* Don't enable the mouse controller until we've registered IRQ handler */ + if (request_irq(IRQ_MSRXINT, mouse_interrupt, SA_SHIRQ, "PS/2 Mouse", AUX_DEV)) { + aux_count--; + return -EBUSY; + } + MSECLKDIV = 0; + MSEPRECNT = 127; + MSECR &= ~MSECR_ENA; + mdelay(50); + MSECR = MSECR_ENA; + mdelay(50); + MSEDATA = 0xf4; + mdelay(50); + if (MSESTAT & 0x0100) { + MSESTAT = 0x0100; /* clear IRQ status */ + } +/* kbd_write_command_w(KBD_CCMD_MOUSE_ENABLE); *//* Enable the + auxiliary port on + controller. */ + aux_write_ack(AUX_ENABLE_DEV); /* Enable aux device */ + // kbd_write_cmd(AUX_INTS_ON); /* Enable controller ints */ + + // send_data(KBD_CMD_ENABLE); /* try to workaround toshiba4030cdt problem */ + + return 0; +} + +/* + * Put bytes from input queue to buffer. + */ + +static ssize_t +read_aux(struct file *file, char *buffer, size_t count, loff_t * ppos) +{ + DECLARE_WAITQUEUE(wait, current); + ssize_t i = count; + unsigned char c; + + if (queue_empty()) { + if (file->f_flags & O_NONBLOCK) + return -EAGAIN; + add_wait_queue(&queue->proc_list, &wait); + repeat: + set_current_state(TASK_INTERRUPTIBLE); + if (queue_empty() && !signal_pending(current)) { + schedule(); + goto repeat; + } + current->state = TASK_RUNNING; + remove_wait_queue(&queue->proc_list, &wait); + } + while (i > 0 && !queue_empty()) { + c = get_from_queue(); + put_user(c, buffer++); + i--; + } + if (count - i) { + file->f_dentry->d_inode->i_atime = CURRENT_TIME; + return count - i; + } + if (signal_pending(current)) + return -ERESTARTSYS; + return 0; +} + +/* + * Write to the aux device. + */ + +static ssize_t +write_aux(struct file *file, const char *buffer, size_t count, + loff_t * ppos) +{ + ssize_t retval = 0; + + if (count) { + ssize_t written = 0; + + if (count > 32) + count = 32; /* Limit to 32 bytes. */ + do { + char c; + get_user(c, buffer++); + aux_write_dev(c); + written++; + } + while (--count); + retval = -EIO; + if (written) { + retval = written; + file->f_dentry->d_inode->i_mtime = CURRENT_TIME; + } + } + + return retval; +} + +static unsigned int aux_poll(struct file *file, poll_table * wait) +{ + poll_wait(file, &queue->proc_list, wait); + if (!queue_empty()) + return POLLIN | POLLRDNORM; + return 0; +} + +struct file_operations psaux_fops = { + read: read_aux, + write: write_aux, + poll: aux_poll, + open: open_aux, + release: release_aux, + fasync: fasync_aux, +}; + +/* + * Initialize driver. + */ +static struct miscdevice psaux_mouse = { + PSMOUSE_MINOR, "psaux", &psaux_fops +}; + + +static int __init psaux_init(void) +{ + int ret; + + if (!request_mem_region(_MSECR, 512, "psaux")) + return -EBUSY; + + if (!detect_auxiliary_port()) { + ret = -EIO; + goto out; + } + + misc_register(&psaux_mouse); + queue = (struct aux_queue *) kmalloc(sizeof(*queue), GFP_KERNEL); + memset(queue, 0, sizeof(*queue)); + queue->head = queue->tail = 0; + init_waitqueue_head(&queue->proc_list); + +#ifdef CONFIG_PSMOUSE + aux_write_ack(AUX_SET_SAMPLE); + aux_write_ack(100); /* 100 samples/sec */ + aux_write_ack(AUX_SET_RES); + aux_write_ack(3); /* 8 counts per mm */ + aux_write_ack(AUX_SET_SCALE21); /* 2:1 scaling */ +#endif + ret = 0; + + out: + if (ret) + release_mem_region(_MSECR, 512); + return ret; +} + +#endif /* CONFIG_PSMOUSE */ diff -urN orig/drivers/char/serial_21285.c linux/drivers/char/serial_21285.c --- orig/drivers/char/serial_21285.c Mon Aug 5 13:30:47 2002 +++ linux/drivers/char/serial_21285.c Thu Jan 1 01:00:00 1970 @@ -1,498 +0,0 @@ -/* - * linux/drivers/char/serial_21285.c - * - * Driver for the serial port on the 21285 StrongArm-110 core logic chip. - * - * Based on drivers/char/serial.c - */ - -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include - -#include -#include -#include -#include -#include - -#define BAUD_BASE (mem_fclk_21285/64) - -#define SERIAL_21285_NAME "ttyFB" -#define SERIAL_21285_MAJOR 204 -#define SERIAL_21285_MINOR 4 - -#define SERIAL_21285_AUXNAME "cuafb" -#define SERIAL_21285_AUXMAJOR 205 -#define SERIAL_21285_AUXMINOR 4 - -static struct tty_driver rs285_driver, callout_driver; -static int rs285_refcount; -static struct tty_struct *rs285_table[1]; - -static struct termios *rs285_termios[1]; -static struct termios *rs285_termios_locked[1]; - -static char wbuf[1000], *putp = wbuf, *getp = wbuf, x_char; -static struct tty_struct *rs285_tty; -static int rs285_use_count; - -static int rs285_write_room(struct tty_struct *tty) -{ - return putp >= getp ? (sizeof(wbuf) - (long) putp + (long) getp) : ((long) getp - (long) putp - 1); -} - -static void rs285_rx_int(int irq, void *dev_id, struct pt_regs *regs) -{ - if (!rs285_tty) { - disable_irq(IRQ_CONRX); - return; - } - while (!(*CSR_UARTFLG & 0x10)) { - int ch, flag; - ch = *CSR_UARTDR; - flag = *CSR_RXSTAT; - if (flag & 4) - tty_insert_flip_char(rs285_tty, 0, TTY_OVERRUN); - if (flag & 2) - flag = TTY_PARITY; - else if (flag & 1) - flag = TTY_FRAME; - tty_insert_flip_char(rs285_tty, ch, flag); - } - tty_flip_buffer_push(rs285_tty); -} - -static void rs285_send_xchar(struct tty_struct *tty, char ch) -{ - x_char = ch; - enable_irq(IRQ_CONTX); -} - -static void rs285_throttle(struct tty_struct *tty) -{ - if (I_IXOFF(tty)) - rs285_send_xchar(tty, STOP_CHAR(tty)); -} - -static void rs285_unthrottle(struct tty_struct *tty) -{ - if (I_IXOFF(tty)) { - if (x_char) - x_char = 0; - else - rs285_send_xchar(tty, START_CHAR(tty)); - } -} - -static void rs285_tx_int(int irq, void *dev_id, struct pt_regs *regs) -{ - while (!(*CSR_UARTFLG & 0x20)) { - if (x_char) { - *CSR_UARTDR = x_char; - x_char = 0; - continue; - } - if (putp == getp) { - disable_irq(IRQ_CONTX); - break; - } - *CSR_UARTDR = *getp; - if (++getp >= wbuf + sizeof(wbuf)) - getp = wbuf; - } - if (rs285_tty) - wake_up_interruptible(&rs285_tty->write_wait); -} - -static inline int rs285_xmit(int ch) -{ - if (putp + 1 == getp || (putp + 1 == wbuf + sizeof(wbuf) && getp == wbuf)) - return 0; - *putp = ch; - if (++putp >= wbuf + sizeof(wbuf)) - putp = wbuf; - enable_irq(IRQ_CONTX); - return 1; -} - -static int rs285_write(struct tty_struct *tty, int from_user, - const u_char * buf, int count) -{ - int i; - - if (from_user && verify_area(VERIFY_READ, buf, count)) - return -EINVAL; - - for (i = 0; i < count; i++) { - char ch; - if (from_user) - __get_user(ch, buf + i); - else - ch = buf[i]; - if (!rs285_xmit(ch)) - break; - } - return i; -} - -static void rs285_put_char(struct tty_struct *tty, u_char ch) -{ - rs285_xmit(ch); -} - -static int rs285_chars_in_buffer(struct tty_struct *tty) -{ - return sizeof(wbuf) - rs285_write_room(tty); -} - -static void rs285_flush_buffer(struct tty_struct *tty) -{ - disable_irq(IRQ_CONTX); - putp = getp = wbuf; - if (x_char) - enable_irq(IRQ_CONTX); -} - -static inline void rs285_set_cflag(int cflag) -{ - int h_lcr, baud, quot; - - switch (cflag & CSIZE) { - case CS5: - h_lcr = 0x10; - break; - case CS6: - h_lcr = 0x30; - break; - case CS7: - h_lcr = 0x50; - break; - default: /* CS8 */ - h_lcr = 0x70; - break; - - } - if (cflag & CSTOPB) - h_lcr |= 0x08; - if (cflag & PARENB) - h_lcr |= 0x02; - if (!(cflag & PARODD)) - h_lcr |= 0x04; - - switch (cflag & CBAUD) { - case B200: baud = 200; break; - case B300: baud = 300; break; - case B1200: baud = 1200; break; - case B1800: baud = 1800; break; - case B2400: baud = 2400; break; - case B4800: baud = 4800; break; - default: - case B9600: baud = 9600; break; - case B19200: baud = 19200; break; - case B38400: baud = 38400; break; - case B57600: baud = 57600; break; - case B115200: baud = 115200; break; - } - - /* - * The documented expression for selecting the divisor is: - * BAUD_BASE / baud - 1 - * However, typically BAUD_BASE is not divisible by baud, so - * we want to select the divisor that gives us the minimum - * error. Therefore, we want: - * int(BAUD_BASE / baud - 0.5) -> - * int(BAUD_BASE / baud - (baud >> 1) / baud) -> - * int((BAUD_BASE - (baud >> 1)) / baud) - */ - quot = (BAUD_BASE - (baud >> 1)) / baud; - - *CSR_UARTCON = 0; - *CSR_L_UBRLCR = quot & 0xff; - *CSR_M_UBRLCR = (quot >> 8) & 0x0f; - *CSR_H_UBRLCR = h_lcr; - *CSR_UARTCON = 1; -} - -static void rs285_set_termios(struct tty_struct *tty, struct termios *old) -{ - if (old && tty->termios->c_cflag == old->c_cflag) - return; - rs285_set_cflag(tty->termios->c_cflag); -} - - -static void rs285_stop(struct tty_struct *tty) -{ - disable_irq(IRQ_CONTX); -} - -static void rs285_start(struct tty_struct *tty) -{ - enable_irq(IRQ_CONTX); -} - -static void rs285_wait_until_sent(struct tty_struct *tty, int timeout) -{ - int orig_jiffies = jiffies; - while (*CSR_UARTFLG & 8) { - current->state = TASK_INTERRUPTIBLE; - schedule_timeout(1); - if (signal_pending(current)) - break; - if (timeout && time_after(jiffies, orig_jiffies + timeout)) - break; - } - current->state = TASK_RUNNING; -} - -static int rs285_open(struct tty_struct *tty, struct file *filp) -{ - int line; - - MOD_INC_USE_COUNT; - line = MINOR(tty->device) - tty->driver.minor_start; - if (line) { - MOD_DEC_USE_COUNT; - return -ENODEV; - } - - tty->driver_data = NULL; - if (!rs285_tty) - rs285_tty = tty; - - enable_irq(IRQ_CONRX); - rs285_use_count++; - return 0; -} - -static void rs285_close(struct tty_struct *tty, struct file *filp) -{ - if (!--rs285_use_count) { - rs285_wait_until_sent(tty, 0); - disable_irq(IRQ_CONRX); - disable_irq(IRQ_CONTX); - rs285_tty = NULL; - } - MOD_DEC_USE_COUNT; -} - -static int __init rs285_init(void) -{ - int baud = B9600; - - if (machine_is_personal_server()) - baud = B57600; - - rs285_driver.magic = TTY_DRIVER_MAGIC; - rs285_driver.driver_name = "serial_21285"; - rs285_driver.name = SERIAL_21285_NAME; - rs285_driver.major = SERIAL_21285_MAJOR; - rs285_driver.minor_start = SERIAL_21285_MINOR; - rs285_driver.num = 1; - rs285_driver.type = TTY_DRIVER_TYPE_SERIAL; - rs285_driver.subtype = SERIAL_TYPE_NORMAL; - rs285_driver.init_termios = tty_std_termios; - rs285_driver.init_termios.c_cflag = baud | CS8 | CREAD | HUPCL | CLOCAL; - rs285_driver.flags = TTY_DRIVER_REAL_RAW; - rs285_driver.refcount = &rs285_refcount; - rs285_driver.table = rs285_table; - rs285_driver.termios = rs285_termios; - rs285_driver.termios_locked = rs285_termios_locked; - - rs285_driver.open = rs285_open; - rs285_driver.close = rs285_close; - rs285_driver.write = rs285_write; - rs285_driver.put_char = rs285_put_char; - rs285_driver.write_room = rs285_write_room; - rs285_driver.chars_in_buffer = rs285_chars_in_buffer; - rs285_driver.flush_buffer = rs285_flush_buffer; - rs285_driver.throttle = rs285_throttle; - rs285_driver.unthrottle = rs285_unthrottle; - rs285_driver.send_xchar = rs285_send_xchar; - rs285_driver.set_termios = rs285_set_termios; - rs285_driver.stop = rs285_stop; - rs285_driver.start = rs285_start; - rs285_driver.wait_until_sent = rs285_wait_until_sent; - - callout_driver = rs285_driver; - callout_driver.name = SERIAL_21285_AUXNAME; - callout_driver.major = SERIAL_21285_AUXMAJOR; - callout_driver.subtype = SERIAL_TYPE_CALLOUT; - - if (request_irq(IRQ_CONRX, rs285_rx_int, 0, "rs285", NULL)) - panic("Couldn't get rx irq for rs285"); - - if (request_irq(IRQ_CONTX, rs285_tx_int, 0, "rs285", NULL)) - panic("Couldn't get tx irq for rs285"); - - if (tty_register_driver(&rs285_driver)) - printk(KERN_ERR "Couldn't register 21285 serial driver\n"); - if (tty_register_driver(&callout_driver)) - printk(KERN_ERR "Couldn't register 21285 callout driver\n"); - - return 0; -} - -static void __exit rs285_fini(void) -{ - unsigned long flags; - int ret; - - save_flags(flags); - cli(); - ret = tty_unregister_driver(&callout_driver); - if (ret) - printk(KERN_ERR "Unable to unregister 21285 callout driver " - "(%d)\n", ret); - ret = tty_unregister_driver(&rs285_driver); - if (ret) - printk(KERN_ERR "Unable to unregister 21285 driver (%d)\n", - ret); - free_irq(IRQ_CONTX, NULL); - free_irq(IRQ_CONRX, NULL); - restore_flags(flags); -} - -module_init(rs285_init); -module_exit(rs285_fini); - -#ifdef CONFIG_SERIAL_21285_CONSOLE -/************** console driver *****************/ - -static void rs285_console_write(struct console *co, const char *s, u_int count) -{ - int i; - - disable_irq(IRQ_CONTX); - for (i = 0; i < count; i++) { - while (*CSR_UARTFLG & 0x20); - *CSR_UARTDR = s[i]; - if (s[i] == '\n') { - while (*CSR_UARTFLG & 0x20); - *CSR_UARTDR = '\r'; - } - } - enable_irq(IRQ_CONTX); -} - -static kdev_t rs285_console_device(struct console *c) -{ - return MKDEV(SERIAL_21285_MAJOR, SERIAL_21285_MINOR); -} - -static int __init rs285_console_setup(struct console *co, char *options) -{ - int baud = 9600; - int bits = 8; - int parity = 'n'; - int cflag = CREAD | HUPCL | CLOCAL; - - if (machine_is_personal_server()) - baud = 57600; - - if (options) { - char *s = options; - baud = simple_strtoul(options, NULL, 10); - while (*s >= '0' && *s <= '9') - s++; - if (*s) - parity = *s++; - if (*s) - bits = *s - '0'; - } - - /* - * Now construct a cflag setting. - */ - switch (baud) { - case 1200: - cflag |= B1200; - break; - case 2400: - cflag |= B2400; - break; - case 4800: - cflag |= B4800; - break; - case 9600: - cflag |= B9600; - break; - case 19200: - cflag |= B19200; - break; - case 38400: - cflag |= B38400; - break; - case 57600: - cflag |= B57600; - break; - case 115200: - cflag |= B115200; - break; - default: - cflag |= B9600; - break; - } - switch (bits) { - case 7: - cflag |= CS7; - break; - default: - cflag |= CS8; - break; - } - switch (parity) { - case 'o': - case 'O': - cflag |= PARODD; - break; - case 'e': - case 'E': - cflag |= PARENB; - break; - } - co->cflag = cflag; - rs285_set_cflag(cflag); - rs285_console_write(NULL, "\e[2J\e[Hboot ", 12); - if (options) - rs285_console_write(NULL, options, strlen(options)); - else - rs285_console_write(NULL, "no options", 10); - rs285_console_write(NULL, "\n", 1); - - return 0; -} - -static struct console rs285_cons = -{ - name: SERIAL_21285_NAME, - write: rs285_console_write, - device: rs285_console_device, - setup: rs285_console_setup, - flags: CON_PRINTBUFFER, - index: -1, -}; - -void __init rs285_console_init(void) -{ - register_console(&rs285_cons); -} - -#endif /* CONFIG_SERIAL_21285_CONSOLE */ - -MODULE_LICENSE("GPL"); -EXPORT_NO_SYMBOLS; diff -urN orig/drivers/char/serial_amba.c linux/drivers/char/serial_amba.c --- orig/drivers/char/serial_amba.c Mon Aug 5 13:30:47 2002 +++ linux/drivers/char/serial_amba.c Thu Jan 1 01:00:00 1970 @@ -1,2015 +0,0 @@ -/* - * linux/drivers/char/serial_amba.c - * - * Driver for AMBA serial ports - * - * Based on drivers/char/serial.c, by Linus Torvalds, Theodore Ts'o. - * - * Copyright 1999 ARM Limited - * Copyright (C) 2000 Deep Blue Solutions Ltd. - * - * This program is free software; you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation; either version 2 of the License, or - * (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program; if not, write to the Free Software - * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA - * - * - * This is a generic driver for ARM AMBA-type serial ports. They - * have a lot of 16550-like features, but are not register compatable. - * Note that although they do have CTS, DCD and DSR inputs, they do - * not have an RI input, nor do they have DTR or RTS outputs. If - * required, these have to be supplied via some other means (eg, GPIO) - * and hooked into this driver. - * - * This could very easily become a generic serial driver for dumb UARTs - * (eg, {82,16x}50, 21285, SA1100). - */ - -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include - -#include -#include -#include -#include -#include - -#include - -#define SERIAL_AMBA_NAME "ttyAM" -#define SERIAL_AMBA_MAJOR 204 -#define SERIAL_AMBA_MINOR 16 -#define SERIAL_AMBA_NR 2 - -#define CALLOUT_AMBA_NAME "cuaam" -#define CALLOUT_AMBA_MAJOR 205 -#define CALLOUT_AMBA_MINOR 16 -#define CALLOUT_AMBA_NR SERIAL_AMBA_NR - -#ifndef TRUE -#define TRUE 1 -#endif -#ifndef FALSE -#define FALSE 0 -#endif - -#define DEBUG 0 -#define DEBUG_LEDS 0 - -#if DEBUG_LEDS -extern int get_leds(void); -extern int set_leds(int); -#endif - -/* - * Access routines for the AMBA UARTs - */ -#define UART_GET_INT_STATUS(p) IO_READ((p)->uart_base + AMBA_UARTIIR) -#define UART_GET_FR(p) IO_READ((p)->uart_base + AMBA_UARTFR) -#define UART_GET_CHAR(p) IO_READ((p)->uart_base + AMBA_UARTDR) -#define UART_PUT_CHAR(p, c) IO_WRITE((p)->uart_base + AMBA_UARTDR, (c)) -#define UART_GET_RSR(p) IO_READ((p)->uart_base + AMBA_UARTRSR) -#define UART_GET_CR(p) IO_READ((p)->uart_base + AMBA_UARTCR) -#define UART_PUT_CR(p,c) IO_WRITE((p)->uart_base + AMBA_UARTCR, (c)) -#define UART_GET_LCRL(p) IO_READ((p)->uart_base + AMBA_UARTLCR_L) -#define UART_PUT_LCRL(p,c) IO_WRITE((p)->uart_base + AMBA_UARTLCR_L, (c)) -#define UART_GET_LCRM(p) IO_READ((p)->uart_base + AMBA_UARTLCR_M) -#define UART_PUT_LCRM(p,c) IO_WRITE((p)->uart_base + AMBA_UARTLCR_M, (c)) -#define UART_GET_LCRH(p) IO_READ((p)->uart_base + AMBA_UARTLCR_H) -#define UART_PUT_LCRH(p,c) IO_WRITE((p)->uart_base + AMBA_UARTLCR_H, (c)) -#define UART_RX_DATA(s) (((s) & AMBA_UARTFR_RXFE) == 0) -#define UART_TX_READY(s) (((s) & AMBA_UARTFR_TXFF) == 0) -#define UART_TX_EMPTY(p) ((UART_GET_FR(p) & AMBA_UARTFR_TMSK) == 0) - -#define AMBA_UARTRSR_ANY (AMBA_UARTRSR_OE|AMBA_UARTRSR_BE|AMBA_UARTRSR_PE|AMBA_UARTRSR_FE) -#define AMBA_UARTFR_MODEM_ANY (AMBA_UARTFR_DCD|AMBA_UARTFR_DSR|AMBA_UARTFR_CTS) - -/* - * Things needed by tty driver - */ -static struct tty_driver ambanormal_driver, ambacallout_driver; -static int ambauart_refcount; -static struct tty_struct *ambauart_table[SERIAL_AMBA_NR]; -static struct termios *ambauart_termios[SERIAL_AMBA_NR]; -static struct termios *ambauart_termios_locked[SERIAL_AMBA_NR]; - -#if defined(CONFIG_SERIAL_AMBA_CONSOLE) && defined(CONFIG_MAGIC_SYSRQ) -#define SUPPORT_SYSRQ -#endif - -/* - * Things needed internally to this driver - */ - -/* - * tmp_buf is used as a temporary buffer by serial_write. We need to - * lock it in case the copy_from_user blocks while swapping in a page, - * and some other program tries to do a serial write at the same time. - * Since the lock will only come under contention when the system is - * swapping and available memory is low, it makes sense to share one - * buffer across all the serial ports, since it significantly saves - * memory if large numbers of serial ports are open. - */ -static u_char *tmp_buf; -static DECLARE_MUTEX(tmp_buf_sem); - -#define HIGH_BITS_OFFSET ((sizeof(long)-sizeof(int))*8) - -/* number of characters left in xmit buffer before we ask for more */ -#define WAKEUP_CHARS 256 -#define AMBA_ISR_PASS_LIMIT 256 - -#define EVT_WRITE_WAKEUP 0 - -struct amba_icount { - __u32 cts; - __u32 dsr; - __u32 rng; - __u32 dcd; - __u32 rx; - __u32 tx; - __u32 frame; - __u32 overrun; - __u32 parity; - __u32 brk; - __u32 buf_overrun; -}; - -/* - * Static information about the port - */ -struct amba_port { - unsigned int uart_base; - unsigned int irq; - unsigned int uartclk; - unsigned int fifosize; - unsigned int tiocm_support; - void (*set_mctrl)(struct amba_port *, u_int mctrl); -}; - -/* - * This is the state information which is persistent across opens - */ -struct amba_state { - struct amba_icount icount; - unsigned int line; - unsigned int close_delay; - unsigned int closing_wait; - unsigned int custom_divisor; - unsigned int flags; - struct termios normal_termios; - struct termios callout_termios; - - int count; - struct amba_info *info; -}; - -#define AMBA_XMIT_SIZE 1024 -/* - * This is the state information which is only valid when the port is open. - */ -struct amba_info { - struct amba_port *port; - struct amba_state *state; - struct tty_struct *tty; - unsigned char x_char; - unsigned char old_status; - unsigned char read_status_mask; - unsigned char ignore_status_mask; - struct circ_buf xmit; - unsigned int flags; -#ifdef SUPPORT_SYSRQ - unsigned long sysrq; -#endif - - unsigned int event; - unsigned int timeout; - unsigned int lcr_h; - unsigned int mctrl; - int blocked_open; - pid_t session; - pid_t pgrp; - - struct tasklet_struct tlet; - - wait_queue_head_t open_wait; - wait_queue_head_t close_wait; - wait_queue_head_t delta_msr_wait; -}; - -#ifdef CONFIG_SERIAL_AMBA_CONSOLE -static struct console ambauart_cons; -#endif -static void ambauart_change_speed(struct amba_info *info, struct termios *old_termios); -static void ambauart_wait_until_sent(struct tty_struct *tty, int timeout); - -#if 1 //def CONFIG_SERIAL_INTEGRATOR -static void amba_set_mctrl_null(struct amba_port *port, u_int mctrl) -{ -} - -static struct amba_port amba_ports[SERIAL_AMBA_NR] = { - { - uart_base: IO_ADDRESS(INTEGRATOR_UART0_BASE), - irq: IRQ_UARTINT0, - uartclk: 14745600, - fifosize: 8, - set_mctrl: amba_set_mctrl_null, - }, - { - uart_base: IO_ADDRESS(INTEGRATOR_UART1_BASE), - irq: IRQ_UARTINT1, - uartclk: 14745600, - fifosize: 8, - set_mctrl: amba_set_mctrl_null, - } -}; -#endif - -static struct amba_state amba_state[SERIAL_AMBA_NR]; - -static void ambauart_enable_rx_interrupt(struct amba_info *info) -{ - unsigned int cr; - - cr = UART_GET_CR(info->port); - cr |= AMBA_UARTCR_RIE | AMBA_UARTCR_RTIE; - UART_PUT_CR(info->port, cr); -} - -static void ambauart_disable_rx_interrupt(struct amba_info *info) -{ - unsigned int cr; - - cr = UART_GET_CR(info->port); - cr &= ~(AMBA_UARTCR_RIE | AMBA_UARTCR_RTIE); - UART_PUT_CR(info->port, cr); -} - -static void ambauart_enable_tx_interrupt(struct amba_info *info) -{ - unsigned int cr; - - cr = UART_GET_CR(info->port); - cr |= AMBA_UARTCR_TIE; - UART_PUT_CR(info->port, cr); -} - -static void ambauart_disable_tx_interrupt(struct amba_info *info) -{ - unsigned int cr; - - cr = UART_GET_CR(info->port); - cr &= ~AMBA_UARTCR_TIE; - UART_PUT_CR(info->port, cr); -} - -static void ambauart_stop(struct tty_struct *tty) -{ - struct amba_info *info = tty->driver_data; - unsigned long flags; - - save_flags(flags); cli(); - ambauart_disable_tx_interrupt(info); - restore_flags(flags); -} - -static void ambauart_start(struct tty_struct *tty) -{ - struct amba_info *info = tty->driver_data; - unsigned long flags; - - save_flags(flags); cli(); - if (info->xmit.head != info->xmit.tail - && info->xmit.buf) - ambauart_enable_tx_interrupt(info); - restore_flags(flags); -} - - -/* - * This routine is used by the interrupt handler to schedule - * processing in the software interrupt portion of the driver. - */ -static void ambauart_event(struct amba_info *info, int event) -{ - info->event |= 1 << event; - tasklet_schedule(&info->tlet); -} - -static void -#ifdef SUPPORT_SYSRQ -ambauart_rx_chars(struct amba_info *info, struct pt_regs *regs) -#else -ambauart_rx_chars(struct amba_info *info) -#endif -{ - struct tty_struct *tty = info->tty; - unsigned int status, ch, rsr, flg, ignored = 0; - struct amba_icount *icount = &info->state->icount; - struct amba_port *port = info->port; - - status = UART_GET_FR(port); - while (UART_RX_DATA(status)) { - ch = UART_GET_CHAR(port); - - if (tty->flip.count >= TTY_FLIPBUF_SIZE) - goto ignore_char; - icount->rx++; - - flg = TTY_NORMAL; - - /* - * Note that the error handling code is - * out of the main execution path - */ - rsr = UART_GET_RSR(port); - if (rsr & AMBA_UARTRSR_ANY) - goto handle_error; -#ifdef SUPPORT_SYSRQ - if (info->sysrq) { - if (ch && time_before(jiffies, info->sysrq)) { - handle_sysrq(ch, regs, NULL, NULL); - info->sysrq = 0; - goto ignore_char; - } - info->sysrq = 0; - } -#endif - error_return: - *tty->flip.flag_buf_ptr++ = flg; - *tty->flip.char_buf_ptr++ = ch; - tty->flip.count++; - ignore_char: - status = UART_GET_FR(port); - } -out: - tty_flip_buffer_push(tty); - return; - -handle_error: - if (rsr & AMBA_UARTRSR_BE) { - rsr &= ~(AMBA_UARTRSR_FE | AMBA_UARTRSR_PE); - icount->brk++; - -#ifdef SUPPORT_SYSRQ - if (info->state->line == ambauart_cons.index) { - if (!info->sysrq) { - info->sysrq = jiffies + HZ*5; - goto ignore_char; - } - } -#endif - } else if (rsr & AMBA_UARTRSR_PE) - icount->parity++; - else if (rsr & AMBA_UARTRSR_FE) - icount->frame++; - if (rsr & AMBA_UARTRSR_OE) - icount->overrun++; - - if (rsr & info->ignore_status_mask) { - if (++ignored > 100) - goto out; - goto ignore_char; - } - rsr &= info->read_status_mask; - - if (rsr & AMBA_UARTRSR_BE) - flg = TTY_BREAK; - else if (rsr & AMBA_UARTRSR_PE) - flg = TTY_PARITY; - else if (rsr & AMBA_UARTRSR_FE) - flg = TTY_FRAME; - - if (rsr & AMBA_UARTRSR_OE) { - /* - * CHECK: does overrun affect the current character? - * ASSUMPTION: it does not. - */ - *tty->flip.flag_buf_ptr++ = flg; - *tty->flip.char_buf_ptr++ = ch; - tty->flip.count++; - if (tty->flip.count >= TTY_FLIPBUF_SIZE) - goto ignore_char; - ch = 0; - flg = TTY_OVERRUN; - } -#ifdef SUPPORT_SYSRQ - info->sysrq = 0; -#endif - goto error_return; -} - -static void ambauart_tx_chars(struct amba_info *info) -{ - struct amba_port *port = info->port; - int count; - - if (info->x_char) { - UART_PUT_CHAR(port, info->x_char); - info->state->icount.tx++; - info->x_char = 0; - return; - } - if (info->xmit.head == info->xmit.tail - || info->tty->stopped - || info->tty->hw_stopped) { - ambauart_disable_tx_interrupt(info); - return; - } - - count = port->fifosize; - do { - UART_PUT_CHAR(port, info->xmit.buf[info->xmit.tail]); - info->xmit.tail = (info->xmit.tail + 1) & (AMBA_XMIT_SIZE - 1); - info->state->icount.tx++; - if (info->xmit.head == info->xmit.tail) - break; - } while (--count > 0); - - if (CIRC_CNT(info->xmit.head, - info->xmit.tail, - AMBA_XMIT_SIZE) < WAKEUP_CHARS) - ambauart_event(info, EVT_WRITE_WAKEUP); - - if (info->xmit.head == info->xmit.tail) { - ambauart_disable_tx_interrupt(info); - } -} - -static void ambauart_modem_status(struct amba_info *info) -{ - unsigned int status, delta; - struct amba_icount *icount = &info->state->icount; - - status = UART_GET_FR(info->port) & AMBA_UARTFR_MODEM_ANY; - - delta = status ^ info->old_status; - info->old_status = status; - - if (!delta) - return; - - if (delta & AMBA_UARTFR_DCD) { - icount->dcd++; -#ifdef CONFIG_HARD_PPS - if ((info->flags & ASYNC_HARDPPS_CD) && - (status & AMBA_UARTFR_DCD) - hardpps(); -#endif - if (info->flags & ASYNC_CHECK_CD) { - if (status & AMBA_UARTFR_DCD) - wake_up_interruptible(&info->open_wait); - else if (!((info->flags & ASYNC_CALLOUT_ACTIVE) && - (info->flags & ASYNC_CALLOUT_NOHUP))) { - if (info->tty) - tty_hangup(info->tty); - } - } - } - - if (delta & AMBA_UARTFR_DSR) - icount->dsr++; - - if (delta & AMBA_UARTFR_CTS) { - icount->cts++; - - if (info->flags & ASYNC_CTS_FLOW) { - status &= AMBA_UARTFR_CTS; - - if (info->tty->hw_stopped) { - if (status) { - info->tty->hw_stopped = 0; - ambauart_enable_tx_interrupt(info); - ambauart_event(info, EVT_WRITE_WAKEUP); - } - } else { - if (!status) { - info->tty->hw_stopped = 1; - ambauart_disable_tx_interrupt(info); - } - } - } - } - wake_up_interruptible(&info->delta_msr_wait); - -} - -static void ambauart_int(int irq, void *dev_id, struct pt_regs *regs) -{ - struct amba_info *info = dev_id; - unsigned int status, pass_counter = 0; - -#if DEBUG_LEDS - // tell the world - set_leds(get_leds() | RED_LED); -#endif - - status = UART_GET_INT_STATUS(info->port); - do { - /* - * FIXME: what about clearing the interrupts? - */ - - if (status & (AMBA_UARTIIR_RTIS | AMBA_UARTIIR_RIS)) -#ifdef SUPPORT_SYSRQ - ambauart_rx_chars(info, regs); -#else - ambauart_rx_chars(info); -#endif - if (status & AMBA_UARTIIR_TIS) - ambauart_tx_chars(info); - if (status & AMBA_UARTIIR_MIS) - ambauart_modem_status(info); - if (pass_counter++ > AMBA_ISR_PASS_LIMIT) - break; - - status = UART_GET_INT_STATUS(info->port); - } while (status & (AMBA_UARTIIR_RTIS | AMBA_UARTIIR_RIS | AMBA_UARTIIR_TIS)); - -#if DEBUG_LEDS - // tell the world - set_leds(get_leds() & ~RED_LED); -#endif -} - -static void ambauart_tasklet_action(unsigned long data) -{ - struct amba_info *info = (struct amba_info *)data; - struct tty_struct *tty; - - tty = info->tty; - if (!tty || !test_and_clear_bit(EVT_WRITE_WAKEUP, &info->event)) - return; - - if ((tty->flags & (1 << TTY_DO_WRITE_WAKEUP)) && - tty->ldisc.write_wakeup) - (tty->ldisc.write_wakeup)(tty); - wake_up_interruptible(&tty->write_wait); -} - -static int ambauart_startup(struct amba_info *info) -{ - unsigned long flags; - unsigned long page; - int retval = 0; - - page = get_zeroed_page(GFP_KERNEL); - if (!page) - return -ENOMEM; - - save_flags(flags); cli(); - - if (info->flags & ASYNC_INITIALIZED) { - free_page(page); - goto errout; - } - - if (info->xmit.buf) - free_page(page); - else - info->xmit.buf = (unsigned char *) page; - - /* - * Allocate the IRQ - */ - retval = request_irq(info->port->irq, ambauart_int, 0, "amba", info); - if (retval) { - if (capable(CAP_SYS_ADMIN)) { - if (info->tty) - set_bit(TTY_IO_ERROR, &info->tty->flags); - retval = 0; - } - goto errout; - } - - info->mctrl = 0; - if (info->tty->termios->c_cflag & CBAUD) - info->mctrl = TIOCM_RTS | TIOCM_DTR; - info->port->set_mctrl(info->port, info->mctrl); - - /* - * initialise the old status of the modem signals - */ - info->old_status = UART_GET_FR(info->port) & AMBA_UARTFR_MODEM_ANY; - - /* - * Finally, enable interrupts - */ - ambauart_enable_rx_interrupt(info); - - if (info->tty) - clear_bit(TTY_IO_ERROR, &info->tty->flags); - info->xmit.head = info->xmit.tail = 0; - - /* - * Set up the tty->alt_speed kludge - */ - if (info->tty) { - if ((info->flags & ASYNC_SPD_MASK) == ASYNC_SPD_HI) - info->tty->alt_speed = 57600; - if ((info->flags & ASYNC_SPD_MASK) == ASYNC_SPD_VHI) - info->tty->alt_speed = 115200; - if ((info->flags & ASYNC_SPD_MASK) == ASYNC_SPD_SHI) - info->tty->alt_speed = 230400; - if ((info->flags & ASYNC_SPD_MASK) == ASYNC_SPD_WARP) - info->tty->alt_speed = 460800; - } - - /* - * and set the speed of the serial port - */ - ambauart_change_speed(info, 0); - - info->flags |= ASYNC_INITIALIZED; - restore_flags(flags); - return 0; - -errout: - restore_flags(flags); - return retval; -} - -/* - * This routine will shutdown a serial port; interrupts are disabled, and - * DTR is dropped if the hangup on close termio flag is on. - */ -static void ambauart_shutdown(struct amba_info *info) -{ - unsigned long flags; - - if (!(info->flags & ASYNC_INITIALIZED)) - return; - - save_flags(flags); cli(); /* Disable interrupts */ - - /* - * clear delta_msr_wait queue to avoid mem leaks: we may free the irq - * here so the queue might never be woken up - */ - wake_up_interruptible(&info->delta_msr_wait); - - /* - * Free the IRQ - */ - free_irq(info->port->irq, info); - - if (info->xmit.buf) { - unsigned long pg = (unsigned long) info->xmit.buf; - info->xmit.buf = NULL; - free_page(pg); - } - - /* - * disable all interrupts, disable the port - */ - UART_PUT_CR(info->port, 0); - - /* disable break condition and fifos */ - UART_PUT_LCRH(info->port, UART_GET_LCRH(info->port) & - ~(AMBA_UARTLCR_H_BRK | AMBA_UARTLCR_H_FEN)); - - if (!info->tty || (info->tty->termios->c_cflag & HUPCL)) - info->mctrl &= ~(TIOCM_DTR|TIOCM_RTS); - info->port->set_mctrl(info->port, info->mctrl); - - /* kill off our tasklet */ - tasklet_kill(&info->tlet); - if (info->tty) - set_bit(TTY_IO_ERROR, &info->tty->flags); - - info->flags &= ~ASYNC_INITIALIZED; - restore_flags(flags); -} - -static void ambauart_change_speed(struct amba_info *info, struct termios *old_termios) -{ - unsigned int lcr_h, baud, quot, cflag, old_cr, bits; - unsigned long flags; - - if (!info->tty || !info->tty->termios) - return; - - cflag = info->tty->termios->c_cflag; - -#if DEBUG - printk("ambauart_set_cflag(0x%x) called\n", cflag); -#endif - /* byte size and parity */ - switch (cflag & CSIZE) { - case CS5: lcr_h = AMBA_UARTLCR_H_WLEN_5; bits = 7; break; - case CS6: lcr_h = AMBA_UARTLCR_H_WLEN_6; bits = 8; break; - case CS7: lcr_h = AMBA_UARTLCR_H_WLEN_7; bits = 9; break; - default: lcr_h = AMBA_UARTLCR_H_WLEN_8; bits = 10; break; // CS8 - } - if (cflag & CSTOPB) { - lcr_h |= AMBA_UARTLCR_H_STP2; - bits ++; - } - if (cflag & PARENB) { - lcr_h |= AMBA_UARTLCR_H_PEN; - bits++; - if (!(cflag & PARODD)) - lcr_h |= AMBA_UARTLCR_H_EPS; - } - if (info->port->fifosize > 1) - lcr_h |= AMBA_UARTLCR_H_FEN; - - do { - /* Determine divisor based on baud rate */ - baud = tty_get_baud_rate(info->tty); - if (!baud) - baud = 9600; - - if (baud == 38400 && - ((info->flags & ASYNC_SPD_MASK) == ASYNC_SPD_CUST)) - quot = info->state->custom_divisor; - else - quot = (info->port->uartclk / (16 * baud)) - 1; - - if (!quot && old_termios) { - info->tty->termios->c_cflag &= ~CBAUD; - info->tty->termios->c_cflag |= (old_termios->c_cflag & CBAUD); - old_termios = NULL; - } - } while (quot == 0 && old_termios); - - /* As a last resort, if the quotient is zero, default to 9600 bps */ - if (!quot) - quot = (info->port->uartclk / (16 * 9600)) - 1; - - info->timeout = (info->port->fifosize * HZ * bits * quot) / - (info->port->uartclk / 16); - info->timeout += HZ/50; /* Add .02 seconds of slop */ - - if (cflag & CRTSCTS) - info->flags |= ASYNC_CTS_FLOW; - else - info->flags &= ~ASYNC_CTS_FLOW; - if (cflag & CLOCAL) - info->flags &= ~ASYNC_CHECK_CD; - else - info->flags |= ASYNC_CHECK_CD; - - /* - * Set up parity check flag - */ -#define RELEVENT_IFLAG(iflag) ((iflag) & (IGNBRK|BRKINT|IGNPAR|PARMRK|INPCK)) - - info->read_status_mask = AMBA_UARTRSR_OE; - if (I_INPCK(info->tty)) - info->read_status_mask |= AMBA_UARTRSR_FE | AMBA_UARTRSR_PE; - if (I_BRKINT(info->tty) || I_PARMRK(info->tty)) - info->read_status_mask |= AMBA_UARTRSR_BE; - - /* - * Characters to ignore - */ - info->ignore_status_mask = 0; - if (I_IGNPAR(info->tty)) - info->ignore_status_mask |= AMBA_UARTRSR_FE | AMBA_UARTRSR_PE; - if (I_IGNBRK(info->tty)) { - info->ignore_status_mask |= AMBA_UARTRSR_BE; - /* - * If we're ignoring parity and break indicators, - * ignore overruns to (for real raw support). - */ - if (I_IGNPAR(info->tty)) - info->ignore_status_mask |= AMBA_UARTRSR_OE; - } - - /* first, disable everything */ - save_flags(flags); cli(); - old_cr = UART_GET_CR(info->port) &= ~AMBA_UARTCR_MSIE; - - if ((info->flags & ASYNC_HARDPPS_CD) || - (cflag & CRTSCTS) || - !(cflag & CLOCAL)) - old_cr |= AMBA_UARTCR_MSIE; - - UART_PUT_CR(info->port, 0); - restore_flags(flags); - - /* Set baud rate */ - UART_PUT_LCRM(info->port, ((quot & 0xf00) >> 8)); - UART_PUT_LCRL(info->port, (quot & 0xff)); - - /* - * ----------v----------v----------v----------v----- - * NOTE: MUST BE WRITTEN AFTER UARTLCR_M & UARTLCR_L - * ----------^----------^----------^----------^----- - */ - UART_PUT_LCRH(info->port, lcr_h); - UART_PUT_CR(info->port, old_cr); -} - -static void ambauart_put_char(struct tty_struct *tty, u_char ch) -{ - struct amba_info *info = tty->driver_data; - unsigned long flags; - - if (!tty || !info->xmit.buf) - return; - - save_flags(flags); cli(); - if (CIRC_SPACE(info->xmit.head, info->xmit.tail, AMBA_XMIT_SIZE) != 0) { - info->xmit.buf[info->xmit.head] = ch; - info->xmit.head = (info->xmit.head + 1) & (AMBA_XMIT_SIZE - 1); - } - restore_flags(flags); -} - -static void ambauart_flush_chars(struct tty_struct *tty) -{ - struct amba_info *info = tty->driver_data; - unsigned long flags; - - if (info->xmit.head == info->xmit.tail - || tty->stopped - || tty->hw_stopped - || !info->xmit.buf) - return; - - save_flags(flags); cli(); - ambauart_enable_tx_interrupt(info); - restore_flags(flags); -} - -static int ambauart_write(struct tty_struct *tty, int from_user, - const u_char * buf, int count) -{ - struct amba_info *info = tty->driver_data; - unsigned long flags; - int c, ret = 0; - - if (!tty || !info->xmit.buf || !tmp_buf) - return 0; - - save_flags(flags); - if (from_user) { - down(&tmp_buf_sem); - while (1) { - int c1; - c = CIRC_SPACE_TO_END(info->xmit.head, - info->xmit.tail, - AMBA_XMIT_SIZE); - if (count < c) - c = count; - if (c <= 0) - break; - - c -= copy_from_user(tmp_buf, buf, c); - if (!c) { - if (!ret) - ret = -EFAULT; - break; - } - cli(); - c1 = CIRC_SPACE_TO_END(info->xmit.head, - info->xmit.tail, - AMBA_XMIT_SIZE); - if (c1 < c) - c = c1; - memcpy(info->xmit.buf + info->xmit.head, tmp_buf, c); - info->xmit.head = (info->xmit.head + c) & - (AMBA_XMIT_SIZE - 1); - restore_flags(flags); - buf += c; - count -= c; - ret += c; - } - up(&tmp_buf_sem); - } else { - cli(); - while (1) { - c = CIRC_SPACE_TO_END(info->xmit.head, - info->xmit.tail, - AMBA_XMIT_SIZE); - if (count < c) - c = count; - if (c <= 0) - break; - memcpy(info->xmit.buf + info->xmit.head, buf, c); - info->xmit.head = (info->xmit.head + c) & - (AMBA_XMIT_SIZE - 1); - buf += c; - count -= c; - ret += c; - } - restore_flags(flags); - } - if (info->xmit.head != info->xmit.tail - && !tty->stopped - && !tty->hw_stopped) - ambauart_enable_tx_interrupt(info); - return ret; -} - -static int ambauart_write_room(struct tty_struct *tty) -{ - struct amba_info *info = tty->driver_data; - - return CIRC_SPACE(info->xmit.head, info->xmit.tail, AMBA_XMIT_SIZE); -} - -static int ambauart_chars_in_buffer(struct tty_struct *tty) -{ - struct amba_info *info = tty->driver_data; - - return CIRC_CNT(info->xmit.head, info->xmit.tail, AMBA_XMIT_SIZE); -} - -static void ambauart_flush_buffer(struct tty_struct *tty) -{ - struct amba_info *info = tty->driver_data; - unsigned long flags; - -#if DEBUG - printk("ambauart_flush_buffer(%d) called\n", - MINOR(tty->device) - tty->driver.minor_start); -#endif - save_flags(flags); cli(); - info->xmit.head = info->xmit.tail = 0; - restore_flags(flags); - wake_up_interruptible(&tty->write_wait); - if ((tty->flags & (1 << TTY_DO_WRITE_WAKEUP)) && - tty->ldisc.write_wakeup) - (tty->ldisc.write_wakeup)(tty); -} - -/* - * This function is used to send a high-priority XON/XOFF character to - * the device - */ -static void ambauart_send_xchar(struct tty_struct *tty, char ch) -{ - struct amba_info *info = tty->driver_data; - - info->x_char = ch; - if (ch) - ambauart_enable_tx_interrupt(info); -} - -static void ambauart_throttle(struct tty_struct *tty) -{ - struct amba_info *info = tty->driver_data; - unsigned long flags; - - if (I_IXOFF(tty)) - ambauart_send_xchar(tty, STOP_CHAR(tty)); - - if (tty->termios->c_cflag & CRTSCTS) { - save_flags(flags); cli(); - info->mctrl &= ~TIOCM_RTS; - info->port->set_mctrl(info->port, info->mctrl); - restore_flags(flags); - } -} - -static void ambauart_unthrottle(struct tty_struct *tty) -{ - struct amba_info *info = (struct amba_info *) tty->driver_data; - unsigned long flags; - - if (I_IXOFF(tty)) { - if (info->x_char) - info->x_char = 0; - else - ambauart_send_xchar(tty, START_CHAR(tty)); - } - - if (tty->termios->c_cflag & CRTSCTS) { - save_flags(flags); cli(); - info->mctrl |= TIOCM_RTS; - info->port->set_mctrl(info->port, info->mctrl); - restore_flags(flags); - } -} - -static int get_serial_info(struct amba_info *info, struct serial_struct *retinfo) -{ - struct amba_state *state = info->state; - struct amba_port *port = info->port; - struct serial_struct tmp; - - memset(&tmp, 0, sizeof(tmp)); - tmp.type = 0; - tmp.line = state->line; - tmp.port = port->uart_base; - if (HIGH_BITS_OFFSET) - tmp.port_high = port->uart_base >> HIGH_BITS_OFFSET; - tmp.irq = port->irq; - tmp.flags = 0; - tmp.xmit_fifo_size = port->fifosize; - tmp.baud_base = port->uartclk / 16; - tmp.close_delay = state->close_delay; - tmp.closing_wait = state->closing_wait; - tmp.custom_divisor = state->custom_divisor; - - if (copy_to_user(retinfo, &tmp, sizeof(*retinfo))) - return -EFAULT; - return 0; -} - -static int set_serial_info(struct amba_info *info, - struct serial_struct *newinfo) -{ - struct serial_struct new_serial; - struct amba_state *state, old_state; - struct amba_port *port; - unsigned long new_port; - unsigned int i, change_irq, change_port; - int retval = 0; - - if (copy_from_user(&new_serial, newinfo, sizeof(new_serial))) - return -EFAULT; - - state = info->state; - old_state = *state; - port = info->port; - - new_port = new_serial.port; - if (HIGH_BITS_OFFSET) - new_port += (unsigned long) new_serial.port_high << HIGH_BITS_OFFSET; - - change_irq = new_serial.irq != port->irq; - change_port = new_port != port->uart_base; - - if (!capable(CAP_SYS_ADMIN)) { - if (change_irq || change_port || - (new_serial.baud_base != port->uartclk / 16) || - (new_serial.close_delay != state->close_delay) || - (new_serial.xmit_fifo_size != port->fifosize) || - ((new_serial.flags & ~ASYNC_USR_MASK) != - (state->flags & ~ASYNC_USR_MASK))) - return -EPERM; - state->flags = ((state->flags & ~ASYNC_USR_MASK) | - (new_serial.flags & ASYNC_USR_MASK)); - info->flags = ((info->flags & ~ASYNC_USR_MASK) | - (new_serial.flags & ASYNC_USR_MASK)); - state->custom_divisor = new_serial.custom_divisor; - goto check_and_exit; - } - - if ((new_serial.irq >= NR_IRQS) || (new_serial.irq < 0) || - (new_serial.baud_base < 9600)) - return -EINVAL; - - if (new_serial.type && change_port) { - for (i = 0; i < SERIAL_AMBA_NR; i++) - if ((port != amba_ports + i) && - amba_ports[i].uart_base != new_port) - return -EADDRINUSE; - } - - if ((change_port || change_irq) && (state->count > 1)) - return -EBUSY; - - /* - * OK, past this point, all the error checking has been done. - * At this point, we start making changes..... - */ - port->uartclk = new_serial.baud_base * 16; - state->flags = ((state->flags & ~ASYNC_FLAGS) | - (new_serial.flags & ASYNC_FLAGS)); - info->flags = ((state->flags & ~ASYNC_INTERNAL_FLAGS) | - (info->flags & ASYNC_INTERNAL_FLAGS)); - state->custom_divisor = new_serial.custom_divisor; - state->close_delay = new_serial.close_delay * HZ / 100; - state->closing_wait = new_serial.closing_wait * HZ / 100; - info->tty->low_latency = (info->flags & ASYNC_LOW_LATENCY) ? 1 : 0; - port->fifosize = new_serial.xmit_fifo_size; - - if (change_port || change_irq) { - /* - * We need to shutdown the serial port at the old - * port/irq combination. - */ - ambauart_shutdown(info); - port->irq = new_serial.irq; - port->uart_base = new_port; - } - -check_and_exit: - if (!port->uart_base) - return 0; - if (info->flags & ASYNC_INITIALIZED) { - if ((old_state.flags & ASYNC_SPD_MASK) != - (state->flags & ASYNC_SPD_MASK) || - (old_state.custom_divisor != state->custom_divisor)) { - if ((state->flags & ASYNC_SPD_MASK) == ASYNC_SPD_HI) - info->tty->alt_speed = 57600; - if ((state->flags & ASYNC_SPD_MASK) == ASYNC_SPD_VHI) - info->tty->alt_speed = 115200; - if ((state->flags & ASYNC_SPD_MASK) == ASYNC_SPD_SHI) - info->tty->alt_speed = 230400; - if ((state->flags & ASYNC_SPD_MASK) == ASYNC_SPD_WARP) - info->tty->alt_speed = 460800; - ambauart_change_speed(info, NULL); - } - } else - retval = ambauart_startup(info); - return retval; -} - - -/* - * get_lsr_info - get line status register info - */ -static int get_lsr_info(struct amba_info *info, unsigned int *value) -{ - unsigned int result, status; - unsigned long flags; - - save_flags(flags); cli(); - status = UART_GET_FR(info->port); - restore_flags(flags); - result = status & AMBA_UARTFR_BUSY ? TIOCSER_TEMT : 0; - - /* - * If we're about to load something into the transmit - * register, we'll pretend the transmitter isn't empty to - * avoid a race condition (depending on when the transmit - * interrupt happens). - */ - if (info->x_char || - ((CIRC_CNT(info->xmit.head, info->xmit.tail, - AMBA_XMIT_SIZE) > 0) && - !info->tty->stopped && !info->tty->hw_stopped)) - result &= TIOCSER_TEMT; - - return put_user(result, value); -} - -static int get_modem_info(struct amba_info *info, unsigned int *value) -{ - unsigned int result = info->mctrl; - unsigned int status; - - status = UART_GET_FR(info->port); - if (status & AMBA_UARTFR_DCD) - result |= TIOCM_CAR; - if (status & AMBA_UARTFR_DSR) - result |= TIOCM_DSR; - if (status & AMBA_UARTFR_CTS) - result |= TIOCM_CTS; - - return put_user(result, value); -} - -static int set_modem_info(struct amba_info *info, unsigned int cmd, - unsigned int *value) -{ - unsigned int arg, old; - unsigned long flags; - - if (get_user(arg, value)) - return -EFAULT; - - old = info->mctrl; - switch (cmd) { - case TIOCMBIS: - info->mctrl |= arg; - break; - - case TIOCMBIC: - info->mctrl &= ~arg; - break; - - case TIOCMSET: - info->mctrl = arg; - break; - - default: - return -EINVAL; - } - save_flags(flags); cli(); - if (old != info->mctrl) - info->port->set_mctrl(info->port, info->mctrl); - restore_flags(flags); - return 0; -} - -static void ambauart_break_ctl(struct tty_struct *tty, int break_state) -{ - struct amba_info *info = tty->driver_data; - unsigned long flags; - unsigned int lcr_h; - - save_flags(flags); cli(); - lcr_h = UART_GET_LCRH(info->port); - if (break_state == -1) - lcr_h |= AMBA_UARTLCR_H_BRK; - else - lcr_h &= ~AMBA_UARTLCR_H_BRK; - UART_PUT_LCRH(info->port, lcr_h); - restore_flags(flags); -} - -static int ambauart_ioctl(struct tty_struct *tty, struct file *file, - unsigned int cmd, unsigned long arg) -{ - struct amba_info *info = tty->driver_data; - struct amba_icount cprev, cnow; - struct serial_icounter_struct icount; - unsigned long flags; - - if ((cmd != TIOCGSERIAL) && (cmd != TIOCSSERIAL) && - (cmd != TIOCSERCONFIG) && (cmd != TIOCSERGSTRUCT) && - (cmd != TIOCMIWAIT) && (cmd != TIOCGICOUNT)) { - if (tty->flags & (1 << TTY_IO_ERROR)) - return -EIO; - } - - switch (cmd) { - case TIOCMGET: - return get_modem_info(info, (unsigned int *)arg); - case TIOCMBIS: - case TIOCMBIC: - case TIOCMSET: - return set_modem_info(info, cmd, (unsigned int *)arg); - case TIOCGSERIAL: - return get_serial_info(info, - (struct serial_struct *)arg); - case TIOCSSERIAL: - return set_serial_info(info, - (struct serial_struct *)arg); - case TIOCSERGETLSR: /* Get line status register */ - return get_lsr_info(info, (unsigned int *)arg); - /* - * Wait for any of the 4 modem inputs (DCD,RI,DSR,CTS) to change - * - mask passed in arg for lines of interest - * (use |'ed TIOCM_RNG/DSR/CD/CTS for masking) - * Caller should use TIOCGICOUNT to see which one it was - */ - case TIOCMIWAIT: - save_flags(flags); cli(); - /* note the counters on entry */ - cprev = info->state->icount; - /* Force modem status interrupts on */ - UART_PUT_CR(info->port, UART_GET_CR(info->port) | AMBA_UARTCR_MSIE); - restore_flags(flags); - while (1) { - interruptible_sleep_on(&info->delta_msr_wait); - /* see if a signal did it */ - if (signal_pending(current)) - return -ERESTARTSYS; - save_flags(flags); cli(); - cnow = info->state->icount; /* atomic copy */ - restore_flags(flags); - if (cnow.rng == cprev.rng && cnow.dsr == cprev.dsr && - cnow.dcd == cprev.dcd && cnow.cts == cprev.cts) - return -EIO; /* no change => error */ - if (((arg & TIOCM_RNG) && (cnow.rng != cprev.rng)) || - ((arg & TIOCM_DSR) && (cnow.dsr != cprev.dsr)) || - ((arg & TIOCM_CD) && (cnow.dcd != cprev.dcd)) || - ((arg & TIOCM_CTS) && (cnow.cts != cprev.cts))) { - return 0; - } - cprev = cnow; - } - /* NOTREACHED */ - - /* - * Get counter of input serial line interrupts (DCD,RI,DSR,CTS) - * Return: write counters to the user passed counter struct - * NB: both 1->0 and 0->1 transitions are counted except for - * RI where only 0->1 is counted. - */ - case TIOCGICOUNT: - save_flags(flags); cli(); - cnow = info->state->icount; - restore_flags(flags); - icount.cts = cnow.cts; - icount.dsr = cnow.dsr; - icount.rng = cnow.rng; - icount.dcd = cnow.dcd; - icount.rx = cnow.rx; - icount.tx = cnow.tx; - icount.frame = cnow.frame; - icount.overrun = cnow.overrun; - icount.parity = cnow.parity; - icount.brk = cnow.brk; - icount.buf_overrun = cnow.buf_overrun; - - return copy_to_user((void *)arg, &icount, sizeof(icount)) - ? -EFAULT : 0; - - default: - return -ENOIOCTLCMD; - } - return 0; -} - -static void ambauart_set_termios(struct tty_struct *tty, struct termios *old_termios) -{ - struct amba_info *info = tty->driver_data; - unsigned long flags; - unsigned int cflag = tty->termios->c_cflag; - - if ((cflag ^ old_termios->c_cflag) == 0 && - RELEVENT_IFLAG(tty->termios->c_iflag ^ old_termios->c_iflag) == 0) - return; - - ambauart_change_speed(info, old_termios); - - /* Handle transition to B0 status */ - if ((old_termios->c_cflag & CBAUD) && - !(cflag & CBAUD)) { - save_flags(flags); cli(); - info->mctrl &= ~(TIOCM_RTS | TIOCM_DTR); - info->port->set_mctrl(info->port, info->mctrl); - restore_flags(flags); - } - - /* Handle transition away from B0 status */ - if (!(old_termios->c_cflag & CBAUD) && - (cflag & CBAUD)) { - save_flags(flags); cli(); - info->mctrl |= TIOCM_DTR; - if (!(cflag & CRTSCTS) || - !test_bit(TTY_THROTTLED, &tty->flags)) - info->mctrl |= TIOCM_RTS; - info->port->set_mctrl(info->port, info->mctrl); - restore_flags(flags); - } - - /* Handle turning off CRTSCTS */ - if ((old_termios->c_cflag & CRTSCTS) && - !(cflag & CRTSCTS)) { - tty->hw_stopped = 0; - ambauart_start(tty); - } - -#if 0 - /* - * No need to wake up processes in open wait, since they - * sample the CLOCAL flag once, and don't recheck it. - * XXX It's not clear whether the current behavior is correct - * or not. Hence, this may change..... - */ - if (!(old_termios->c_cflag & CLOCAL) && - (tty->termios->c_cflag & CLOCAL)) - wake_up_interruptible(&info->open_wait); -#endif -} - -static void ambauart_close(struct tty_struct *tty, struct file *filp) -{ - struct amba_info *info = tty->driver_data; - struct amba_state *state; - unsigned long flags; - - if (!info) - return; - - state = info->state; - -#if DEBUG - printk("ambauart_close() called\n"); -#endif - - save_flags(flags); cli(); - - if (tty_hung_up_p(filp)) { - MOD_DEC_USE_COUNT; - restore_flags(flags); - return; - } - - if ((tty->count == 1) && (state->count != 1)) { - /* - * Uh, oh. tty->count is 1, which means that the tty - * structure will be freed. state->count should always - * be one in these conditions. If it's greater than - * one, we've got real problems, since it means the - * serial port won't be shutdown. - */ - printk("ambauart_close: bad serial port count; tty->count is 1, " - "state->count is %d\n", state->count); - state->count = 1; - } - if (--state->count < 0) { - printk("rs_close: bad serial port count for %s%d: %d\n", - tty->driver.name, info->state->line, state->count); - state->count = 0; - } - if (state->count) { - MOD_DEC_USE_COUNT; - restore_flags(flags); - return; - } - info->flags |= ASYNC_CLOSING; - restore_flags(flags); - /* - * Save the termios structure, since this port may have - * separate termios for callout and dialin. - */ - if (info->flags & ASYNC_NORMAL_ACTIVE) - info->state->normal_termios = *tty->termios; - if (info->flags & ASYNC_CALLOUT_ACTIVE) - info->state->callout_termios = *tty->termios; - /* - * Now we wait for the transmit buffer to clear; and we notify - * the line discipline to only process XON/XOFF characters. - */ - tty->closing = 1; - if (info->state->closing_wait != ASYNC_CLOSING_WAIT_NONE) - tty_wait_until_sent(tty, info->state->closing_wait); - /* - * At this point, we stop accepting input. To do this, we - * disable the receive line status interrupts. - */ - if (info->flags & ASYNC_INITIALIZED) { - ambauart_disable_rx_interrupt(info); - /* - * Before we drop DTR, make sure the UART transmitter - * has completely drained; this is especially - * important if there is a transmit FIFO! - */ - ambauart_wait_until_sent(tty, info->timeout); - } - ambauart_shutdown(info); - if (tty->driver.flush_buffer) - tty->driver.flush_buffer(tty); - if (tty->ldisc.flush_buffer) - tty->ldisc.flush_buffer(tty); - tty->closing = 0; - info->event = 0; - info->tty = NULL; - if (info->blocked_open) { - if (info->state->close_delay) { - set_current_state(TASK_INTERRUPTIBLE); - schedule_timeout(info->state->close_delay); - } - wake_up_interruptible(&info->open_wait); - } - info->flags &= ~(ASYNC_NORMAL_ACTIVE|ASYNC_CALLOUT_ACTIVE| - ASYNC_CLOSING); - wake_up_interruptible(&info->close_wait); - MOD_DEC_USE_COUNT; -} - -static void ambauart_wait_until_sent(struct tty_struct *tty, int timeout) -{ - struct amba_info *info = (struct amba_info *) tty->driver_data; - unsigned long char_time, expire; - unsigned int status; - - if (info->port->fifosize == 0) - return; - - /* - * Set the check interval to be 1/5 of the estimated time to - * send a single character, and make it at least 1. The check - * interval should also be less than the timeout. - * - * Note: we have to use pretty tight timings here to satisfy - * the NIST-PCTS. - */ - char_time = (info->timeout - HZ/50) / info->port->fifosize; - char_time = char_time / 5; - if (char_time == 0) - char_time = 1; - if (timeout && timeout < char_time) - char_time = timeout; - /* - * If the transmitter hasn't cleared in twice the approximate - * amount of time to send the entire FIFO, it probably won't - * ever clear. This assumes the UART isn't doing flow - * control, which is currently the case. Hence, if it ever - * takes longer than info->timeout, this is probably due to a - * UART bug of some kind. So, we clamp the timeout parameter at - * 2*info->timeout. - */ - if (!timeout || timeout > 2 * info->timeout) - timeout = 2 * info->timeout; - - expire = jiffies + timeout; -#if DEBUG - printk("ambauart_wait_until_sent(%d), jiff=%lu, expire=%lu...\n", - MINOR(tty->device) - tty->driver.minor_start, jiffies, - expire); -#endif - while (UART_GET_FR(info->port) & AMBA_UARTFR_BUSY) { - set_current_state(TASK_INTERRUPTIBLE); - schedule_timeout(char_time); - if (signal_pending(current)) - break; - if (timeout && time_after(jiffies, expire)) - break; - status = UART_GET_FR(info->port); - } - set_current_state(TASK_RUNNING); -} - -static void ambauart_hangup(struct tty_struct *tty) -{ - struct amba_info *info = tty->driver_data; - struct amba_state *state = info->state; - - ambauart_flush_buffer(tty); - if (info->flags & ASYNC_CLOSING) - return; - ambauart_shutdown(info); - info->event = 0; - state->count = 0; - info->flags &= ~(ASYNC_NORMAL_ACTIVE|ASYNC_CALLOUT_ACTIVE); - info->tty = NULL; - wake_up_interruptible(&info->open_wait); -} - -static int block_til_ready(struct tty_struct *tty, struct file *filp, - struct amba_info *info) -{ - DECLARE_WAITQUEUE(wait, current); - struct amba_state *state = info->state; - unsigned long flags; - int do_clocal = 0, extra_count = 0, retval; - - /* - * If the device is in the middle of being closed, then block - * until it's done, and then try again. - */ - if (tty_hung_up_p(filp) || - (info->flags & ASYNC_CLOSING)) { - if (info->flags & ASYNC_CLOSING) - interruptible_sleep_on(&info->close_wait); - return (info->flags & ASYNC_HUP_NOTIFY) ? - -EAGAIN : -ERESTARTSYS; - } - - /* - * If this is a callout device, then just make sure the normal - * device isn't being used. - */ - if (tty->driver.subtype == SERIAL_TYPE_CALLOUT) { - if (info->flags & ASYNC_NORMAL_ACTIVE) - return -EBUSY; - if ((info->flags & ASYNC_CALLOUT_ACTIVE) && - (info->flags & ASYNC_SESSION_LOCKOUT) && - (info->session != current->session)) - return -EBUSY; - if ((info->flags & ASYNC_CALLOUT_ACTIVE) && - (info->flags & ASYNC_PGRP_LOCKOUT) && - (info->pgrp != current->pgrp)) - return -EBUSY; - info->flags |= ASYNC_CALLOUT_ACTIVE; - return 0; - } - - /* - * If non-blocking mode is set, or the port is not enabled, - * then make the check up front and then exit. - */ - if ((filp->f_flags & O_NONBLOCK) || - (tty->flags & (1 << TTY_IO_ERROR))) { - if (info->flags & ASYNC_CALLOUT_ACTIVE) - return -EBUSY; - info->flags |= ASYNC_NORMAL_ACTIVE; - return 0; - } - - if (info->flags & ASYNC_CALLOUT_ACTIVE) { - if (state->normal_termios.c_cflag & CLOCAL) - do_clocal = 1; - } else { - if (tty->termios->c_cflag & CLOCAL) - do_clocal = 1; - } - - /* - * Block waiting for the carrier detect and the line to become - * free (i.e., not in use by the callout). While we are in - * this loop, state->count is dropped by one, so that - * rs_close() knows when to free things. We restore it upon - * exit, either normal or abnormal. - */ - retval = 0; - add_wait_queue(&info->open_wait, &wait); - save_flags(flags); cli(); - if (!tty_hung_up_p(filp)) { - extra_count = 1; - state->count--; - } - restore_flags(flags); - info->blocked_open++; - while (1) { - save_flags(flags); cli(); - if (!(info->flags & ASYNC_CALLOUT_ACTIVE) && - (tty->termios->c_cflag & CBAUD)) { - info->mctrl = TIOCM_DTR | TIOCM_RTS; - info->port->set_mctrl(info->port, info->mctrl); - } - restore_flags(flags); - set_current_state(TASK_INTERRUPTIBLE); - if (tty_hung_up_p(filp) || - !(info->flags & ASYNC_INITIALIZED)) { - if (info->flags & ASYNC_HUP_NOTIFY) - retval = -EAGAIN; - else - retval = -ERESTARTSYS; - break; - } - if (!(info->flags & ASYNC_CALLOUT_ACTIVE) && - !(info->flags & ASYNC_CLOSING) && - (do_clocal || (UART_GET_FR(info->port) & AMBA_UARTFR_DCD))) - break; - if (signal_pending(current)) { - retval = -ERESTARTSYS; - break; - } - schedule(); - } - set_current_state(TASK_RUNNING); - remove_wait_queue(&info->open_wait, &wait); - if (extra_count) - state->count++; - info->blocked_open--; - if (retval) - return retval; - info->flags |= ASYNC_NORMAL_ACTIVE; - return 0; -} - -static struct amba_info *ambauart_get(int line) -{ - struct amba_info *info; - struct amba_state *state = amba_state + line; - - state->count++; - if (state->info) - return state->info; - info = kmalloc(sizeof(struct amba_info), GFP_KERNEL); - if (info) { - memset(info, 0, sizeof(struct amba_info)); - init_waitqueue_head(&info->open_wait); - init_waitqueue_head(&info->close_wait); - init_waitqueue_head(&info->delta_msr_wait); - info->flags = state->flags; - info->state = state; - info->port = amba_ports + line; - tasklet_init(&info->tlet, ambauart_tasklet_action, - (unsigned long)info); - } - if (state->info) { - kfree(info); - return state->info; - } - state->info = info; - return info; -} - -static int ambauart_open(struct tty_struct *tty, struct file *filp) -{ - struct amba_info *info; - int retval, line = MINOR(tty->device) - tty->driver.minor_start; - -#if DEBUG - printk("ambauart_open(%d) called\n", line); -#endif - - // is this a line that we've got? - MOD_INC_USE_COUNT; - if (line >= SERIAL_AMBA_NR) { - MOD_DEC_USE_COUNT; - return -ENODEV; - } - - info = ambauart_get(line); - if (!info) - return -ENOMEM; - - tty->driver_data = info; - info->tty = tty; - info->tty->low_latency = (info->flags & ASYNC_LOW_LATENCY) ? 1 : 0; - - /* - * Make sure we have the temporary buffer allocated - */ - if (!tmp_buf) { - unsigned long page = get_zeroed_page(GFP_KERNEL); - if (tmp_buf) - free_page(page); - else if (!page) { - MOD_DEC_USE_COUNT; - return -ENOMEM; - } - tmp_buf = (u_char *)page; - } - - /* - * If the port is in the middle of closing, bail out now. - */ - if (tty_hung_up_p(filp) || - (info->flags & ASYNC_CLOSING)) { - if (info->flags & ASYNC_CLOSING) - interruptible_sleep_on(&info->close_wait); - MOD_DEC_USE_COUNT; - return -EAGAIN; - } - - /* - * Start up the serial port - */ - retval = ambauart_startup(info); - if (retval) { - MOD_DEC_USE_COUNT; - return retval; - } - - retval = block_til_ready(tty, filp, info); - if (retval) { - MOD_DEC_USE_COUNT; - return retval; - } - - if ((info->state->count == 1) && - (info->flags & ASYNC_SPLIT_TERMIOS)) { - if (tty->driver.subtype == SERIAL_TYPE_NORMAL) - *tty->termios = info->state->normal_termios; - else - *tty->termios = info->state->callout_termios; - } -#ifdef CONFIG_SERIAL_AMBA_CONSOLE - if (ambauart_cons.cflag && ambauart_cons.index == line) { - tty->termios->c_cflag = ambauart_cons.cflag; - ambauart_cons.cflag = 0; - } -#endif - ambauart_change_speed(info, NULL); - info->session = current->session; - info->pgrp = current->pgrp; - return 0; -} - -int __init ambauart_init(void) -{ - int i; - - ambanormal_driver.magic = TTY_DRIVER_MAGIC; - ambanormal_driver.driver_name = "serial_amba"; - ambanormal_driver.name = SERIAL_AMBA_NAME; - ambanormal_driver.major = SERIAL_AMBA_MAJOR; - ambanormal_driver.minor_start = SERIAL_AMBA_MINOR; - ambanormal_driver.num = SERIAL_AMBA_NR; - ambanormal_driver.type = TTY_DRIVER_TYPE_SERIAL; - ambanormal_driver.subtype = SERIAL_TYPE_NORMAL; - ambanormal_driver.init_termios = tty_std_termios; - ambanormal_driver.init_termios.c_cflag = B38400 | CS8 | CREAD | HUPCL | CLOCAL; - ambanormal_driver.flags = TTY_DRIVER_REAL_RAW | TTY_DRIVER_NO_DEVFS; - ambanormal_driver.refcount = &ambauart_refcount; - ambanormal_driver.table = ambauart_table; - ambanormal_driver.termios = ambauart_termios; - ambanormal_driver.termios_locked = ambauart_termios_locked; - - ambanormal_driver.open = ambauart_open; - ambanormal_driver.close = ambauart_close; - ambanormal_driver.write = ambauart_write; - ambanormal_driver.put_char = ambauart_put_char; - ambanormal_driver.flush_chars = ambauart_flush_chars; - ambanormal_driver.write_room = ambauart_write_room; - ambanormal_driver.chars_in_buffer = ambauart_chars_in_buffer; - ambanormal_driver.flush_buffer = ambauart_flush_buffer; - ambanormal_driver.ioctl = ambauart_ioctl; - ambanormal_driver.throttle = ambauart_throttle; - ambanormal_driver.unthrottle = ambauart_unthrottle; - ambanormal_driver.send_xchar = ambauart_send_xchar; - ambanormal_driver.set_termios = ambauart_set_termios; - ambanormal_driver.stop = ambauart_stop; - ambanormal_driver.start = ambauart_start; - ambanormal_driver.hangup = ambauart_hangup; - ambanormal_driver.break_ctl = ambauart_break_ctl; - ambanormal_driver.wait_until_sent = ambauart_wait_until_sent; - ambanormal_driver.read_proc = NULL; - - /* - * The callout device is just like the normal device except for - * the major number and the subtype code. - */ - ambacallout_driver = ambanormal_driver; - ambacallout_driver.name = CALLOUT_AMBA_NAME; - ambacallout_driver.major = CALLOUT_AMBA_MAJOR; - ambacallout_driver.subtype = SERIAL_TYPE_CALLOUT; - ambacallout_driver.read_proc = NULL; - ambacallout_driver.proc_entry = NULL; - - if (tty_register_driver(&ambanormal_driver)) - panic("Couldn't register AMBA serial driver\n"); - if (tty_register_driver(&ambacallout_driver)) - panic("Couldn't register AMBA callout driver\n"); - - for (i = 0; i < SERIAL_AMBA_NR; i++) { - struct amba_state *state = amba_state + i; - state->line = i; - state->close_delay = 5 * HZ / 10; - state->closing_wait = 30 * HZ; - state->callout_termios = ambacallout_driver.init_termios; - state->normal_termios = ambanormal_driver.init_termios; - } - - return 0; -} - -__initcall(ambauart_init); - -#ifdef CONFIG_SERIAL_AMBA_CONSOLE -/************** console driver *****************/ - -/* - * This code is currently never used; console->read is never called. - * Therefore, although we have an implementation, we don't use it. - * FIXME: the "const char *s" should be fixed to "char *s" some day. - * (when the definition in include/linux/console.h is also fixed) - */ -#ifdef used_and_not_const_char_pointer -static int ambauart_console_read(struct console *co, const char *s, u_int count) -{ - struct amba_port *port = &amba_ports[co->index]; - unsigned int status; - char *w; - int c; -#if DEBUG - printk("ambauart_console_read() called\n"); -#endif - - c = 0; - w = s; - while (c < count) { - status = UART_GET_FR(port); - if (UART_RX_DATA(status)) { - *w++ = UART_GET_CHAR(port); - c++; - } else { - // nothing more to get, return - return c; - } - } - // return the count - return c; -} -#endif - -/* - * Print a string to the serial port trying not to disturb - * any possible real use of the port... - * - * The console must be locked when we get here. - */ -static void ambauart_console_write(struct console *co, const char *s, u_int count) -{ - struct amba_port *port = &amba_ports[co->index]; - unsigned int status, old_cr; - int i; - - /* - * First save the CR then disable the interrupts - */ - old_cr = UART_GET_CR(port); - UART_PUT_CR(port, AMBA_UARTCR_UARTEN); - - /* - * Now, do each character - */ - for (i = 0; i < count; i++) { - do { - status = UART_GET_FR(port); - } while (!UART_TX_READY(status)); - UART_PUT_CHAR(port, s[i]); - if (s[i] == '\n') { - do { - status = UART_GET_FR(port); - } while (!UART_TX_READY(status)); - UART_PUT_CHAR(port, '\r'); - } - } - - /* - * Finally, wait for transmitter to become empty - * and restore the TCR - */ - do { - status = UART_GET_FR(port); - } while (status & AMBA_UARTFR_BUSY); - UART_PUT_CR(port, old_cr); -} - -static kdev_t ambauart_console_device(struct console *c) -{ - return MKDEV(SERIAL_AMBA_MAJOR, SERIAL_AMBA_MINOR + c->index); -} - -static int __init ambauart_console_setup(struct console *co, char *options) -{ - struct amba_port *port; - int baud = 38400; - int bits = 8; - int parity = 'n'; - u_int cflag = CREAD | HUPCL | CLOCAL; - u_int lcr_h, quot; - - if (co->index >= SERIAL_AMBA_NR) - co->index = 0; - - port = &amba_ports[co->index]; - - if (options) { - char *s = options; - baud = simple_strtoul(s, NULL, 10); - while (*s >= '0' && *s <= '9') - s++; - if (*s) parity = *s++; - if (*s) bits = *s - '0'; - } - - /* - * Now construct a cflag setting. - */ - switch (baud) { - case 1200: cflag |= B1200; break; - case 2400: cflag |= B2400; break; - case 4800: cflag |= B4800; break; - default: cflag |= B9600; baud = 9600; break; - case 19200: cflag |= B19200; break; - case 38400: cflag |= B38400; break; - case 57600: cflag |= B57600; break; - case 115200: cflag |= B115200; break; - } - switch (bits) { - case 7: cflag |= CS7; lcr_h = AMBA_UARTLCR_H_WLEN_7; break; - default: cflag |= CS8; lcr_h = AMBA_UARTLCR_H_WLEN_8; break; - } - switch (parity) { - case 'o': - case 'O': cflag |= PARODD; lcr_h |= AMBA_UARTLCR_H_PEN; break; - case 'e': - case 'E': cflag |= PARENB; lcr_h |= AMBA_UARTLCR_H_PEN | - AMBA_UARTLCR_H_EPS; break; - } - - co->cflag = cflag; - - if (port->fifosize > 1) - lcr_h |= AMBA_UARTLCR_H_FEN; - - quot = (port->uartclk / (16 * baud)) - 1; - - UART_PUT_LCRL(port, (quot & 0xff)); - UART_PUT_LCRM(port, (quot >> 8)); - UART_PUT_LCRH(port, lcr_h); - - /* we will enable the port as we need it */ - UART_PUT_CR(port, 0); - - return 0; -} - -static struct console ambauart_cons = -{ - name: SERIAL_AMBA_NAME, - write: ambauart_console_write, -#ifdef used_and_not_const_char_pointer - read: ambauart_console_read, -#endif - device: ambauart_console_device, - setup: ambauart_console_setup, - flags: CON_PRINTBUFFER, - index: -1, -}; - -void __init ambauart_console_init(void) -{ - register_console(&ambauart_cons); -} - -#endif /* CONFIG_SERIAL_AMBA_CONSOLE */ - -MODULE_LICENSE("GPL"); -EXPORT_NO_SYMBOLS; diff -urN orig/drivers/char/tty_io.c linux/drivers/char/tty_io.c --- orig/drivers/char/tty_io.c Mon Aug 5 13:30:48 2002 +++ linux/drivers/char/tty_io.c Wed Jan 22 16:40:11 2003 @@ -154,14 +154,14 @@ extern void tty3215_init(void); extern void tub3270_con_init(void); extern void tub3270_init(void); -extern void rs285_console_init(void); -extern void sa1100_rs_console_init(void); +extern void uart_console_init(void); extern void sgi_serial_console_init(void); extern void sci_console_init(void); extern void tx3912_console_init(void); extern void tx3912_rs_init(void); extern void txx927_console_init(void); extern void sb1250_serial_console_init(void); +extern void rs285_console_init(void); #ifndef MIN #define MIN(a,b) ((a) < (b) ? (a) : (b)) @@ -2230,18 +2230,12 @@ #ifdef CONFIG_STDIO_CONSOLE stdio_console_init(); #endif -#ifdef CONFIG_SERIAL_21285_CONSOLE - rs285_console_init(); -#endif -#ifdef CONFIG_SERIAL_SA1100_CONSOLE - sa1100_rs_console_init(); +#ifdef CONFIG_SERIAL_CORE_CONSOLE + uart_console_init(); #endif #ifdef CONFIG_ARC_CONSOLE arc_console_init(); #endif -#ifdef CONFIG_SERIAL_AMBA_CONSOLE - ambauart_console_init(); -#endif #ifdef CONFIG_SERIAL_TX3912_CONSOLE tx3912_console_init(); #endif @@ -2250,6 +2244,9 @@ #endif #ifdef CONFIG_SIBYTE_SB1250_DUART_CONSOLE sb1250_serial_console_init(); +#endif +#ifdef CONFIG_SERIAL_21285_CONSOLE + rs285_console_init(); #endif } diff -urN orig/drivers/char/wdt285.c linux/drivers/char/wdt285.c --- orig/drivers/char/wdt285.c Mon Aug 5 13:30:48 2002 +++ linux/drivers/char/wdt285.c Mon Aug 5 13:43:32 2002 @@ -151,7 +151,7 @@ if (get_user(new_margin, (int *)arg)) return -EFAULT; /* Arbitrary, can't find the card's limits */ - if ((new_marg < 0) || (new_margin > 60)) + if ((new_margin < 0) || (new_margin > 60)) return -EINVAL; soft_margin = new_margin; watchdog_ping(); diff -urN orig/drivers/i2c/Config.in linux/drivers/i2c/Config.in --- orig/drivers/i2c/Config.in Wed Feb 27 14:24:58 2002 +++ linux/drivers/i2c/Config.in Thu Feb 27 23:20:12 2003 @@ -13,6 +13,14 @@ dep_tristate ' Philips style parallel port adapter' CONFIG_I2C_PHILIPSPAR $CONFIG_I2C_ALGOBIT $CONFIG_PARPORT dep_tristate ' ELV adapter' CONFIG_I2C_ELV $CONFIG_I2C_ALGOBIT dep_tristate ' Velleman K9000 adapter' CONFIG_I2C_VELLEMAN $CONFIG_I2C_ALGOBIT + dep_tristate ' Guide GPIO adapter' CONFIG_I2C_GUIDE $CONFIG_I2C_ALGOBIT + if [ "$CONFIG_ARCH_OMAHA" = "y" ]; then + dep_tristate ' Omaha I2C interface' CONFIG_I2C_OMAHA $CONFIG_I2C + fi + if [ "$CONFIG_ARCH_SA1100" = "y" ]; then + dep_tristate ' Frodo I2C adapter' CONFIG_I2C_FRODO $CONFIG_I2C_ALGOBIT + dep_tristate ' SA1100 I2C GPIO adapter' CONFIG_I2C_BIT_SA1100_GPIO $CONFIG_I2C_ALGOBIT + fi fi dep_tristate 'I2C PCF 8584 interfaces' CONFIG_I2C_ALGOPCF $CONFIG_I2C diff -urN orig/drivers/i2c/Makefile linux/drivers/i2c/Makefile --- orig/drivers/i2c/Makefile Wed Feb 27 14:24:58 2002 +++ linux/drivers/i2c/Makefile Thu Feb 27 23:20:12 2003 @@ -7,18 +7,28 @@ export-objs := i2c-core.o i2c-algo-bit.o i2c-algo-pcf.o \ i2c-algo-ite.o i2c-proc.o +# Init order: core, chardev, bit adapters, pcf adapters + obj-$(CONFIG_I2C) += i2c-core.o obj-$(CONFIG_I2C_CHARDEV) += i2c-dev.o + +# Bit adapters obj-$(CONFIG_I2C_ALGOBIT) += i2c-algo-bit.o obj-$(CONFIG_I2C_PHILIPSPAR) += i2c-philips-par.o obj-$(CONFIG_I2C_ELV) += i2c-elv.o obj-$(CONFIG_I2C_VELLEMAN) += i2c-velleman.o +obj-$(CONFIG_I2C_GUIDE) += i2c-guide.o +obj-$(CONFIG_I2C_FRODO) += i2c-frodo.o +obj-$(CONFIG_I2C_OMAHA) += i2c-omaha.o + +# PCF adapters obj-$(CONFIG_I2C_ALGOPCF) += i2c-algo-pcf.o obj-$(CONFIG_I2C_ELEKTOR) += i2c-elektor.o obj-$(CONFIG_ITE_I2C_ALGO) += i2c-algo-ite.o obj-$(CONFIG_ITE_I2C_ADAP) += i2c-adap-ite.o obj-$(CONFIG_I2C_PROC) += i2c-proc.o obj-$(CONFIG_I2C_KEYWEST) += i2c-keywest.o + # This is needed for automatic patch generation: sensors code starts here # This is needed for automatic patch generation: sensors code ends here diff -urN orig/drivers/i2c/i2c-algo-bit.c linux/drivers/i2c/i2c-algo-bit.c --- orig/drivers/i2c/i2c-algo-bit.c Mon Aug 5 13:30:49 2002 +++ linux/drivers/i2c/i2c-algo-bit.c Thu Feb 27 23:20:12 2003 @@ -170,7 +170,14 @@ * 1 if the device acknowledged * 0 if the device did not ack * -ETIMEDOUT if an error occurred (while raising the scl line) - */ + + * tsong@iders.ca: an instruction to disable any timeconsuming interrupt + here except the heart beat of timer2 should be added before setsda(adap, sb). + because when interrupt occurs during + scl is set high, the interrupt service routine is served and may take long, + after the interrupt returns, sda could be sampled by the device(slave) + more than once, and this cause error problem. +*/ static int i2c_outb(struct i2c_adapter *i2c_adap, char c) { int i; @@ -583,9 +590,7 @@ printk("\n"); } -#ifdef MODULE MOD_INC_USE_COUNT; -#endif i2c_add_adapter(adap); return 0; @@ -601,15 +606,13 @@ DEB2(printk("i2c-algo-bit.o: adapter unregistered: %s\n",adap->name)); -#ifdef MODULE MOD_DEC_USE_COUNT; -#endif return 0; } -int __init i2c_algo_bit_init (void) +static int __init i2c_algo_bit_init (void) { - printk("i2c-algo-bit.o: i2c bit algorithm module\n"); + printk(KERN_DEBUG "i2c-algo-bit.o: i2c bit algorithm module\n"); return 0; } @@ -618,7 +621,6 @@ EXPORT_SYMBOL(i2c_bit_add_bus); EXPORT_SYMBOL(i2c_bit_del_bus); -#ifdef MODULE MODULE_AUTHOR("Simon G. Vogl "); MODULE_DESCRIPTION("I2C-Bus bit-banging algorithm"); MODULE_LICENSE("GPL"); @@ -632,12 +634,4 @@ MODULE_PARM_DESC(i2c_debug, "debug level - 0 off; 1 normal; 2,3 more verbose; 9 bit-protocol"); -int init_module(void) -{ - return i2c_algo_bit_init(); -} - -void cleanup_module(void) -{ -} -#endif +module_init(i2c_algo_bit_init); diff -urN orig/drivers/i2c/i2c-algo-pcf.c linux/drivers/i2c/i2c-algo-pcf.c --- orig/drivers/i2c/i2c-algo-pcf.c Fri Oct 26 16:46:10 2001 +++ linux/drivers/i2c/i2c-algo-pcf.c Fri Oct 5 17:42:36 2001 @@ -476,9 +476,7 @@ return i; } -#ifdef MODULE MOD_INC_USE_COUNT; -#endif i2c_add_adapter(adap); @@ -516,13 +514,11 @@ return res; DEB2(printk("i2c-algo-pcf.o: adapter unregistered: %s\n",adap->name)); -#ifdef MODULE MOD_DEC_USE_COUNT; -#endif return 0; } -int __init i2c_algo_pcf_init (void) +static int __init i2c_algo_pcf_init (void) { printk("i2c-algo-pcf.o: i2c pcf8584 algorithm module\n"); return 0; @@ -532,7 +528,6 @@ EXPORT_SYMBOL(i2c_pcf_add_bus); EXPORT_SYMBOL(i2c_pcf_del_bus); -#ifdef MODULE MODULE_AUTHOR("Hans Berglund "); MODULE_DESCRIPTION("I2C-Bus PCF8584 algorithm"); MODULE_LICENSE("GPL"); @@ -544,13 +539,4 @@ MODULE_PARM_DESC(i2c_debug, "debug level - 0 off; 1 normal; 2,3 more verbose; 9 pcf-protocol"); - -int init_module(void) -{ - return i2c_algo_pcf_init(); -} - -void cleanup_module(void) -{ -} -#endif +module_init(i2c_algo_pcf_init); diff -urN orig/drivers/i2c/i2c-core.c linux/drivers/i2c/i2c-core.c --- orig/drivers/i2c/i2c-core.c Mon Aug 5 13:30:49 2002 +++ linux/drivers/i2c/i2c-core.c Mon Aug 5 13:43:34 2002 @@ -72,7 +72,7 @@ static int driver_count; /**** debug level */ -static int i2c_debug=1; +static int i2c_debug = 0; /* --------------------------------------------------- * /proc entry declarations @@ -82,7 +82,7 @@ #ifdef CONFIG_PROC_FS static int i2cproc_init(void); -static int i2cproc_cleanup(void); +static void i2cproc_cleanup(void); #if (LINUX_VERSION_CODE <= KERNEL_VERSION(2,3,27)) static void monitor_bus_i2c(struct inode *inode, int fill); @@ -740,14 +740,13 @@ return 0; } -int i2cproc_cleanup(void) +static void i2cproc_cleanup(void) { if (i2cproc_initialized >= 1) { remove_proc_entry("i2c",proc_bus); i2cproc_initialized -= 2; } - return 0; } @@ -1263,7 +1262,7 @@ static int __init i2c_init(void) { - printk("i2c-core.o: i2c core module\n"); + printk(KERN_DEBUG "i2c-core.o: i2c core module\n"); memset(adapters,0,sizeof(adapters)); memset(drivers,0,sizeof(drivers)); adap_count=0; @@ -1277,98 +1276,6 @@ return 0; } -#ifndef MODULE -#ifdef CONFIG_I2C_CHARDEV - extern int i2c_dev_init(void); -#endif -#ifdef CONFIG_I2C_ALGOBIT - extern int i2c_algo_bit_init(void); -#endif -#ifdef CONFIG_I2C_PHILIPSPAR - extern int i2c_bitlp_init(void); -#endif -#ifdef CONFIG_I2C_ELV - extern int i2c_bitelv_init(void); -#endif -#ifdef CONFIG_I2C_VELLEMAN - extern int i2c_bitvelle_init(void); -#endif -#ifdef CONFIG_I2C_BITVIA - extern int i2c_bitvia_init(void); -#endif - -#ifdef CONFIG_I2C_ALGOPCF - extern int i2c_algo_pcf_init(void); -#endif -#ifdef CONFIG_I2C_ELEKTOR - extern int i2c_pcfisa_init(void); -#endif - -#ifdef CONFIG_I2C_ALGO8XX - extern int i2c_algo_8xx_init(void); -#endif -#ifdef CONFIG_I2C_RPXLITE - extern int i2c_rpx_init(void); -#endif -#ifdef CONFIG_I2C_PROC - extern int sensors_init(void); -#endif - -/* This is needed for automatic patch generation: sensors code starts here */ -/* This is needed for automatic patch generation: sensors code ends here */ - -int __init i2c_init_all(void) -{ - /* --------------------- global ----- */ - i2c_init(); - -#ifdef CONFIG_I2C_CHARDEV - i2c_dev_init(); -#endif - /* --------------------- bit -------- */ -#ifdef CONFIG_I2C_ALGOBIT - i2c_algo_bit_init(); -#endif -#ifdef CONFIG_I2C_PHILIPSPAR - i2c_bitlp_init(); -#endif -#ifdef CONFIG_I2C_ELV - i2c_bitelv_init(); -#endif -#ifdef CONFIG_I2C_VELLEMAN - i2c_bitvelle_init(); -#endif - - /* --------------------- pcf -------- */ -#ifdef CONFIG_I2C_ALGOPCF - i2c_algo_pcf_init(); -#endif -#ifdef CONFIG_I2C_ELEKTOR - i2c_pcfisa_init(); -#endif - - /* --------------------- 8xx -------- */ -#ifdef CONFIG_I2C_ALGO8XX - i2c_algo_8xx_init(); -#endif -#ifdef CONFIG_I2C_RPXLITE - i2c_rpx_init(); -#endif - - /* -------------- proc interface ---- */ -#ifdef CONFIG_I2C_PROC - sensors_init(); -#endif -/* This is needed for automatic patch generation: sensors code starts here */ -/* This is needed for automatic patch generation: sensors code ends here */ - - return 0; -} - -#endif - - - EXPORT_SYMBOL(i2c_add_adapter); EXPORT_SYMBOL(i2c_del_adapter); EXPORT_SYMBOL(i2c_add_driver); @@ -1405,20 +1312,11 @@ EXPORT_SYMBOL(i2c_get_functionality); EXPORT_SYMBOL(i2c_check_functionality); -#ifdef MODULE MODULE_AUTHOR("Simon G. Vogl "); MODULE_DESCRIPTION("I2C-Bus main module"); MODULE_PARM(i2c_debug, "i"); MODULE_PARM_DESC(i2c_debug,"debug level"); MODULE_LICENSE("GPL"); -int init_module(void) -{ - return i2c_init(); -} - -void cleanup_module(void) -{ - i2cproc_cleanup(); -} -#endif +module_init(i2c_init); +module_exit(i2cproc_cleanup); diff -urN orig/drivers/i2c/i2c-dev.c linux/drivers/i2c/i2c-dev.c --- orig/drivers/i2c/i2c-dev.c Fri Oct 26 16:46:10 2001 +++ linux/drivers/i2c/i2c-dev.c Fri Oct 5 17:46:29 2001 @@ -53,11 +53,6 @@ #include #include -#ifdef MODULE -extern int init_module(void); -extern int cleanup_module(void); -#endif /* def MODULE */ - /* struct file_operations changed too often in the 2.1 series for nice code */ #if LINUX_KERNEL_VERSION < KERNEL_VERSION(2,4,9) @@ -79,13 +74,8 @@ static int i2cdev_command(struct i2c_client *client, unsigned int cmd, void *arg); -#ifdef MODULE -static -#else -extern -#endif - int __init i2c_dev_init(void); -static int i2cdev_cleanup(void); +static int __init i2c_dev_init(void); +static void i2cdev_cleanup(void); static struct file_operations i2cdev_fops = { #if LINUX_KERNEL_VERSION >= KERNEL_VERSION(2,4,0) @@ -483,11 +473,11 @@ return -1; } -int __init i2c_dev_init(void) +static int __init i2c_dev_init(void) { int res; - printk("i2c-dev.o: i2c /dev entries driver module\n"); + printk(KERN_DEBUG "i2c-dev.o: i2c /dev entries driver module\n"); i2cdev_initialized = 0; #ifdef CONFIG_DEVFS_FS @@ -513,7 +503,7 @@ return 0; } -int i2cdev_cleanup(void) +static void i2cdev_cleanup(void) { int res; @@ -521,9 +511,9 @@ if ((res = i2c_del_driver(&i2cdev_driver))) { printk("i2c-dev.o: Driver deregistration failed, " "module not removed.\n"); - return res; + return; } - i2cdev_initialized --; + i2cdev_initialized --; } if (i2cdev_initialized >= 1) { @@ -535,30 +525,18 @@ #endif printk("i2c-dev.o: unable to release major %d for i2c bus\n", I2C_MAJOR); - return res; + return; } i2cdev_initialized --; } - return 0; } EXPORT_NO_SYMBOLS; -#ifdef MODULE - MODULE_AUTHOR("Frodo Looijaard and Simon G. Vogl "); MODULE_DESCRIPTION("I2C /dev entries driver"); MODULE_LICENSE("GPL"); -int init_module(void) -{ - return i2c_dev_init(); -} - -int cleanup_module(void) -{ - return i2cdev_cleanup(); -} - -#endif /* def MODULE */ +module_init(i2c_dev_init); +module_exit(i2cdev_cleanup); diff -urN orig/drivers/i2c/i2c-elektor.c linux/drivers/i2c/i2c-elektor.c --- orig/drivers/i2c/i2c-elektor.c Fri Oct 26 16:46:10 2001 +++ linux/drivers/i2c/i2c-elektor.c Mon Oct 8 23:55:48 2001 @@ -184,16 +184,12 @@ static void pcf_isa_inc_use(struct i2c_adapter *adap) { -#ifdef MODULE MOD_INC_USE_COUNT; -#endif } static void pcf_isa_dec_use(struct i2c_adapter *adap) { -#ifdef MODULE MOD_DEC_USE_COUNT; -#endif } @@ -222,7 +218,7 @@ pcf_isa_unreg, }; -int __init i2c_pcfisa_init(void) +static int __init i2c_pcfisa_init(void) { #ifdef __alpha__ /* check to see we have memory mapped PCF8584 connected to the @@ -294,10 +290,14 @@ return 0; } +static void i2c_pcfisa_exit(void) +{ + i2c_pcf_del_bus(&pcf_isa_ops); + pcf_isa_exit(); +} EXPORT_NO_SYMBOLS; -#ifdef MODULE MODULE_AUTHOR("Hans Berglund "); MODULE_DESCRIPTION("I2C-Bus adapter routines for PCF8584 ISA bus adapter"); MODULE_LICENSE("GPL"); @@ -309,15 +309,5 @@ MODULE_PARM(mmapped, "i"); MODULE_PARM(i2c_debug, "i"); -int init_module(void) -{ - return i2c_pcfisa_init(); -} - -void cleanup_module(void) -{ - i2c_pcf_del_bus(&pcf_isa_ops); - pcf_isa_exit(); -} - -#endif +module_init(i2c_pcfisa_init); +module_exit(i2c_pcfisa_exit); diff -urN orig/drivers/i2c/i2c-elv.c linux/drivers/i2c/i2c-elv.c --- orig/drivers/i2c/i2c-elv.c Fri Oct 26 16:46:10 2001 +++ linux/drivers/i2c/i2c-elv.c Mon Oct 8 23:55:49 2001 @@ -132,16 +132,12 @@ static void bit_elv_inc_use(struct i2c_adapter *adap) { -#ifdef MODULE MOD_INC_USE_COUNT; -#endif } static void bit_elv_dec_use(struct i2c_adapter *adap) { -#ifdef MODULE MOD_DEC_USE_COUNT; -#endif } /* ------------------------------------------------------------------------ @@ -168,7 +164,7 @@ bit_elv_unreg, }; -int __init i2c_bitelv_init(void) +static int __init i2c_bitelv_init(void) { printk("i2c-elv.o: i2c ELV parallel port adapter module\n"); if (base==0) { @@ -195,9 +191,14 @@ } +static void __exit i2c_bitelv_exit(void) +{ + i2c_bit_del_bus(&bit_elv_ops); + bit_elv_exit(); +} + EXPORT_NO_SYMBOLS; -#ifdef MODULE MODULE_AUTHOR("Simon G. Vogl "); MODULE_DESCRIPTION("I2C-Bus adapter routines for ELV parallel port adapter"); MODULE_LICENSE("GPL"); @@ -205,15 +206,5 @@ MODULE_PARM(base, "i"); -int init_module(void) -{ - return i2c_bitelv_init(); -} - -void cleanup_module(void) -{ - i2c_bit_del_bus(&bit_elv_ops); - bit_elv_exit(); -} - -#endif +module_init(i2c_bitelv_init); +module_exit(i2c_bitelv_exit); diff -urN orig/drivers/i2c/i2c-frodo.c linux/drivers/i2c/i2c-frodo.c --- orig/drivers/i2c/i2c-frodo.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/i2c/i2c-frodo.c Thu Oct 24 13:01:38 2002 @@ -0,0 +1,114 @@ + +/* + * linux/drivers/i2c/i2c-frodo.c + * + * Author: Abraham van der Merwe + * + * An I2C adapter driver for the 2d3D, Inc. StrongARM SA-1110 + * Development board (Frodo). + * + * This source code is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * version 2 as published by the Free Software Foundation. + */ + +#include +#include +#include + +#include +#include +#include + +#include + +#include +#include + +static void frodo_setsda (void *data,int state) +{ + if (state) + frodo_cpld_set (FRODO_CPLD_I2C,FRODO_I2C_SDA_OUT); + else + frodo_cpld_clear (FRODO_CPLD_I2C,FRODO_I2C_SDA_OUT); +} + +static void frodo_setscl (void *data,int state) +{ + if (state) + frodo_cpld_set (FRODO_CPLD_I2C,FRODO_I2C_SCL_OUT); + else + frodo_cpld_clear (FRODO_CPLD_I2C,FRODO_I2C_SCL_OUT); +} + +static int frodo_getsda (void *data) +{ + return ((frodo_cpld_read (FRODO_CPLD_I2C) & FRODO_I2C_SDA_IN) != 0); +} + +static int frodo_getscl (void *data) +{ + return ((frodo_cpld_read (FRODO_CPLD_I2C) & FRODO_I2C_SCL_IN) != 0); +} + +static struct i2c_algo_bit_data bit_frodo_data = { + setsda: frodo_setsda, + setscl: frodo_setscl, + getsda: frodo_getsda, + getscl: frodo_getscl, + udelay: 80, + mdelay: 80, + timeout: 100 +}; + +static int frodo_client_register (struct i2c_client *client) +{ + return (0); +} + +static int frodo_client_unregister (struct i2c_client *client) +{ + return (0); +} + +static void frodo_inc_use (struct i2c_adapter *adapter) +{ + MOD_INC_USE_COUNT; +} + +static void frodo_dec_use (struct i2c_adapter *adapter) +{ + MOD_DEC_USE_COUNT; +} + +static struct i2c_adapter frodo_ops = { + name: "Frodo adapter driver", + id: I2C_HW_B_FRODO, + algo: NULL, + algo_data: &bit_frodo_data, + inc_use: frodo_inc_use, + dec_use: frodo_dec_use, + client_register: frodo_client_register, + client_unregister: frodo_client_unregister +}; + +static int __init i2c_frodo_init (void) +{ + return (i2c_bit_add_bus (&frodo_ops)); +} + +EXPORT_NO_SYMBOLS; + +static void __exit i2c_frodo_exit (void) +{ + i2c_bit_del_bus (&frodo_ops); +} + +MODULE_AUTHOR ("Abraham van der Merwe "); +MODULE_DESCRIPTION ("I2C-Bus adapter routines for Frodo"); +MODULE_LICENSE ("GPL"); +EXPORT_NO_SYMBOLS; + +module_init (i2c_frodo_init); +module_exit (i2c_frodo_exit); + diff -urN orig/drivers/i2c/i2c-guide.c linux/drivers/i2c/i2c-guide.c --- orig/drivers/i2c/i2c-guide.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/i2c/i2c-guide.c Thu Feb 27 23:20:12 2003 @@ -0,0 +1,199 @@ +/************************************************************************************\ +Copyright : Copyright (C) 1995-2000 Simon G. Vogl + Copyright 2002 IDERs Incorporated +File Name : i2c-guide.c +Description : this i2c driver uses the GPIO port B pin 0 and pin 1 on the cs89712. +Notes : To change the bit rate, change the structure i2c_algo_bit_data + : to 10 10 100 +Contact : tsong@iders.ca +License : This source code is free software; you can redistribute it and/or + modify it under the terms of the GNU General Public License + version 2 as published by the Free Software Foundation. +\************************************************************************************/ + +#include +#include +#include +#include +#include /* for 2.0 kernels to get NULL */ +#include /* for 2.0 kernels to get ENODEV */ +#include + +#include // io operation ep_writel() +#include // io operation clps_writel() +#include // io operation clps_writel() + +#include +#include + +/* ----- global defines ----------------------------------------------- */ + +#define DEB(x) /* should be reasonable open, close &c. */ +#define DEB2(x) /* low level debugging - very slow */ +#define DEBE(x) x /* error messages */ + /* Pin Port Inverted name */ +#define I2C_SDA 0x08 /* port B ctrl pin 3 (inv) */ +#define I2C_SCL 0x04 /* port B ctrl pin 2 (inv) */ + +#define I2C_SDAIN 0x08 /* use the same pin with output */ +#define I2C_SCLIN 0x04 /* use the same pin with output */ + +#define I2C_DMASK 0xf7 /* inverse of I2C_SDA */ +#define I2C_CMASK 0xfb /* inverse of I2c_SCL */ + +#define PORTB_PIN0_SDA_OUTPUT 0x08 /* pin 3 direction of port B output */ +#define PORTB_PIN0_SDA_INPUT 0xf7 /* pin 3 direction of port B input */ + +#define PORTB_PIN1_SCL_OUTPUT 0x04 /* pin 2 direction of port B output */ +#define PORTB_PIN1_SCL_INPUT 0xfb /* pin 2 direction of port B input */ + +int base = 0; +#define DEFAULT_BASE PBDR + +/* ----- local functions --------------------------------------------------- */ + +static void bit_guide_setscl(void* data, int state) +{ + if (state) { + // set port B pin2 input + clps_writeb((clps_readb(PBDDR)) & PORTB_PIN1_SCL_INPUT, PBDDR); + } + else { + // clear + clps_writeb((clps_readb(PBDR)) & I2C_CMASK, PBDR); + // set port B pin2 output + clps_writeb((clps_readb(PBDDR)) | PORTB_PIN1_SCL_OUTPUT, PBDDR); + } +} + +static void bit_guide_setsda(void* data, int state) +{ + if (state) { + clps_writeb((clps_readb(PBDDR)) & PORTB_PIN0_SDA_INPUT, PBDDR); + // float pin 0 (actually drive high by pull up resistor) + // clps_writeb((clps_readb(PBDR)) | I2C_SDA, PBDR); // set Jan4 ori: eff + // printk("set sda high, state=%i\n",state); + } + else { + // clear + clps_writeb((clps_readb(PBDR)) & I2C_DMASK, PBDR); + // set port B pin 0 output + clps_writeb((clps_readb(PBDDR)) | PORTB_PIN0_SDA_OUTPUT, PBDDR); + } +} + +static int bit_guide_getscl(void *data) +{ + return ( 0 != ( (clps_readb(PBDR)) & I2C_SCLIN ) ); +} + +static int bit_guide_getsda(void *data) +{ + // set port B pin 0 input Jan4 ori eff + clps_writeb((clps_readb(PBDDR)) & PORTB_PIN0_SDA_INPUT, PBDDR); + return ( 0 != ( (clps_readb(PBDR) ) & I2C_SDAIN ) ); +} + +static int bit_guide_init(void) +{ + bit_guide_setsda((void*)base,1); + bit_guide_setscl((void*)base,1); + return 0; +} + +static int bit_guide_reg(struct i2c_client *client) +{ + return 0; +} + +static int bit_guide_unreg(struct i2c_client *client) +{ + return 0; +} + +static void bit_guide_inc_use(struct i2c_adapter *adap) +{ +#ifdef MODULE + MOD_INC_USE_COUNT; +#endif +} + +static void bit_guide_dec_use(struct i2c_adapter *adap) +{ +#ifdef MODULE + MOD_DEC_USE_COUNT; +#endif +} + +/* ------------------------------------------------------------------------ + * Encapsulate the above functions in the correct operations structure. + * This is only done when more than one hardware adapter is supported. + */ + +/* last line (us, ms, timout) + * us dominates the bit rate: 10us means: 100Kbit/sec(25 means 40kbps) + * 10ms not known + * 100ms timeout + */ +static struct i2c_algo_bit_data bit_guide_data = { + NULL, + bit_guide_setsda, + bit_guide_setscl, + bit_guide_getsda, + bit_guide_getscl, + 50, 10, 100, /* orginal (non-guide) value 10, 10, 100 */ +}; + +static struct i2c_adapter bit_guide_ops = { + "Guide Port B: PIN2-SCL/PIN3-SDA", + I2C_HW_B_GUIDE, + NULL, + &bit_guide_data, + bit_guide_inc_use, + bit_guide_dec_use, + bit_guide_reg, + bit_guide_unreg, +}; + +static int __init i2c_bitguide_init(void) +{ + printk("i2c-guide.o: Guide i2c port B adapter module.\n"); + clps_writeb((clps_readb(PBDDR)) & 0xfd, PBDDR); // set service reuest pb1 as input + if (base==0) { + /* probe some values */ + base=DEFAULT_BASE; + bit_guide_data.data=(void*)DEFAULT_BASE; + if (bit_guide_init()==0) { + if(i2c_bit_add_bus(&bit_guide_ops) < 0) + return -ENODEV; + } else { + return -ENODEV; + } + } else { + bit_guide_data.data=(void*)base; + if (bit_guide_init()==0) { + if(i2c_bit_add_bus(&bit_guide_ops) < 0) + return -ENODEV; + } else { + return -ENODEV; + } + } + printk("i2c-guide.o: found device at %#x.\n",base); + return 0; +} + +EXPORT_NO_SYMBOLS; + +MODULE_AUTHOR("T. C. Song "); +MODULE_DESCRIPTION("I2C-Bus adapter routines for Guide (cs89712) GPIO port B"); +MODULE_LICENSE("GPL"); + +MODULE_PARM(base, "i"); + +module_init(i2c_bitguide_init); +/* for completeness, we should have a module_exit() function, but the + GUIDE requires this to always be loaded. If it is unloaded, the + operation of the GUIDE is undefined. + Nobody has written the i2c_bitguide_exit() routine yet, so it is not included. +module_exit(i2c_bitguide_exit); +*/ diff -urN orig/drivers/i2c/i2c-omaha.c linux/drivers/i2c/i2c-omaha.c --- orig/drivers/i2c/i2c-omaha.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/i2c/i2c-omaha.c Thu Oct 24 14:58:03 2002 @@ -0,0 +1,276 @@ +/* ------------------------------------------------------------------------- * + Copyright ARM Limited 2002. All rights reserved. + + i2c driver for Omaha + + Notes:Based on i2c-elv.c + + The S3C2400X01 has better support for I2C, but bit oriented operations + are directly supported by the other I2C layers, so we use that method + of performing I2C operations. + + Copyright (C) 1995-2000 Simon G. Vogl + + This program is free software; you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation; either version 2 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program; if not, write to the Free Software + Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ +/* ------------------------------------------------------------------------- */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include + +#include +#include + +/* ----- global defines ----------------------------------------------- */ +#define DEB(x) if (i2c_debug>=1) x; +#define DEB2(x) if (i2c_debug>=2) x; +#define DEB3(x) if (i2c_debug>=3) x +#define DEBE(x) x // error messages +#define DEBSTAT(x) if (i2c_debug>=3) x; /* print several statistical values*/ +#define DEBPROTO(x) if (i2c_debug>=9) { x; } + /* debug the protocol by showing transferred bits */ + +/* Register and bitdefs for Omaha */ + +// Port G control registers +static volatile unsigned int pgcon = IO_ADDRESS(PLAT_PERIPHERAL_BASE+OMAHA_PGCON); +static volatile unsigned int pgdat = IO_ADDRESS(PLAT_PERIPHERAL_BASE+OMAHA_PGDAT); + +static volatile unsigned int opencr = IO_ADDRESS(PLAT_PERIPHERAL_BASE+OMAHA_OPENCR); + +static int base = IO_ADDRESS(PLAT_PERIPHERAL_BASE+OMAHA_PGCON); + +// Open drain control registers +#define OPC_CMD BIT2 +#define OPC_DAT BIT3 + +// data bits in GPIO Port G data register +#define OMAHA_SDA BIT5 +#define OMAHA_SCL BIT6 +#define IIC_WP BIT3 // Write Protect for EEPROM + +// input/out select bits in GPIO G control register +#define IIC_BITS (BIT12|BIT10|BIT6); + + +/* ----- local functions ---------------------------------------------- */ + + +static void bit_omaha_setscl(void *data, int state) +{ + unsigned int tmp; + + if (state) + { + tmp = __raw_readl(pgdat); + tmp |= OMAHA_SCL; + __raw_writel(tmp,pgdat); + } + else + { + tmp = __raw_readl(pgdat); + tmp &= ~OMAHA_SCL; + __raw_writel(tmp,pgdat); + } +} + +static void bit_omaha_setsda(void *data, int state) +{ + unsigned int tmp; + + // ensure that sda is an output at the moment + tmp = __raw_readl(pgcon); + tmp = tmp | BIT10; + __raw_writel(tmp,pgcon); + + if (state) + { + tmp = __raw_readl(pgdat); + tmp |= OMAHA_SDA; + __raw_writel(tmp,pgdat); + } + else + { + tmp = __raw_readl(pgdat); + tmp &= ~OMAHA_SDA; + __raw_writel(tmp,pgdat); + } +} + +static int bit_omaha_getscl(void *data) +{ + if (__raw_readl(pgdat) & OMAHA_SCL) + return 1; + else + return 0; +} + +static int bit_omaha_getsda(void *data) +{ + unsigned int tmp; + + // ensure that sda is an output at the moment + tmp = __raw_readl(pgcon); + tmp = tmp & ~BIT10; + __raw_writel(tmp,pgcon); + + if (__raw_readl(pgdat) & OMAHA_SDA) + return 1; + else + return 0; +} + +static int bit_omaha_init(void) +{ + // Have we got some mmapped space? + if (request_region(base, 0x100, "i2c (omaha bus adapter)") < 0 ) + { + printk("i2c-omaha.o: requested I/O region (0x%08x) is in use.\n", base); + return -ENODEV; + } + + return 0; +} + + +static int bit_omaha_reg(struct i2c_client *client) +{ + return 0; +} + + +static int bit_omaha_unreg(struct i2c_client *client) +{ + return 0; +} + +static void bit_omaha_inc_use(struct i2c_adapter *adap) +{ + MOD_INC_USE_COUNT; +} + +static void bit_omaha_dec_use(struct i2c_adapter *adap) +{ + MOD_DEC_USE_COUNT; +} + + + +/* ------------------------------------------------------------------------ + * Encapsulate the above functions in the correct operations structure. + * This is only done when more than one hardware adapter is supported. + */ +static struct i2c_algo_bit_data bit_omaha_data = { + NULL, + bit_omaha_setsda, + bit_omaha_setscl, + bit_omaha_getsda, + bit_omaha_getscl, + 10, 10, 20, /* waits, timeout */ +}; + +static struct i2c_adapter bit_omaha_ops = { + "BIT-Type Omaha I2C adapter", + I2C_HW_B_OMAHA, + NULL, + &bit_omaha_data, + bit_omaha_inc_use, + bit_omaha_dec_use, + bit_omaha_reg, + bit_omaha_unreg, +}; + +static int __init i2c_omaha_init (void) +{ + unsigned int tmp; + + printk("i2c-omaha.o: i2c omaha adapter module\n"); + + if (bit_omaha_init() == 0) { + if(i2c_bit_add_bus(&bit_omaha_ops) < 0) + { + printk("Could not add bus!\n"); + return -ENODEV; + } + } else { + printk("Could not pcf_omaha_init\n"); + return -ENODEV; + } + + // Program Port G bits to output function + tmp = __raw_readl(pgcon); + tmp |= IIC_BITS; + __raw_writel(tmp,pgcon); + + // Ensure SDA and SCL are open-drain + tmp = __raw_readl(opencr); + tmp = tmp | OPC_CMD | OPC_DAT; + __raw_writel(tmp,opencr); + + bit_omaha_setsda((void*)base,1); + bit_omaha_setscl((void*)base,1); + + // Disable WP + tmp = __raw_readl(pgdat); + tmp = tmp & ~IIC_WP; + __raw_writel(tmp,pgdat); + + return 0; +} + +static void bit_omaha_exit(void) +{ + release_region(base , 2); +} + +static void i2c_omaha_exit(void) +{ + + i2c_bit_del_bus(&bit_omaha_ops); + + bit_omaha_exit(); + +} + +EXPORT_NO_SYMBOLS; + +MODULE_AUTHOR("ARM Limited "); +MODULE_DESCRIPTION("I2C-Bus adapter routines for Omaha"); +MODULE_LICENSE("GPL"); + +MODULE_PARM(base, "i"); +MODULE_PARM(irq, "i"); +MODULE_PARM(clock, "i"); +MODULE_PARM(own, "i"); +MODULE_PARM(mmapped, "i"); +MODULE_PARM(i2c_debug, "i"); + + +module_init(i2c_omaha_init); +module_exit(i2c_omaha_exit); + + diff -urN orig/drivers/i2c/i2c-philips-par.c linux/drivers/i2c/i2c-philips-par.c --- orig/drivers/i2c/i2c-philips-par.c Sun Oct 14 20:53:01 2001 +++ linux/drivers/i2c/i2c-philips-par.c Fri Sep 28 21:40:51 2001 @@ -259,7 +259,7 @@ }; #endif -int __init i2c_bitlp_init(void) +static int __init i2c_bitlp_init(void) { #if LINUX_VERSION_CODE < KERNEL_VERSION(2,3,4) struct parport *port; @@ -276,7 +276,7 @@ return 0; } -void __exit i2c_bitlp_exit(void) +static void __exit i2c_bitlp_exit(void) { #if LINUX_VERSION_CODE >= KERNEL_VERSION(2,3,4) parport_unregister_driver(&i2c_driver); @@ -295,14 +295,5 @@ MODULE_PARM(type, "i"); -#ifdef MODULE -int init_module(void) -{ - return i2c_bitlp_init(); -} - -void cleanup_module(void) -{ - i2c_bitlp_exit(); -} -#endif +module_init(i2c_bitlp_init); +module_exit(i2c_bitlp_exit); diff -urN orig/drivers/i2c/i2c-velleman.c linux/drivers/i2c/i2c-velleman.c --- orig/drivers/i2c/i2c-velleman.c Fri Oct 26 16:46:10 2001 +++ linux/drivers/i2c/i2c-velleman.c Mon Oct 8 23:55:49 2001 @@ -121,16 +121,12 @@ static void bit_velle_inc_use(struct i2c_adapter *adap) { -#ifdef MODULE MOD_INC_USE_COUNT; -#endif } static void bit_velle_dec_use(struct i2c_adapter *adap) { -#ifdef MODULE MOD_DEC_USE_COUNT; -#endif } /* ------------------------------------------------------------------------ @@ -158,7 +154,7 @@ bit_velle_unreg, }; -int __init i2c_bitvelle_init(void) +static int __init i2c_bitvelle_init(void) { printk("i2c-velleman.o: i2c Velleman K8000 adapter module\n"); if (base==0) { @@ -184,24 +180,19 @@ return 0; } +static void __exit i2c_bitvelle_exit(void) +{ + i2c_bit_del_bus(&bit_velle_ops); + bit_velle_exit(); +} + EXPORT_NO_SYMBOLS; -#ifdef MODULE MODULE_AUTHOR("Simon G. Vogl "); MODULE_DESCRIPTION("I2C-Bus adapter routines for Velleman K8000 adapter"); MODULE_LICENSE("GPL"); MODULE_PARM(base, "i"); -int init_module(void) -{ - return i2c_bitvelle_init(); -} - -void cleanup_module(void) -{ - i2c_bit_del_bus(&bit_velle_ops); - bit_velle_exit(); -} - -#endif +module_init(i2c_bitvelle_init); +module_exit(i2c_bitvelle_exit); diff -urN orig/drivers/ide/Config.in linux/drivers/ide/Config.in --- orig/drivers/ide/Config.in Mon Aug 5 13:30:49 2002 +++ linux/drivers/ide/Config.in Tue Jan 21 23:21:47 2003 @@ -117,6 +117,9 @@ define_bool CONFIG_BLK_DEV_IDEDMA $CONFIG_BLK_DEV_IDEDMA_ICS dep_bool ' RapIDE interface support' CONFIG_BLK_DEV_IDE_RAPIDE $CONFIG_ARCH_ACORN fi + if [ "$CONFIG_ARCH_RISCSTATION" = "y" ]; then + dep_bool ' RiscStation IDE' CONFIG_BLK_DEV_IDE_RISCSTATION $CONFIG_ARCH_RISCSTATION + fi if [ "$CONFIG_AMIGA" = "y" ]; then dep_bool ' Amiga Gayle IDE interface support' CONFIG_BLK_DEV_GAYLE $CONFIG_AMIGA dep_mbool ' Amiga IDE Doubler support (EXPERIMENTAL)' CONFIG_BLK_DEV_IDEDOUBLER $CONFIG_BLK_DEV_GAYLE $CONFIG_EXPERIMENTAL diff -urN orig/drivers/ide/Makefile linux/drivers/ide/Makefile --- orig/drivers/ide/Makefile Mon Aug 5 13:30:49 2002 +++ linux/drivers/ide/Makefile Tue Jan 21 23:21:47 2003 @@ -44,6 +44,7 @@ ide-obj-$(CONFIG_BLK_DEV_HPT366) += hpt366.o ide-obj-$(CONFIG_BLK_DEV_HT6560B) += ht6560b.o ide-obj-$(CONFIG_BLK_DEV_IDE_ICSIDE) += icside.o +ide-obj-$(CONFIG_BLK_DEV_IDE_RISCSTATION) += rstation-ide.o ide-obj-$(CONFIG_BLK_DEV_ADMA) += ide-adma.o ide-obj-$(CONFIG_BLK_DEV_IDEDMA_PCI) += ide-dma.o ide-obj-$(CONFIG_BLK_DEV_IDEPCI) += ide-pci.o diff -urN orig/drivers/ide/icside.c linux/drivers/ide/icside.c --- orig/drivers/ide/icside.c Fri Nov 16 10:09:59 2001 +++ linux/drivers/ide/icside.c Sun Sep 15 18:10:01 2002 @@ -27,6 +27,7 @@ #include extern char *ide_xfer_verbose (byte xfer_rate); +extern char *ide_dmafunc_verbose(ide_dma_action_t dmafunc); /* * Maximum number of interfaces per card @@ -75,6 +76,12 @@ ICS_ARCIN_V6_IDESTEPPING }; +struct icside_state { + unsigned int channel; + unsigned int enabled; + unsigned int irq_port; +}; + static const card_ids icside_cids[] = { { MANU_ICS, PROD_ICS_IDE }, { MANU_ICS2, PROD_ICS2_IDE }, @@ -94,7 +101,7 @@ static void icside_irqenable_arcin_v5 (struct expansion_card *ec, int irqnr) { unsigned int memc_port = (unsigned int)ec->irq_data; - outb (0, memc_port + ICS_ARCIN_V5_INTROFFSET); + outb(0, memc_port + ICS_ARCIN_V5_INTROFFSET); } /* Prototype: icside_irqdisable_arcin_v5 (struct expansion_card *ec, int irqnr) @@ -103,7 +110,7 @@ static void icside_irqdisable_arcin_v5 (struct expansion_card *ec, int irqnr) { unsigned int memc_port = (unsigned int)ec->irq_data; - inb (memc_port + ICS_ARCIN_V5_INTROFFSET); + inb(memc_port + ICS_ARCIN_V5_INTROFFSET); } static const expansioncard_ops_t icside_ops_arcin_v5 = { @@ -122,10 +129,21 @@ */ static void icside_irqenable_arcin_v6 (struct expansion_card *ec, int irqnr) { - unsigned int ide_base_port = (unsigned int)ec->irq_data; + struct icside_state *state = ec->irq_data; + unsigned int base = state->irq_port; - outb (0, ide_base_port + ICS_ARCIN_V6_INTROFFSET_1); - outb (0, ide_base_port + ICS_ARCIN_V6_INTROFFSET_2); + state->enabled = 1; + + switch (state->channel) { + case 0: + outb(0, base + ICS_ARCIN_V6_INTROFFSET_1); + inb(base + ICS_ARCIN_V6_INTROFFSET_2); + break; + case 1: + outb(0, base + ICS_ARCIN_V6_INTROFFSET_2); + inb(base + ICS_ARCIN_V6_INTROFFSET_1); + break; + } } /* Prototype: icside_irqdisable_arcin_v6 (struct expansion_card *ec, int irqnr) @@ -133,10 +151,12 @@ */ static void icside_irqdisable_arcin_v6 (struct expansion_card *ec, int irqnr) { - unsigned int ide_base_port = (unsigned int)ec->irq_data; + struct icside_state *state = ec->irq_data; + + state->enabled = 0; - inb (ide_base_port + ICS_ARCIN_V6_INTROFFSET_1); - inb (ide_base_port + ICS_ARCIN_V6_INTROFFSET_2); + inb (state->irq_port + ICS_ARCIN_V6_INTROFFSET_1); + inb (state->irq_port + ICS_ARCIN_V6_INTROFFSET_2); } /* Prototype: icside_irqprobe(struct expansion_card *ec) @@ -144,10 +164,10 @@ */ static int icside_irqpending_arcin_v6(struct expansion_card *ec) { - unsigned int ide_base_port = (unsigned int)ec->irq_data; + struct icside_state *state = ec->irq_data; - return inb(ide_base_port + ICS_ARCIN_V6_INTRSTAT_1) & 1 || - inb(ide_base_port + ICS_ARCIN_V6_INTRSTAT_2) & 1; + return inb(state->irq_port + ICS_ARCIN_V6_INTRSTAT_1) & 1 || + inb(state->irq_port + ICS_ARCIN_V6_INTRSTAT_2) & 1; } static const expansioncard_ops_t icside_ops_arcin_v6 = { @@ -173,10 +193,10 @@ addr = ecard_address (ec, ECARD_IOC, ECARD_FAST) + ICS_IDENT_OFFSET; - id = inb (addr) & 1; - id |= (inb (addr + 1) & 1) << 1; - id |= (inb (addr + 2) & 1) << 2; - id |= (inb (addr + 3) & 1) << 3; + id = inb(addr) & 1; + id |= (inb(addr + 1) & 1) << 1; + id |= (inb(addr + 2) & 1) << 2; + id |= (inb(addr + 3) & 1) << 3; switch (id) { case 0: /* A3IN */ @@ -210,6 +230,39 @@ return iftype; } +/* + * Handle routing of interrupts. This is called before + * we write a command to the drive. + */ +static void icside_maskproc(ide_drive_t *drive, int mask) +{ + ide_hwif_t *hwif = HWIF(drive); + struct icside_state *state = hwif->hw.priv; + unsigned long flags; + + local_irq_save(flags); + + state->channel = hwif->channel; + + if (state->enabled && !mask) { + switch (hwif->channel) { + case 0: + outb(0, state->irq_port + ICS_ARCIN_V6_INTROFFSET_1); + inb(state->irq_port + ICS_ARCIN_V6_INTROFFSET_2); + break; + case 1: + outb(0, state->irq_port + ICS_ARCIN_V6_INTROFFSET_2); + inb(state->irq_port + ICS_ARCIN_V6_INTROFFSET_1); + break; + } + } else { + inb(state->irq_port + ICS_ARCIN_V6_INTROFFSET_2); + inb(state->irq_port + ICS_ARCIN_V6_INTROFFSET_1); + } + + local_irq_restore(flags); +} + #ifdef CONFIG_BLK_DEV_IDEDMA_ICS /* * SG-DMA support. @@ -223,125 +276,136 @@ #define NR_ENTRIES 256 #define TABLE_SIZE (NR_ENTRIES * 8) -static int ide_build_sglist(ide_hwif_t *hwif, struct request *rq) +static void ide_build_rq_sglist(ide_hwif_t *hwif, struct request *rq) { - struct buffer_head *bh; struct scatterlist *sg = hwif->sg_table; + struct buffer_head *bh; int nents = 0; if (rq->cmd == READ) hwif->sg_dma_direction = PCI_DMA_FROMDEVICE; else hwif->sg_dma_direction = PCI_DMA_TODEVICE; + bh = rq->bh; do { - unsigned char *virt_addr = bh->b_data; - unsigned int size = bh->b_size; + char *virt_addr; + + memset(sg, 0, sizeof(*sg)); + sg->address = virt_addr = bh->b_data; + + do { + virt_addr += bh->b_size; - while ((bh = bh->b_reqnext) != NULL) { - if ((virt_addr + size) != (unsigned char *)bh->b_data) + bh = bh->b_reqnext; + if (bh == NULL) break; - size += bh->b_size; - } - memset(&sg[nents], 0, sizeof(*sg)); - sg[nents].address = virt_addr; - sg[nents].length = size; + + } while (virt_addr == bh->b_data); + + sg->length = virt_addr - (char *)sg->address; + sg++; nents++; } while (bh != NULL); - return pci_map_sg(NULL, sg, nents, hwif->sg_dma_direction); + hwif->sg_nents = nents; } -static int -icside_build_dmatable(ide_drive_t *drive, int reading) +static void ide_build_tf_sglist(ide_hwif_t *hwif, struct request *rq) { - return HWIF(drive)->sg_nents = ide_build_sglist(HWIF(drive), HWGROUP(drive)->rq); -} + ide_task_t *args = rq->special; + struct scatterlist *sg = hwif->sg_table; -/* Teardown mappings after DMA has completed. */ -static void icside_destroy_dmatable(ide_drive_t *drive) -{ - struct scatterlist *sg = HWIF(drive)->sg_table; - int nents = HWIF(drive)->sg_nents; + if (args->command_type == IDE_DRIVE_TASK_RAW_WRITE) + hwif->sg_dma_direction = PCI_DMA_FROMDEVICE; + else + hwif->sg_dma_direction = PCI_DMA_TODEVICE; + + memset(&sg[0], 0, sizeof(*sg)); + sg[0].address = rq->buffer; + sg[0].length = rq->nr_sectors * SECTOR_SIZE; - pci_unmap_sg(NULL, sg, nents, HWIF(drive)->sg_dma_direction); + hwif->sg_nents = 1; } -static int -icside_config_if(ide_drive_t *drive, int xfer_mode) +/* + * Configure the IOMD to give the appropriate timings for the transfer + * mode being requested. We take the advice of the ATA standards, and + * calculate the cycle time based on the transfer mode, and the EIDE + * MW DMA specs that the drive provides in the IDENTIFY command. + * + * We have the following IOMD DMA modes to choose from: + * + * Type Active Recovery Cycle + * A 250 (250) 312 (550) 562 (800) + * B 187 250 437 + * C 125 (125) 125 (375) 250 (500) + * D 62 125 187 + * + * (figures in brackets are actual measured timings) + * + * However, we also need to take care of the read/write active and + * recovery timings: + * + * Read Write + * Mode Active -- Recovery -- Cycle IOMD type + * MW0 215 50 215 480 A + * MW1 80 50 50 150 C + * MW2 70 25 25 120 C + */ +static int icside_set_speed(ide_drive_t *drive, byte xfer_mode) { - int func = ide_dma_off; + int on = 0, cycle_time = 0, use_dma_info = 0; + + /* + * Limit the transfer speed to MW_DMA_2. + */ + if (xfer_mode > XFER_MW_DMA_2) + xfer_mode = XFER_MW_DMA_2; switch (xfer_mode) { case XFER_MW_DMA_2: - /* - * The cycle time is limited to 250ns by the r/w - * pulse width (90ns), however we should still - * have a maximum burst transfer rate of 8MB/s. - */ - drive->drive_data = 250; + cycle_time = 250; + use_dma_info = 1; break; - case XFER_MW_DMA_1: - drive->drive_data = 250; + cycle_time = 250; + use_dma_info = 1; break; - case XFER_MW_DMA_0: - drive->drive_data = 480; + cycle_time = 480; break; - default: - drive->drive_data = 0; + case XFER_SW_DMA_2: + case XFER_SW_DMA_1: + case XFER_SW_DMA_0: + cycle_time = 480; break; } - if (!drive->init_speed) - drive->init_speed = (byte) xfer_mode; + /* + * If we're going to be doing MW_DMA_1 or MW_DMA_2, we should + * take care to note the values in the ID... + */ + if (use_dma_info && drive->id->eide_dma_time > cycle_time) + cycle_time = drive->id->eide_dma_time; + + drive->drive_data = cycle_time; - if (drive->drive_data && - ide_config_drive_speed(drive, (byte) xfer_mode) == 0) - func = ide_dma_on; + if (cycle_time && ide_config_drive_speed(drive, xfer_mode) == 0) + on = 1; else drive->drive_data = 480; - printk("%s: %s selected (peak %dMB/s)\n", drive->name, - ide_xfer_verbose(xfer_mode), 2000 / drive->drive_data); + if (!drive->init_speed) + drive->init_speed = xfer_mode; + else if (cycle_time) + printk("%s: %s selected (peak %dMB/s)\n", drive->name, + ide_xfer_verbose(xfer_mode), 2000 / drive->drive_data); - drive->current_speed = (byte) xfer_mode; + drive->current_speed = xfer_mode; - return func; -} - -static int -icside_set_speed(ide_drive_t *drive, byte speed) -{ - return icside_config_if(drive, speed); -} - -/* - * dma_intr() is the handler for disk read/write DMA interrupts - */ -static ide_startstop_t icside_dmaintr(ide_drive_t *drive) -{ - int i; - byte stat, dma_stat; - - dma_stat = HWIF(drive)->dmaproc(ide_dma_end, drive); - stat = GET_STAT(); /* get drive status */ - if (OK_STAT(stat,DRIVE_READY,drive->bad_wstat|DRQ_STAT)) { - if (!dma_stat) { - struct request *rq = HWGROUP(drive)->rq; - rq = HWGROUP(drive)->rq; - for (i = rq->nr_sectors; i > 0;) { - i -= rq->current_nr_sectors; - ide_end_request(1, HWGROUP(drive)); - } - return ide_stopped; - } - printk("%s: dma_intr: bad DMA status (dma_stat=%x)\n", - drive->name, dma_stat); - } - return ide_error(drive, "dma_intr", stat); + return on; } /* @@ -350,16 +414,16 @@ * This should be defined in one place only. */ struct drive_list_entry { - char * id_model; - char * id_firmware; + const char * id_model; + const char * id_firmware; }; -static struct drive_list_entry drive_whitelist [] = { +static const struct drive_list_entry drive_whitelist [] = { { "Micropolis 2112A", "ALL" }, { "CONNER CTMA 4000", "ALL" }, { "CONNER CTT8000-A", "ALL" }, { "ST34342A", "ALL" }, - { NULL, 0 } + { NULL, NULL } }; static struct drive_list_entry drive_blacklist [] = { @@ -396,10 +460,11 @@ { "PLEXTOR CD-R PX-W8432T", "ALL" }, { "ATAPI CD-ROM DRIVE 40X MAXIMUM", "ALL" }, { "_NEC DV5800A", "ALL" }, - { NULL, 0 } + { NULL, NULL } }; -static int in_drive_list(struct hd_driveid *id, struct drive_list_entry * drive_table) +static int +in_drive_list(struct hd_driveid *id, const struct drive_list_entry *drive_table) { for ( ; drive_table->id_model ; drive_table++) if ((!strcmp(drive_table->id_model, id->model)) && @@ -409,43 +474,30 @@ return 0; } -/* - * For both Blacklisted and Whitelisted drives. - * This is setup to be called as an extern for future support - * to other special driver code. - */ -static int check_drive_lists(ide_drive_t *drive, int good_bad) +static void icside_dma_enable(ide_drive_t *drive, int on, int verbose) { - struct hd_driveid *id = drive->id; + if (!on && verbose) + printk("%s: DMA disabled\n", drive->name); - if (good_bad) { - return in_drive_list(id, drive_whitelist); - } else { - int blacklist = in_drive_list(id, drive_blacklist); - if (blacklist) - printk("%s: Disabling DMA for %s\n", drive->name, id->model); - return(blacklist); - } - return 0; + drive->using_dma = on; } -static int -icside_dma_check(ide_drive_t *drive) +static int icside_dma_check(ide_drive_t *drive) { struct hd_driveid *id = drive->id; ide_hwif_t *hwif = HWIF(drive); - int autodma = hwif->autodma; int xfer_mode = XFER_PIO_2; - int func = ide_dma_off_quietly; + int on; - if (!id || !(id->capability & 1) || !autodma) + if (!id || !(id->capability & 1) || !hwif->autodma) goto out; /* * Consult the list of known "bad" drives */ - if (check_drive_lists(drive, 0)) { - func = ide_dma_off; + if (in_drive_list(drive->id, drive_blacklist)) { + printk("%s: Disabling DMA for %s (blacklisted)\n", + drive->name, drive->id->model); goto out; } @@ -455,13 +507,10 @@ if (id->field_valid & 2) { if (id->dma_mword & 4) { xfer_mode = XFER_MW_DMA_2; - func = ide_dma_on; } else if (id->dma_mword & 2) { xfer_mode = XFER_MW_DMA_1; - func = ide_dma_on; } else if (id->dma_mword & 1) { xfer_mode = XFER_MW_DMA_0; - func = ide_dma_on; } goto out; } @@ -469,106 +518,245 @@ /* * Consult the list of known "good" drives */ - if (check_drive_lists(drive, 1)) { + if (in_drive_list(drive->id, drive_whitelist)) { if (id->eide_dma_time > 150) goto out; xfer_mode = XFER_MW_DMA_1; - func = ide_dma_on; } out: - func = icside_config_if(drive, xfer_mode); + on = icside_set_speed(drive, xfer_mode); + + icside_dma_enable(drive, on, 0); - return hwif->dmaproc(func, drive); + return 0; } -static int -icside_dma_verbose(ide_drive_t *drive) +static int icside_dma_stop(ide_drive_t *drive) { - printk(", DMA"); - return 1; + ide_hwif_t *hwif = HWIF(drive); + + drive->waiting_for_dma = 0; + + disable_dma(hwif->hw.dma); + + /* Teardown mappings after DMA has completed. */ + pci_unmap_sg(NULL, hwif->sg_table, hwif->sg_nents, + hwif->sg_dma_direction); + + return get_dma_residue(hwif->hw.dma) != 0; +} + +static void icside_dma_start(ide_drive_t *drive) +{ + ide_hwif_t *hwif = HWIF(drive); + + /* We can not enable DMA on both channels simultaneously. */ + BUG_ON(dma_channel_active(hwif->hw.dma)); + enable_dma(hwif->hw.dma); +} + +/* + * dma_intr() is the handler for disk read/write DMA interrupts + */ +static ide_startstop_t icside_dmaintr(ide_drive_t *drive) +{ + unsigned int stat; + int dma_stat; + + dma_stat = icside_dma_stop(drive); + stat = GET_STAT(); + if (OK_STAT(stat, DRIVE_READY, drive->bad_wstat | DRQ_STAT)) { + if (!dma_stat) { + struct request *rq = HWGROUP(drive)->rq; + int i; + + for (i = rq->nr_sectors; i > 0;) { + i -= rq->current_nr_sectors; + ide_end_request(1, HWGROUP(drive)); + } + + return ide_stopped; + } + printk(KERN_ERR "%s: bad DMA status (dma_stat=%x)\n", + drive->name, dma_stat); + } + + return ide_error(drive, "dma_intr", stat); } static int -icside_dmaproc(ide_dma_action_t func, ide_drive_t *drive) +icside_dma_common(ide_drive_t *drive, struct request *rq, + unsigned int dma_mode) { ide_hwif_t *hwif = HWIF(drive); - int count, reading = 0; + unsigned int count; + + /* + * We can not enable DMA on both channels. + */ + BUG_ON(hwif->sg_dma_active); + BUG_ON(dma_channel_active(hwif->hw.dma)); + + if (rq->cmd == IDE_DRIVE_TASKFILE) { + ide_build_tf_sglist(hwif, rq); + } else { + ide_build_rq_sglist(hwif, rq); + } + + count = pci_map_sg(NULL, hwif->sg_table, hwif->sg_nents, + hwif->sg_dma_direction); + if (!count) + return 1; + + /* + * Ensure that we have the right interrupt routed. + */ + icside_maskproc(drive, 0); + /* + * Route the DMA signals to the correct interface. + */ + outb(hwif->select_data, hwif->config_data); + + /* + * Select the correct timing for this drive. + */ + set_dma_speed(hwif->hw.dma, drive->drive_data); + + /* + * Tell the DMA engine about the SG table and + * data direction. + */ + set_dma_sg(hwif->hw.dma, hwif->sg_table, count); + set_dma_mode(hwif->hw.dma, dma_mode); + + return 0; +} + +static int icside_dma_init(ide_drive_t *drive, struct request *rq, int rd) +{ + u8 cmd; + + if (icside_dma_common(drive, rq, rd ? DMA_MODE_READ : DMA_MODE_WRITE)) + return 1; + + drive->waiting_for_dma = 1; + + if (drive->media != ide_disk) + return 0; + + ide_set_handler(drive, icside_dmaintr, 2*WAIT_CMD, NULL); + + if (rq->cmd == IDE_DRIVE_TASKFILE && drive->addressing == 1) { + ide_task_t *args = rq->special; + cmd = args->tfRegister[IDE_COMMAND_OFFSET]; + } else if (drive->addressing) { + cmd = rd ? WIN_READDMA_EXT : WIN_WRITEDMA_EXT; + } else { + cmd = rd ? WIN_READDMA : WIN_WRITEDMA; + } + OUT_BYTE(cmd, IDE_COMMAND_REG); + + icside_dma_start(drive); + + return 0; +} + +static int icside_irq_status(ide_drive_t *drive) +{ + ide_hwif_t *hwif = HWIF(drive); + struct icside_state *state = hwif->hw.priv; + + return inb(state->irq_port + + (hwif->channel ? + ICS_ARCIN_V6_INTRSTAT_2 : + ICS_ARCIN_V6_INTRSTAT_1)) & 1; +} + +static int icside_dma_verbose(ide_drive_t *drive) +{ + printk(", %s (peak %dMB/s)", + ide_xfer_verbose(drive->current_speed), + 2000 / drive->drive_data); + return 1; +} + +static void icside_dma_timeout(ide_drive_t *drive) +{ + printk(KERN_ERR "%s: DMA timeout occured: ", drive->name); + ide_dump_status(drive, "DMA timeout", GET_STAT()); +} + +static void icside_irq_lost(ide_drive_t *drive) +{ + printk(KERN_ERR "%s: IRQ lost\n", drive->name); +} + +static int icside_dmaproc(ide_dma_action_t func, ide_drive_t *drive) +{ switch (func) { case ide_dma_off: - printk("%s: DMA disabled\n", drive->name); - /*FALLTHROUGH*/ - case ide_dma_off_quietly: + icside_dma_enable(drive, 0, func == ide_dma_off_quietly); + return 0; + case ide_dma_on: - drive->using_dma = (func == ide_dma_on); + drive->using_dma = 1; + icside_dma_enable(drive, drive->using_dma, 0); return 0; case ide_dma_check: return icside_dma_check(drive); case ide_dma_read: - reading = 1; case ide_dma_write: - count = icside_build_dmatable(drive, reading); - if (!count) - return 1; - disable_dma(hwif->hw.dma); - - /* Route the DMA signals to - * to the correct interface. - */ - outb(hwif->select_data, hwif->config_data); - - /* Select the correct timing - * for this drive - */ - set_dma_speed(hwif->hw.dma, drive->drive_data); - - set_dma_sg(hwif->hw.dma, HWIF(drive)->sg_table, count); - set_dma_mode(hwif->hw.dma, reading ? DMA_MODE_READ - : DMA_MODE_WRITE); - - drive->waiting_for_dma = 1; - if (drive->media != ide_disk) - return 0; - - ide_set_handler(drive, &icside_dmaintr, WAIT_CMD, NULL); - OUT_BYTE(reading ? WIN_READDMA : WIN_WRITEDMA, - IDE_COMMAND_REG); + return icside_dma_init(drive, HWGROUP(drive)->rq, func == ide_dma_read); case ide_dma_begin: - enable_dma(hwif->hw.dma); + icside_dma_start(drive); return 0; case ide_dma_end: - drive->waiting_for_dma = 0; - disable_dma(hwif->hw.dma); - icside_destroy_dmatable(drive); - return get_dma_residue(hwif->hw.dma) != 0; + return icside_dma_stop(drive); case ide_dma_test_irq: - return inb((unsigned long)hwif->hw.priv) & 1; + return icside_irq_status(drive); case ide_dma_bad_drive: case ide_dma_good_drive: - return check_drive_lists(drive, (func == ide_dma_good_drive)); + /* we check our own internal lists; ide never calls these */ + break; case ide_dma_verbose: return icside_dma_verbose(drive); case ide_dma_timeout: - default: - printk("icside_dmaproc: unsupported %s func: %d\n", - ide_dmafunc_verbose(func), func); + icside_dma_timeout(drive); + return 1; + + case ide_dma_lostirq: + icside_irq_lost(drive); + return 1; + + case ide_dma_retune: + /* not implemented in 2.4 */ + break; } + + printk("icside_dmaproc: unsupported %s (%d) function\n", + ide_dmafunc_verbose(func), func); return 1; } -static int -icside_setup_dma(ide_hwif_t *hwif, int autodma) +static int icside_setup_dma(ide_hwif_t *hwif) { + int autodma = 0; + +#ifdef CONFIG_IDEDMA_ICS_AUTO + autodma = 1; +#endif + printk(" %s: SG-DMA", hwif->name); hwif->sg_table = kmalloc(sizeof(struct scatterlist) * NR_ENTRIES, @@ -578,23 +766,30 @@ hwif->dmatable_cpu = NULL; hwif->dmatable_dma = 0; - hwif->speedproc = icside_set_speed; - hwif->dmaproc = icside_dmaproc; - hwif->autodma = autodma; + hwif->speedproc = icside_set_speed; + hwif->dmaproc = icside_dmaproc; + hwif->autodma = autodma; - printk(" capable%s\n", autodma ? - ", auto-enable" : ""); + printk(" capable%s\n", autodma ? ", auto-enable" : ""); return 1; failed: - printk(" -- ERROR, unable to allocate DMA table\n"); + printk(" disabled, unable to allocate DMA table\n"); return 0; } + +int ide_release_dma(ide_hwif_t *hwif) +{ + if (hwif->sg_table) { + kfree(hwif->sg_table); + hwif->sg_table = NULL; + } + return 1; +} #endif -static ide_hwif_t * -icside_find_hwif(unsigned long dataport) +static ide_hwif_t *icside_find_hwif(unsigned long dataport) { ide_hwif_t *hwif; int index; @@ -611,7 +806,7 @@ goto found; } - return NULL; + hwif = NULL; found: return hwif; } @@ -645,7 +840,7 @@ return hwif; } -static int __init icside_register_v5(struct expansion_card *ec, int autodma) +static int __init icside_register_v5(struct expansion_card *ec) { unsigned long slot_port; ide_hwif_t *hwif; @@ -664,14 +859,15 @@ hwif = icside_setup(slot_port, &icside_cardinfo_v5, ec->irq); - return hwif ? 0 : -1; + return hwif ? 0 : -ENODEV; } -static int __init icside_register_v6(struct expansion_card *ec, int autodma) +static int __init icside_register_v6(struct expansion_card *ec) { unsigned long slot_port, port; + struct icside_state *state; ide_hwif_t *hwif, *mate; - int sel = 0; + unsigned int sel = 0; slot_port = ecard_address(ec, ECARD_IOC, ECARD_FAST); port = ecard_address(ec, ECARD_EASI, ECARD_FAST); @@ -683,55 +879,61 @@ outb(sel, slot_port); - ec->irq_data = (void *)port; - ec->ops = (expansioncard_ops_t *)&icside_ops_arcin_v6; - /* * Be on the safe side - disable interrupts */ inb(port + ICS_ARCIN_V6_INTROFFSET_1); inb(port + ICS_ARCIN_V6_INTROFFSET_2); + /* + * Find and register the interfaces. + */ hwif = icside_setup(port, &icside_cardinfo_v6_1, ec->irq); mate = icside_setup(port, &icside_cardinfo_v6_2, ec->irq); + if (!hwif || !mate) + return -ENODEV; + + state = kmalloc(sizeof(struct icside_state), GFP_KERNEL); + if (!state) + return -ENOMEM; + + state->channel = 0; + state->enabled = 0; + state->irq_port = port; + + ec->irq_data = state; + ec->ops = (expansioncard_ops_t *)&icside_ops_arcin_v6; + + hwif->maskproc = icside_maskproc; + hwif->channel = 0; + hwif->hw.priv = state; + hwif->mate = mate; + hwif->serialized = 1; + hwif->config_data = slot_port; + hwif->select_data = sel; + hwif->hw.dma = ec->dma; + + mate->maskproc = icside_maskproc; + mate->channel = 1; + mate->hw.priv = state; + mate->mate = hwif; + mate->serialized = 1; + mate->config_data = slot_port; + mate->select_data = sel | 1; + mate->hw.dma = ec->dma; + #ifdef CONFIG_BLK_DEV_IDEDMA_ICS - if (ec->dma != NO_DMA) { - if (request_dma(ec->dma, hwif->name)) - goto no_dma; - - if (hwif) { - hwif->config_data = slot_port; - hwif->select_data = sel; - hwif->hw.dma = ec->dma; - hwif->hw.priv = (void *) - (port + ICS_ARCIN_V6_INTRSTAT_1); - hwif->channel = 0; - icside_setup_dma(hwif, autodma); - } - if (mate) { - mate->config_data = slot_port; - mate->select_data = sel | 1; - mate->hw.dma = ec->dma; - mate->hw.priv = (void *) - (port + ICS_ARCIN_V6_INTRSTAT_2); - mate->channel = 1; - icside_setup_dma(mate, autodma); - } + if (ec->dma != NO_DMA && !request_dma(ec->dma, hwif->name)) { + icside_setup_dma(hwif); + icside_setup_dma(mate); } -no_dma: #endif - return hwif || mate ? 0 : -1; + return 0; } int __init icside_init(void) { - int autodma = 0; - -#ifdef CONFIG_IDEDMA_ICS_AUTO - autodma = 1; -#endif - ecard_startfind (); do { @@ -746,11 +948,11 @@ switch (icside_identifyif(ec)) { case ics_if_arcin_v5: - result = icside_register_v5(ec, autodma); + result = icside_register_v5(ec); break; case ics_if_arcin_v6: - result = icside_register_v6(ec, autodma); + result = icside_register_v6(ec); break; default: diff -urN orig/drivers/ide/ide-dma.c linux/drivers/ide/ide-dma.c --- orig/drivers/ide/ide-dma.c Mon Aug 5 13:30:51 2002 +++ linux/drivers/ide/ide-dma.c Mon Aug 5 13:43:36 2002 @@ -752,11 +752,10 @@ void ide_setup_dma (ide_hwif_t *hwif, unsigned long dma_base, unsigned int num_ports) { printk(" %s: BM-DMA at 0x%04lx-0x%04lx", hwif->name, dma_base, dma_base + num_ports - 1); - if (check_region(dma_base, num_ports)) { + if (!request_region(dma_base, num_ports, hwif->name)) { printk(" -- ERROR, PORT ADDRESSES ALREADY IN USE\n"); return; } - request_region(dma_base, num_ports, hwif->name); hwif->dma_base = dma_base; hwif->dmatable_cpu = pci_alloc_consistent(hwif->pci_dev, PRD_ENTRIES * PRD_BYTES, @@ -837,9 +836,15 @@ hwif->dma_extra = extra; switch(dev->device) { + /* + * This is buggy. Device numbers are not unique + * between vendors. We should be checking + * both dev->vendor and dev->device + */ case PCI_DEVICE_ID_AL_M5219: case PCI_DEVICE_ID_AMD_VIPER_7409: case PCI_DEVICE_ID_CMD_643: + case PCI_DEVICE_ID_WINBOND_82C105: outb(inb(dma_base+2) & 0x60, dma_base+2); if (inb(dma_base+2) & 0x80) { printk("%s: simplex device: DMA forced\n", name); diff -urN orig/drivers/ide/ide-pci.c linux/drivers/ide/ide-pci.c --- orig/drivers/ide/ide-pci.c Mon Aug 5 13:30:51 2002 +++ linux/drivers/ide/ide-pci.c Mon Aug 5 15:12:48 2002 @@ -623,7 +623,7 @@ } if (pci_enable_device(dev)) { - printk(KERN_WARNING "%s: (ide_setup_pci_device:) Could not enable device.\n", d->name); + printk(KERN_WARNING "%s: (ide_setup_pci_devices:) Could not enable device.\n", d->name); return; } diff -urN orig/drivers/ide/ide-probe.c linux/drivers/ide/ide-probe.c --- orig/drivers/ide/ide-probe.c Mon Aug 5 13:30:52 2002 +++ linux/drivers/ide/ide-probe.c Mon Aug 5 13:43:39 2002 @@ -903,11 +903,11 @@ printk("%s: UNABLE TO GET MAJOR NUMBER %d\n", hwif->name, hwif->major); return (hwif->present = 0); } - + if (init_irq(hwif)) { int i = hwif->irq; /* - * It failed to initialise. Find the default IRQ for + * It failed to initialise. Find the default IRQ for * this port and try that. */ if (!(hwif->irq = ide_default_irq(hwif->io_ports[IDE_DATA_OFFSET]))) { @@ -924,7 +924,7 @@ printk("%s: probed IRQ %d failed, using default.\n", hwif->name, hwif->irq); } - + init_gendisk(hwif); blk_dev[hwif->major].data = hwif; blk_dev[hwif->major].queue = ide_get_queue; diff -urN orig/drivers/ide/ide-proc.c linux/drivers/ide/ide-proc.c --- orig/drivers/ide/ide-proc.c Mon Aug 5 13:30:52 2002 +++ linux/drivers/ide/ide-proc.c Mon Aug 5 13:43:39 2002 @@ -415,6 +415,7 @@ case ide_cy82c693: name = "cy82c693"; break; case ide_4drives: name = "4drives"; break; case ide_pmac: name = "mac-io"; break; + case ide_acorn: name = "acorn"; break; default: name = "(unknown)"; break; } len = sprintf(page, "%s\n", name); diff -urN orig/drivers/ide/ide.c linux/drivers/ide/ide.c --- orig/drivers/ide/ide.c Mon Aug 5 13:30:52 2002 +++ linux/drivers/ide/ide.c Tue Jan 21 23:21:47 2003 @@ -241,23 +241,14 @@ static void init_hwif_data (unsigned int index) { unsigned int unit; - hw_regs_t hw; ide_hwif_t *hwif = &ide_hwifs[index]; /* bulk initialize hwif & drive info with zeros */ memset(hwif, 0, sizeof(ide_hwif_t)); - memset(&hw, 0, sizeof(hw_regs_t)); /* fill in any non-zero initial values */ hwif->index = index; - ide_init_hwif_ports(&hw, ide_default_io_base(index), 0, &hwif->irq); - memcpy(&hwif->hw, &hw, sizeof(hw)); - memcpy(hwif->io_ports, hw.io_ports, sizeof(hw.io_ports)); - hwif->noprobe = !hwif->io_ports[IDE_DATA_OFFSET]; -#ifdef CONFIG_BLK_DEV_HD - if (hwif->io_ports[IDE_DATA_OFFSET] == HD_DATA) - hwif->noprobe = 1; /* may be overridden by ide_setup() */ -#endif /* CONFIG_BLK_DEV_HD */ + hwif->noprobe = 1; hwif->major = ide_hwif_to_major[index]; hwif->name[0] = 'i'; hwif->name[1] = 'd'; @@ -284,6 +275,28 @@ } /* + * Old compatability function - initialise ports using ide_default_io_base + */ +static void ide_old_init_default_hwifs(void) +{ + unsigned int index; + ide_ioreg_t base; + ide_hwif_t *hwif; + + for (index = 0; index < MAX_HWIFS; index++) { + hwif = &ide_hwifs[index]; + + base = ide_default_io_base(index); + + if (base) { + ide_init_hwif_ports(&hwif->hw, base, 0, &hwif->hw.irq); + memcpy(hwif->io_ports, hwif->hw.io_ports, sizeof(hwif->hw.io_ports)); + hwif->noprobe = 0; + } + } +} + +/* * init_ide_data() sets reasonable default values into all fields * of all instances of the hwifs and drives, but only on the first call. * Subsequent calls have no effect (they don't wipe out anything). @@ -312,8 +325,16 @@ init_hwif_data(index); /* Add default hw interfaces */ + ide_old_init_default_hwifs(); ide_init_default_hwifs(); +#ifdef CONFIG_BLK_DEV_HD + /* Check for any clashes with hd.c driver */ + for (index = 0; index < MAX_HWIFS; ++index) + if (ide_hwifs[index].hw.io_ports[IDE_DATA_OFFSET] == HD_DATA) + hwif->noprobe = 1; /* may be overridden by ide_setup() */ +#endif /* CONFIG_BLK_DEV_HD */ + idebus_parameter = 0; system_bus_speed = 0; } @@ -3634,6 +3655,12 @@ rapide_init(); } #endif /* CONFIG_BLK_DEV_IDE_RAPIDE */ +#ifdef CONFIG_BLK_DEV_IDE_RISCSTATION + { + extern void rside_init(void); + rside_init(); + } +#endif /* CONFIG_BLK_DEV_IDE_RISCSTATION */ #ifdef CONFIG_BLK_DEV_GAYLE { extern void gayle_init(void); diff -urN orig/drivers/ide/rstation-ide.c linux/drivers/ide/rstation-ide.c --- orig/drivers/ide/rstation-ide.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/ide/rstation-ide.c Tue Jan 21 23:21:47 2003 @@ -0,0 +1,78 @@ +/* + * linux/drivers/ide/rs-ide.c + * + * Copyright (c) 2002 Ben Dooks + * Copyright (c) 2002 Simtec Electronics + * + * Simple RiscStation IDE support +*/ + +#include +#include +#include +#include +#include +#include + +#include +#include +#include + +#ifndef CONFIG_ARCH_RISCSTATION +#error "compiling this code for non-riscstation hardware is dangerous!" +#endif + +#define DRV_PREFIX "ide-rs" + +#define IRQ_PRI (40+3) +#define IRQ_SEC (40+4) + +#define PORT_BASE ((0x2b800 - 0x10000) >> 2) +#define SEC_OFF (0x400 >> 2) + +int __init rside_reg(unsigned long base, unsigned int irq); + +int __init rside_init(void) +{ + int iotcr; + + if (!machine_is_riscstation()) { + printk(DRV_PREFIX ": hardware is not a RiscStation!\n"); + return 0; + } + + /* select correct area cycle time */ + + iotcr = inb(IOMD_IOTCR); + outb((iotcr & ~3) | 1, IOMD_IOTCR); + + /* register h/w */ + + rside_reg(PORT_BASE, IRQ_PRI); + rside_reg(PORT_BASE + SEC_OFF, IRQ_SEC); + + return 0; +} + + +int __init rside_reg(unsigned long port, unsigned int irq) +{ + unsigned long addr, i; + hw_regs_t hw; + + hw.irq = irq; + + addr = port; + + for (i = IDE_DATA_OFFSET; i <= IDE_STATUS_OFFSET; i++) { + hw.io_ports[i] = (ide_ioreg_t)addr; + addr += 0x40 >> 2; + } + + hw.io_ports[IDE_CONTROL_OFFSET] = port + ((0xb80 - 0x800) >> 2); + + printk(DRV_PREFIX ": registering channel at %08lx, %08lx, irq %d\n", + port, hw.io_ports[IDE_CONTROL_OFFSET], irq); + + return ide_register_hw(&hw, NULL); +} diff -urN orig/drivers/ide/sl82c105.c linux/drivers/ide/sl82c105.c --- orig/drivers/ide/sl82c105.c Mon Oct 1 23:10:45 2001 +++ linux/drivers/ide/sl82c105.c Sat Mar 30 19:19:18 2002 @@ -5,8 +5,12 @@ * * Maintainer unknown. * - * Drive tuning added from Rebel.com's kernel sources - * -- Russell King (15/11/98) linux@arm.linux.org.uk + * Changelog: + * + * 15/11/1998 RMK Drive tuning added from Rebel.com's kernel + * sources + * 30/03/2002 RMK Add fixes specified in W83C553F errata. + * (with special thanks to Todd Inglett) */ #include @@ -29,6 +33,17 @@ extern char *ide_xfer_verbose (byte xfer_rate); /* + * SL82C105 PCI config register 0x40 bits. + */ +#define CTRL_IDE_IRQB (1 << 30) +#define CTRL_IDE_IRQA (1 << 28) +#define CTRL_LEGIRQ (1 << 11) +#define CTRL_P1F16 (1 << 5) +#define CTRL_P1EN (1 << 4) +#define CTRL_P0F16 (1 << 1) +#define CTRL_P0EN (1 << 0) + +/* * Convert a PIO mode and cycle time to the required on/off * times for the interface. This has protection against run-away * timings. @@ -116,6 +131,7 @@ return 0; } + /* * Check to see if the drive and * chipset is capable of DMA mode @@ -156,21 +172,107 @@ } /* - * Our own dmaproc, only to intercept ide_dma_check + * The SL82C105 holds off all IDE interrupts while in DMA mode until + * all DMA activity is completed. Sometimes this causes problems (eg, + * when the drive wants to report an error condition). + * + * 0x7e is a "chip testing" register. Bit 2 resets the DMA controller + * state machine. We need to kick this to work around various bugs. + */ +static inline void sl82c105_reset_host(struct pci_dev *dev) +{ + u16 val; + + pci_read_config_word(dev, 0x7e, &val); + pci_write_config_word(dev, 0x7e, val | (1 << 2)); + pci_write_config_word(dev, 0x7e, val & ~(1 << 2)); +} + +/* + * If we get an IRQ timeout, it might be that the DMA state machine + * got confused. Fix from Todd Inglett. Details from Winbond. + * + * This function is called when the IDE timer expires, the drive + * indicates that it is READY, and we were waiting for DMA to complete. + */ +static int sl82c105_lostirq(ide_drive_t *drive) +{ + ide_hwif_t *hwif = HWIF(drive); + struct pci_dev *dev = hwif->pci_dev; + u32 val, mask = hwif->channel ? CTRL_IDE_IRQB : CTRL_IDE_IRQA; + unsigned long dma_base = hwif->dma_base; + + printk("sl82c105: lost IRQ: resetting host\n"); + + /* + * Check the raw interrupt from the drive. + */ + pci_read_config_dword(dev, 0x40, &val); + if (val & mask) + printk("sl82c105: drive was requesting IRQ, but host lost it\n"); + + /* + * Was DMA enabled? If so, disable it - we're resetting the + * host. The IDE layer will be handling the drive for us. + */ + val = inb(dma_base); + if (val & 1) { + outb(val & ~1, dma_base); + printk("sl82c105: DMA was enabled\n"); + } + + sl82c105_reset_host(dev); + + /* ide_dmaproc would return 1, so we do as well */ + return 1; +} + +/* + * ATAPI devices can cause the SL82C105 DMA state machine to go gaga. + * Winbond recommend that the DMA state machine is reset prior to + * setting the bus master DMA enable bit. + * + * The generic IDE core will have disabled the BMEN bit before this + * function is called. + */ +static void sl82c105_before_bm_enable(ide_drive_t *drive) +{ + ide_hwif_t *hwif = HWIF(drive); + struct pci_dev *dev = hwif->pci_dev; + + sl82c105_reset_host(dev); +} + +/* + * Our very own dmaproc. We need to intercept various calls + * to fix up the SL82C105 specific behaviour. */ static int sl82c105_dmaproc(ide_dma_action_t func, ide_drive_t *drive) { switch (func) { case ide_dma_check: return sl82c105_check_drive(drive); + case ide_dma_on: if (config_for_dma(drive)) func = ide_dma_off; /* fall through */ + case ide_dma_off_quietly: case ide_dma_off: config_for_pio(drive, 4, 0); break; + + case ide_dma_read: + case ide_dma_write: + case ide_dma_begin: + case ide_dma_timeout: + sl82c105_before_bm_enable(drive); + break; + + case ide_dma_lostirq: + return sl82c105_lostirq(drive); + default: break; } @@ -202,16 +304,20 @@ struct pci_dev *bridge; unsigned char rev; - bridge = pci_find_device(PCI_VENDOR_ID_WINBOND, PCI_DEVICE_ID_WINBOND_83C553, NULL); - /* - * If we are part of a Winbond 553 + * The bridge should be part of the same device, but function 0. */ - if (!bridge || bridge->class >> 8 != PCI_CLASS_BRIDGE_ISA) + bridge = pci_find_slot(dev->bus->number, + PCI_DEVFN(PCI_SLOT(dev->devfn), 0)); + if (!bridge) return -1; - if (bridge->bus != dev->bus || - PCI_SLOT(bridge->devfn) != PCI_SLOT(dev->devfn)) + /* + * Make sure it is a Winbond 553 and is an ISA bridge. + */ + if (bridge->vendor != PCI_VENDOR_ID_WINBOND || + bridge->device != PCI_DEVICE_ID_WINBOND_83C553 || + bridge->class >> 8 != PCI_CLASS_BRIDGE_ISA) return -1; /* @@ -227,30 +333,28 @@ */ unsigned int __init pci_init_sl82c105(struct pci_dev *dev, const char *msg) { - unsigned char ctrl_stat; + u32 val; - /* - * Enable the ports - */ - pci_read_config_byte(dev, 0x40, &ctrl_stat); - pci_write_config_byte(dev, 0x40, ctrl_stat | 0x33); + pci_read_config_dword(dev, 0x40, &val); + val |= CTRL_P0EN | CTRL_P0F16 | CTRL_P1EN | CTRL_P1F16; + pci_write_config_dword(dev, 0x40, val); return dev->irq; } void __init dma_init_sl82c105(ide_hwif_t *hwif, unsigned long dma_base) { - unsigned int rev; + unsigned int bridge_rev; byte dma_state; dma_state = inb(dma_base + 2); - rev = sl82c105_bridge_revision(hwif->pci_dev); - if (rev <= 5) { + bridge_rev = sl82c105_bridge_revision(hwif->pci_dev); + if (bridge_rev <= 5) { hwif->autodma = 0; hwif->drives[0].autotune = 1; hwif->drives[1].autotune = 1; printk(" %s: Winbond 553 bridge revision %d, BM-DMA disabled\n", - hwif->name, rev); + hwif->name, bridge_rev); dma_state &= ~0x60; } else { dma_state |= 0x60; @@ -258,9 +362,11 @@ } outb(dma_state, dma_base + 2); - hwif->dmaproc = NULL; ide_setup_dma(hwif, dma_base, 8); - if (hwif->dmaproc) + + if (bridge_rev <= 5) + hwif->dmaproc = NULL; + else hwif->dmaproc = sl82c105_dmaproc; } diff -urN orig/drivers/l3/Config.in linux/drivers/l3/Config.in --- orig/drivers/l3/Config.in Thu Jan 1 01:00:00 1970 +++ linux/drivers/l3/Config.in Sun Sep 23 20:40:19 2001 @@ -0,0 +1,21 @@ +# +# L3 bus configuration +# +mainmenu_option next_comment +comment 'L3 serial bus support' + +tristate 'L3 support' CONFIG_L3 +dep_bool ' L3 bit-banging interfaces' CONFIG_L3_ALGOBIT $CONFIG_L3 +dep_bool ' SA11x0 GPIO adapter' CONFIG_L3_BIT_SA1100_GPIO $CONFIG_L3_ALGOBIT $CONFIG_ARCH_SA1100 + +comment 'Other L3 adapters' +dep_bool ' SA1111 adapter' CONFIG_L3_SA1111 $CONFIG_L3 +endmenu + +# i2c must come before this +if [ "$CONFIG_L3_BIT_SA1100_GPIO" = "y" -o \ + "$CONFIG_I2C_BIT_SA1100_GPIO" = "y" ]; then + define_bool CONFIG_BIT_SA1100_GPIO y +else + define_bool CONFIG_BIT_SA1100_GPIO n +fi diff -urN orig/drivers/l3/Makefile linux/drivers/l3/Makefile --- orig/drivers/l3/Makefile Thu Jan 1 01:00:00 1970 +++ linux/drivers/l3/Makefile Sun Sep 23 20:33:51 2001 @@ -0,0 +1,23 @@ +# +# Makefile for the L3 bus driver. +# + +O_TARGET := l3.o + +export-objs := l3-core.o l3-algo-bit.o +l3-y := +l3-n := +l3-drv-y := +l3-drv-n := + +# Link order: +# (core, adapters, algorithms, drivers) then clients + +l3-$(CONFIG_L3_ALGOBIT) += l3-algo-bit.o +l3-$(CONFIG_BIT_SA1100_GPIO) += l3-bit-sa1100.o +l3-$(CONFIG_L3_SA1111) += l3-sa1111.o + +obj-$(CONFIG_L3) += l3-core.o $(l3-y) $(l3-drv-y) + +include $(TOPDIR)/Rules.make + diff -urN orig/drivers/l3/l3-algo-bit.c linux/drivers/l3/l3-algo-bit.c --- orig/drivers/l3/l3-algo-bit.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/l3/l3-algo-bit.c Tue Oct 23 20:15:55 2001 @@ -0,0 +1,175 @@ +/* + * L3 bus algorithm module. + * + * Copyright (C) 2001 Russell King, All Rights Reserved. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + * + * Note that L3 buses can share the same pins as I2C buses, so we must + * _not_ generate an I2C start condition. An I2C start condition is + * defined as a high-to-low transition of the data line while the clock + * is high. Therefore, we must only change the data line while the + * clock is low. + */ +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#define setdat(adap,val) adap->setdat(adap->data, val) +#define setclk(adap,val) adap->setclk(adap->data, val) +#define setmode(adap,val) adap->setmode(adap->data, val) +#define setdatin(adap) adap->setdir(adap->data, 1) +#define setdatout(adap) adap->setdir(adap->data, 0) +#define getdat(adap) adap->getdat(adap->data) + +/* + * Send one byte of data to the chip. Data is latched into the chip on + * the rising edge of the clock. + */ +static void sendbyte(struct l3_algo_bit_data *adap, unsigned int byte) +{ + int i; + + for (i = 0; i < 8; i++) { + setclk(adap, 0); + udelay(adap->data_hold); + setdat(adap, byte & 1); + udelay(adap->data_setup); + setclk(adap, 1); + udelay(adap->clock_high); + byte >>= 1; + } +} + +/* + * Send a set of bytes to the chip. We need to pulse the MODE line + * between each byte, but never at the start nor at the end of the + * transfer. + */ +static void sendbytes(struct l3_algo_bit_data *adap, const char *buf, int len) +{ + int i; + + for (i = 0; i < len; i++) { + if (i) { + udelay(adap->mode_hold); + setmode(adap, 0); + udelay(adap->mode); + } + setmode(adap, 1); + udelay(adap->mode_setup); + sendbyte(adap, buf[i]); + } +} + +/* + * Read one byte of data from the chip. Data is latched into the chip on + * the rising edge of the clock. + */ +static unsigned int readbyte(struct l3_algo_bit_data *adap) +{ + unsigned int byte = 0; + int i; + + for (i = 0; i < 8; i++) { + setclk(adap, 0); + udelay(adap->data_hold + adap->data_setup); + setclk(adap, 1); + if (getdat(adap)) + byte |= 1 << i; + udelay(adap->clock_high); + } + + return byte; +} + +/* + * Read a set of bytes from the chip. We need to pulse the MODE line + * between each byte, but never at the start nor at the end of the + * transfer. + */ +static void readbytes(struct l3_algo_bit_data *adap, char *buf, int len) +{ + int i; + + for (i = 0; i < len; i++) { + if (i) { + udelay(adap->mode_hold); + setmode(adap, 0); + } + setmode(adap, 1); + udelay(adap->mode_setup); + buf[i] = readbyte(adap); + } +} + +static int l3_xfer(struct l3_adapter *l3_adap, struct l3_msg msgs[], int num) +{ + struct l3_algo_bit_data *adap = l3_adap->algo_data; + int i; + + /* + * If we share an I2C bus, ensure that it is in STOP mode + */ + setclk(adap, 1); + setdat(adap, 1); + setmode(adap, 1); + setdatout(adap); + udelay(adap->mode); + + for (i = 0; i < num; i++) { + struct l3_msg *pmsg = &msgs[i]; + + if (!(pmsg->flags & L3_M_NOADDR)) { + setmode(adap, 0); + udelay(adap->mode_setup); + sendbyte(adap, pmsg->addr); + udelay(adap->mode_hold); + } + + if (pmsg->flags & L3_M_RD) { + setdatin(adap); + readbytes(adap, pmsg->buf, pmsg->len); + } else { + setdatout(adap); + sendbytes(adap, pmsg->buf, pmsg->len); + } + } + + /* + * Ensure that we leave the bus in I2C stop mode. + */ + setclk(adap, 1); + setdat(adap, 1); + setmode(adap, 0); + setdatin(adap); + + return num; +} + +static struct l3_algorithm l3_bit_algo = { + name: "L3 bit-shift algorithm", + xfer: l3_xfer, +}; + +int l3_bit_add_bus(struct l3_adapter *adap) +{ + adap->algo = &l3_bit_algo; + return l3_add_adapter(adap); +} + +int l3_bit_del_bus(struct l3_adapter *adap) +{ + return l3_del_adapter(adap); +} + +EXPORT_SYMBOL(l3_bit_add_bus); +EXPORT_SYMBOL(l3_bit_del_bus); diff -urN orig/drivers/l3/l3-bit-sa1100.c linux/drivers/l3/l3-bit-sa1100.c --- orig/drivers/l3/l3-bit-sa1100.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/l3/l3-bit-sa1100.c Fri Oct 25 20:29:57 2002 @@ -0,0 +1,277 @@ +/* + * linux/drivers/l3/l3-bit-sa1100.c + * + * Copyright (C) 2001 Russell King + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + * + * This is a combined I2C and L3 bus driver. + */ +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include + +#define NAME "l3-bit-sa1100-gpio" + +struct bit_data { + unsigned int sda; + unsigned int scl; + unsigned int l3_mode; +}; + +static int getsda(void *data) +{ + struct bit_data *bits = data; + + return GPLR & bits->sda; +} + +#ifdef CONFIG_I2C_BIT_SA1100_GPIO +static void i2c_setsda(void *data, int state) +{ + struct bit_data *bits = data; + unsigned long flags; + + local_irq_save(flags); + if (state) + GPDR &= ~bits->sda; + else { + GPCR = bits->sda; + GPDR |= bits->sda; + } + local_irq_restore(flags); +} + +static void i2c_setscl(void *data, int state) +{ + struct bit_data *bits = data; + unsigned long flags; + + local_irq_save(flags); + if (state) + GPDR &= ~bits->scl; + else { + GPCR = bits->scl; + GPDR |= bits->scl; + } + local_irq_restore(flags); +} + +static int i2c_getscl(void *data) +{ + struct bit_data *bits = data; + + return GPLR & bits->scl; +} + +static struct i2c_algo_bit_data i2c_bit_data = { + setsda: i2c_setsda, + setscl: i2c_setscl, + getsda: getsda, + getscl: i2c_getscl, + udelay: 10, + mdelay: 10, + timeout: 100, +}; + +static struct i2c_adapter i2c_adapter = { + name: NAME, + algo_data: &i2c_bit_data, +// inc_use: i2c_inc_use, +// dec_use: i2c_dec_use, +}; + +#define LOCK &i2c_adapter.lock + +static int __init i2c_init(struct bit_data *bits) +{ + i2c_bit_data.data = bits; + return i2c_bit_add_bus(&i2c_adapter); +} + +static void i2c_exit(void) +{ + i2c_bit_del_bus(&i2c_adapter); +} + +#else +static DECLARE_MUTEX(l3_lock); +#define LOCK &l3_lock +#define i2c_init(bits) (0) +#define i2c_exit() do { } while (0) +#endif + +#ifdef CONFIG_L3_BIT_SA1100_GPIO +/* + * iPAQs need the clock line driven hard high and low. + */ +static void l3_setscl(void *data, int state) +{ + struct bit_data *bits = data; + unsigned long flags; + + local_irq_save(flags); + if (state) + GPSR = bits->scl; + else + GPCR = bits->scl; + GPDR |= bits->scl; + local_irq_restore(flags); +} + +static void l3_setsda(void *data, int state) +{ + struct bit_data *bits = data; + + if (state) + GPSR = bits->sda; + else + GPCR = bits->sda; +} + +static void l3_setdir(void *data, int in) +{ + struct bit_data *bits = data; + unsigned long flags; + + local_irq_save(flags); + if (in) + GPDR &= ~bits->sda; + else + GPDR |= bits->sda; + local_irq_restore(flags); +} + +static void l3_setmode(void *data, int state) +{ + struct bit_data *bits = data; + + if (state) + GPSR = bits->l3_mode; + else + GPCR = bits->l3_mode; +} + +static struct l3_algo_bit_data l3_bit_data = { + data: NULL, + setdat: l3_setsda, + setclk: l3_setscl, + setmode: l3_setmode, + setdir: l3_setdir, + getdat: getsda, + data_hold: 1, + data_setup: 1, + clock_high: 1, + mode_hold: 1, + mode_setup: 1, +}; + +static struct l3_adapter l3_adapter = { + owner: THIS_MODULE, + name: NAME, + algo_data: &l3_bit_data, + lock: LOCK, +}; + +static int __init l3_init(struct bit_data *bits) +{ + l3_bit_data.data = bits; + return l3_bit_add_bus(&l3_adapter); +} + +static void __exit l3_exit(void) +{ + l3_bit_del_bus(&l3_adapter); +} +#else +#define l3_init(bits) (0) +#define l3_exit() do { } while (0) +#endif + +static struct bit_data bit_data; + +static int __init bus_init(void) +{ + struct bit_data *bit = &bit_data; + unsigned long flags; + int ret; + + if (machine_is_assabet() || machine_is_pangolin()) { + bit->sda = GPIO_GPIO15; + bit->scl = GPIO_GPIO18; + bit->l3_mode = GPIO_GPIO17; + } + +#if defined(CONFIG_SA1100_H3600) || defined(CONFIG_SA1100_H3100) + if (machine_is_h3600() || machine_is_h3100()) { + bit->sda = GPIO_H3600_L3_DATA; + bit->scl = GPIO_H3600_L3_CLOCK; + bit->l3_mode = GPIO_H3600_L3_MODE; + } +#endif + +#ifdef CONFIG_SA1100_STORK + if (machine_is_stork()) { + bit->sda = GPIO_STORK_L3_I2C_SDA; + bit->scl = GPIO_STORK_L3_I2C_SCL; + bit->l3_mode = GPIO_STORK_L3_MODE; + } +#endif + + if (!bit->sda) + return -ENODEV; + + /* + * Default level for L3 mode is low. + * We set SCL and SDA high (i2c idle state). + */ + local_irq_save(flags); + GPDR &= ~(bit->scl | bit->sda); + GPCR = bit->l3_mode | bit->scl | bit->sda; + GPDR |= bit->l3_mode; + local_irq_restore(flags); + + if (machine_is_assabet()) { + /* + * Release reset on UCB1300, ADI7171 and UDA1341. We + * need to do this here so that we can communicate on + * the I2C/L3 buses. + */ + ASSABET_BCR_set(ASSABET_BCR_CODEC_RST); + mdelay(1); + ASSABET_BCR_clear(ASSABET_BCR_CODEC_RST); + mdelay(1); + ASSABET_BCR_set(ASSABET_BCR_CODEC_RST); + } + + ret = i2c_init(bit); + if (ret == 0 && bit->l3_mode) { + ret = l3_init(bit); + if (ret) + i2c_exit(); + } + + return ret; +} + +static void __exit bus_exit(void) +{ + l3_exit(); + i2c_exit(); +} + +module_init(bus_init); +module_exit(bus_exit); diff -urN orig/drivers/l3/l3-core.c linux/drivers/l3/l3-core.c --- orig/drivers/l3/l3-core.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/l3/l3-core.c Tue Oct 23 23:43:33 2001 @@ -0,0 +1,377 @@ +/* + * linux/drivers/l3/l3-core.c + * + * Copyright (C) 2001 Russell King + * + * General structure taken from i2c-core.c by Simon G. Vogl + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License. + * + * See linux/Documentation/l3 for further documentation. + */ +#include +#include +#include +#include +#include +#include +#include +#include +#include + +static DECLARE_MUTEX(adapter_lock); +static LIST_HEAD(adapter_list); + +static DECLARE_MUTEX(driver_lock); +static LIST_HEAD(driver_list); + +/** + * l3_add_adapter - register a new L3 bus adapter + * @adap: l3_adapter structure for the registering adapter + * + * Make the adapter available for use by clients using name adap->name. + * The adap->adapters list is initialised by this function. + * + * Returns 0; + */ +int l3_add_adapter(struct l3_adapter *adap) +{ + INIT_LIST_HEAD(&adap->clients); + down(&adapter_lock); + list_add(&adap->adapters, &adapter_list); + up(&adapter_lock); + return 0; +} + +/** + * l3_del_adapter - unregister a L3 bus adapter + * @adap: l3_adapter structure to unregister + * + * Remove an adapter from the list of available L3 Bus adapters. + * + * Returns 0; + */ +int l3_del_adapter(struct l3_adapter *adap) +{ + down(&adapter_lock); + list_del(&adap->adapters); + up(&adapter_lock); + return 0; +} + +static struct l3_adapter *__l3_get_adapter(const char *name) +{ + struct list_head *l; + + list_for_each(l, &adapter_list) { + struct l3_adapter *adap = list_entry(l, struct l3_adapter, adapters); + + if (strcmp(adap->name, name) == 0) + return adap; + } + + return NULL; +} + +/** + * l3_get_adapter - get a reference to an adapter + * @name: driver name + * + * Obtain a l3_adapter structure for the specified adapter. If the adapter + * is not currently load, then load it. The adapter will be locked in core + * until all references are released via l3_put_adapter. + */ +struct l3_adapter *l3_get_adapter(const char *name) +{ + struct l3_adapter *adap; + int try; + + for (try = 0; try < 2; try ++) { + down(&adapter_lock); + adap = __l3_get_adapter(name); + if (adap && !try_inc_mod_count(adap->owner)) + adap = NULL; + up(&adapter_lock); + + if (adap) + break; + + if (try == 0) + request_module(name); + } + + return adap; +} + +/** + * l3_put_adapter - release a reference to an adapter + * @adap: driver to release reference + * + * Indicate to the L3 core that you no longer require the adapter reference. + * The adapter module may be unloaded when there are no references to its + * data structure. + * + * You must not use the reference after calling this function. + */ +void l3_put_adapter(struct l3_adapter *adap) +{ + if (adap && adap->owner) + __MOD_DEC_USE_COUNT(adap->owner); +} + +/** + * l3_add_driver - register a new L3 device driver + * @driver - driver structure to make available + * + * Make the driver available for use by clients using name driver->name. + * The driver->drivers list is initialised by this function. + * + * Returns 0; + */ +int l3_add_driver(struct l3_driver *driver) +{ + down(&driver_lock); + list_add(&driver->drivers, &driver_list); + up(&driver_lock); + return 0; +} + +/** + * l3_del_driver - unregister a L3 device driver + * @driver: driver to remove + * + * Remove an driver from the list of available L3 Bus device drivers. + * + * Returns 0; + */ +int l3_del_driver(struct l3_driver *driver) +{ + down(&driver_lock); + list_del(&driver->drivers); + up(&driver_lock); + return 0; +} + +static struct l3_driver *__l3_get_driver(const char *name) +{ + struct list_head *l; + + list_for_each(l, &driver_list) { + struct l3_driver *drv = list_entry(l, struct l3_driver, drivers); + + if (strcmp(drv->name, name) == 0) + return drv; + } + + return NULL; +} + +/** + * l3_get_driver - get a reference to a driver + * @name: driver name + * + * Obtain a l3_driver structure for the specified driver. If the driver is + * not currently load, then load it. The driver will be locked in core + * until all references are released via l3_put_driver. + */ +struct l3_driver *l3_get_driver(const char *name) +{ + struct l3_driver *drv; + int try; + + for (try = 0; try < 2; try ++) { + down(&adapter_lock); + drv = __l3_get_driver(name); + if (drv && !try_inc_mod_count(drv->owner)) + drv = NULL; + up(&adapter_lock); + + if (drv) + break; + + if (try == 0) + request_module(name); + } + + return drv; +} + +/** + * l3_put_driver - release a reference to a driver + * @drv: driver to release reference + * + * Indicate to the L3 core that you no longer require the driver reference. + * The driver module may be unloaded when there are no references to its + * data structure. + * + * You must not use the reference after calling this function. + */ +void l3_put_driver(struct l3_driver *drv) +{ + if (drv && drv->owner) + __MOD_DEC_USE_COUNT(drv->owner); +} + +/** + * l3_attach_client - attach a client to an adapter and driver + * @client: client structure to attach + * @adap: adapter (module) name + * @drv: driver (module) name + * + * Attempt to attach a client (a user of a device driver) to a particular + * driver and adapter. If the specified driver or adapter aren't registered, + * request_module is used to load the relevant modules. + * + * Returns 0 on success, or negative error code. + */ +int l3_attach_client(struct l3_client *client, const char *adap, const char *drv) +{ + struct l3_adapter *adapter = l3_get_adapter(adap); + struct l3_driver *driver = l3_get_driver(drv); + int ret = -ENOENT; + + if (!adapter) + printk(KERN_ERR __FUNCTION__ ": unable to get adapter: %s\n", + adap); + if (!driver) + printk(KERN_ERR __FUNCTION__ ": unable to get driver: %s\n", + drv); + + if (adapter && driver) { + ret = 0; + + client->adapter = adapter; + client->driver = driver; + + list_add(&client->__adap, &adapter->clients); + + if (driver->attach_client) + ret = driver->attach_client(client); + } + + if (ret) { + l3_put_driver(driver); + l3_put_adapter(adapter); + } + return ret; +} + +/** + * l3_detach_client - detach a client from an adapter and driver + * @client: client structure to detach + * + * Detach the client from the adapter and driver. + */ +int l3_detach_client(struct l3_client *client) +{ + struct l3_adapter *adapter = client->adapter; + struct l3_driver *driver = client->driver; + + driver->detach_client(client); + + client->adapter = NULL; + client->driver = NULL; + + l3_put_driver(driver); + l3_put_adapter(adapter); + + list_del(&client->__adap); + + return 0; +} + +/** + * l3_transfer - transfer information on an L3 bus + * @adap: adapter structure to perform transfer on + * @msgs: array of l3_msg structures describing transfer + * @num: number of l3_msg structures + * + * Transfer the specified messages to/from a device on the L3 bus. + * + * Returns number of messages successfully transferred, otherwise negative + * error code. + */ +int l3_transfer(struct l3_adapter *adap, struct l3_msg msgs[], int num) +{ + int ret = -ENOSYS; + + if (adap->algo->xfer) { + down(adap->lock); + ret = adap->algo->xfer(adap, msgs, num); + up(adap->lock); + } + return ret; +} + +/** + * l3_write - send data to a device on an L3 bus + * @client: registered client structure + * @addr: L3 bus address + * @buf: buffer for bytes to send + * @len: number of bytes to send + * + * Send len bytes pointed to by buf to device address addr on the L3 bus + * described by client. + * + * Returns the number of bytes transferred, or negative error code. + */ +int l3_write(struct l3_client *client, int addr, const char *buf, int len) +{ + struct l3_adapter *adap = client->adapter; + struct l3_msg msg; + int ret; + + msg.addr = addr; + msg.flags = 0; + msg.buf = (char *)buf; + msg.len = len; + + ret = l3_transfer(adap, &msg, 1); + return ret == 1 ? len : ret; +} + +/** + * l3_read - receive data from a device on an L3 bus + * @client: registered client structure + * @addr: L3 bus address + * @buf: buffer for bytes to receive + * @len: number of bytes to receive + * + * Receive len bytes from device address addr on the L3 bus described by + * client to a buffer pointed to by buf. + * + * Returns the number of bytes transferred, or negative error code. + */ +int l3_read(struct l3_client *client, int addr, char *buf, int len) +{ + struct l3_adapter *adap = client->adapter; + struct l3_msg msg; + int ret; + + msg.addr = addr; + msg.flags = L3_M_RD; + msg.buf = buf; + msg.len = len; + + ret = l3_transfer(adap, &msg, 1); + return ret == 1 ? len : ret; +} + +EXPORT_SYMBOL(l3_add_adapter); +EXPORT_SYMBOL(l3_del_adapter); +EXPORT_SYMBOL(l3_get_adapter); +EXPORT_SYMBOL(l3_put_adapter); + +EXPORT_SYMBOL(l3_add_driver); +EXPORT_SYMBOL(l3_del_driver); +EXPORT_SYMBOL(l3_get_driver); +EXPORT_SYMBOL(l3_put_driver); + +EXPORT_SYMBOL(l3_attach_client); +EXPORT_SYMBOL(l3_detach_client); + +EXPORT_SYMBOL(l3_transfer); +EXPORT_SYMBOL(l3_write); +EXPORT_SYMBOL(l3_read); diff -urN orig/drivers/l3/l3-sa1111.c linux/drivers/l3/l3-sa1111.c --- orig/drivers/l3/l3-sa1111.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/l3/l3-sa1111.c Mon Aug 5 22:25:08 2002 @@ -0,0 +1,118 @@ +/* + * L3 SA1111 algorithm/adapter module. + * + * By Russell King, + * gratuitously ripped from sa1111-uda1341.c by John Dorsey. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + */ +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include + +static inline unsigned char l3_sa1111_recv_byte(unsigned char addr) +{ + unsigned char dat; + + L3_CAR = addr; + while ((SASR0 & SASR0_L3RD) == 0) + mdelay(1); + dat = L3_CDR; + SASCR = SASCR_RDD; + return dat; +} + +static void l3_sa1111_recv_msg(struct l3_msg *msg) +{ + int len = msg->len; + char *p = msg->buf; + + if (len > 1) { + SACR1 |= SACR1_L3MB; + while ((len--) > 1) + *p++ = l3_sa1111_recv_byte(msg->addr); + } + SACR1 &= ~SACR1_L3MB; + *p = l3_sa1111_recv_byte(msg->addr); +} + +static inline void l3_sa1111_send_byte(unsigned char addr, unsigned char dat) +{ + L3_CAR = addr; + L3_CDR = dat; + while ((SASR0 & SASR0_L3WD) == 0) + mdelay(1); + SASCR = SASCR_DTS; +} + +static void l3_sa1111_send_msg(struct l3_msg *msg) +{ + int len = msg->len; + char *p = msg->buf; + + if (len > 1) { + SACR1 |= SACR1_L3MB; + while ((len--) > 1) + l3_sa1111_send_byte(msg->addr, *p++); + } + SACR1 &= ~SACR1_L3MB; + l3_sa1111_send_byte(msg->addr, *p); +} + +static int l3_sa1111_xfer(struct l3_adapter *adap, struct l3_msg msgs[], int num) +{ + int i; + + for (i = 0; i < num; i++) { + struct l3_msg *pmsg = &msgs[i]; + + if (pmsg->flags & L3_M_RD) + l3_sa1111_recv_msg(pmsg); + else + l3_sa1111_send_msg(pmsg); + } + + return num; +} + +static struct l3_algorithm l3_sa1111_algo = { + name: "L3 SA1111 algorithm", + xfer: l3_sa1111_xfer, +}; + +static DECLARE_MUTEX(sa1111_lock); + +static struct l3_adapter l3_sa1111_adapter = { + owner: THIS_MODULE, + name: "l3-sa1111", + algo: &l3_sa1111_algo, + lock: &sa1111_lock, +}; + +static int __init l3_sa1111_init(void) +{ + int ret = -ENODEV; + if ((machine_is_assabet() && machine_has_neponset()) || + machine_is_jornada720() || machine_is_accelent_sa() || + machine_is_badge4()) + ret = l3_add_adapter(&l3_sa1111_adapter); + return ret; +} + +static void __exit l3_sa1111_exit(void) +{ + l3_del_adapter(&l3_sa1111_adapter); +} + +module_init(l3_sa1111_init); +module_exit(l3_sa1111_exit); diff -urN orig/drivers/media/video/Config.in linux/drivers/media/video/Config.in --- orig/drivers/media/video/Config.in Thu Dec 20 11:04:01 2001 +++ linux/drivers/media/video/Config.in Thu Dec 20 11:02:36 2001 @@ -51,5 +51,8 @@ if [ "$CONFIG_EXPERIMENTAL" = "y" ]; then dep_tristate ' Sony Vaio Picturebook Motion Eye Video For Linux' CONFIG_VIDEO_MEYE $CONFIG_VIDEO_DEV $CONFIG_PCI $CONFIG_SONYPI fi +# unfortunately, this depends on having CONFIG_FB_CYBER2000 +# set as well - we hook off of the VGA driver +dep_tristate ' NetWinder Video for Linux (EXPERIMENTAL)' CONFIG_VIDEO_CYBERPRO $CONFIG_VIDEO_DEV $CONFIG_EXPERIMENTAL $CONFIG_ARCH_NETWINDER endmenu diff -urN orig/drivers/media/video/Makefile linux/drivers/media/video/Makefile --- orig/drivers/media/video/Makefile Fri Nov 23 10:12:13 2001 +++ linux/drivers/media/video/Makefile Sun Oct 28 21:02:41 2001 @@ -48,6 +48,7 @@ obj-$(CONFIG_VIDEO_ZORAN_BUZ) += saa7111.o saa7185.o obj-$(CONFIG_VIDEO_ZORAN_DC10) += saa7110.o adv7175.o obj-$(CONFIG_VIDEO_ZORAN_LML33) += bt819.o bt856.o +obj-$(CONFIG_VIDEO_CYBERPRO) += cyberpro.o i2c-old.o saa7111.o obj-$(CONFIG_VIDEO_LML33) += bt856.o bt819.o obj-$(CONFIG_VIDEO_PMS) += pms.o obj-$(CONFIG_VIDEO_PLANB) += planb.o diff -urN orig/drivers/media/video/cyberpro.c linux/drivers/media/video/cyberpro.c --- orig/drivers/media/video/cyberpro.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/media/video/cyberpro.c Thu Jun 20 23:22:21 2002 @@ -0,0 +1,2091 @@ +/* + * CyberPro 2000 video capture driver for the Rebel.com NetWinder + * + * (C) 1999-2000 Russell King + * + * Re-written from Rebel.com's vidcap driver. + * + * Architecture + * ------------ + * The NetWinder video capture consists of a SAA7111 video decoder chip + * connected to the CyberPro feature bus. The video data is captured to + * the VGA memory, where the CyberPro can overlay (by chromakeying) the + * data onto the VGA display. + * + * The CyberPro also has some nifty features, including a second overlay + * and picture in picture mode. We do not currently use these features. + * + * Power Saving + * ------------ + * Please note that rev.5 NetWinders have the ability to hold the SAA7111 + * decoder chip into reset, which saves power. The only time at which + * this is done is when the driver is unloaded, which implies that this + * is compiled as a module. + * + * In this case, you will want the kernel to automatically load this + * driver when required. Place the following line in /etc/modules.conf + * to enable this: + * + * alias char-major-81-0 cyberpro + * + * The relevant modules will be automatically loaded by modprobe on a + * as and when needed basis. + * + * Capture resolution + * ------------------ + * The maximum useful capture resolution is: + * 625-line UK: 716x576 + * 525-line US: ? + * + * Bugs + * ---- + * 1. The CyberPro chip seems to be prone to randomly scribbling over VGA + * memory [hopefully fixed with new capture enable/freeze stuff] + * 2. read()ing pauses video capture, and sometimes triggers bug 1. + * 3. mmap() is not supported (requires BM-DMA - see bug 4) + * 4. Really, we want to do scatter BM-DMA. Is the CyberPro capable of this? + * The Cyberpro seems to randomly scribble to various PCI addresses if you + * transfer >16 words. + * 5. We shouldn't ignore O_NONBLOCK when reading a frame. + * 6. The incoming stream on the NetWinder is CCIR656, which is YUV422. + * CyberPro docs also call the format we capture and overlay "YUV422", + * but we actually seem to have Y, U, Y, V bytes (is this YUYV format?) + */ +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include + +MODULE_AUTHOR("Russell King "); +MODULE_DESCRIPTION("CyberPro v4l video grabber"); +MODULE_LICENSE("GPL"); + +#include "../../video/cyber2000fb.h" + +/* + * These enable various experimental features. Most of these + * are just plain broken or just don't work at the moment. + */ +/* + * Enable this if you want mmap() access. (see bug 4) + */ +#undef USE_MMAP + +/* + * Enable this if you want mmio access. (slow) + */ +#define USE_MMIO + +/* + * The V4L API is unclear whether VIDIOCSCAPTURE call is allowed while + * capture is running. The default is to disallow the call. + * + * Define this if you do want to allow the call while capture is active. + */ +#undef ALLOW_SCAPTURE_WHILE_CAP + +/* + * We capture two frames + */ +#define NR_FRAMES 2 + +/* + * One frame of video is 202 pages, assuming YUV422 format, 716x576 + */ +#define NR_PAGES 202 + +struct src_info { + unsigned int offset; /* offset of source data */ + unsigned int x; /* source x */ + unsigned int y; /* source y */ + unsigned int width; /* source width */ + unsigned int height; /* source height */ + unsigned int format; /* source format */ +}; + +struct dst_info { + unsigned int x; /* destination x */ + unsigned int y; /* destination y */ + unsigned int width; /* destination width */ + unsigned int height; /* destination height */ + unsigned int chromakey; /* chromakey */ + unsigned int flags; /* flags (eg, chromakey enable) */ +}; + +struct cyberpro_vidinfo; + +struct win_info { + void (*init)(struct cyberpro_vidinfo *dp, struct win_info *wi); + void (*set_src)(struct cyberpro_vidinfo *dp, struct win_info *wi); + void (*set_win)(struct cyberpro_vidinfo *dp, struct win_info *wi); + void (*ctl)(struct cyberpro_vidinfo *dp, struct win_info *wi, int on_off); + + /* public */ + struct src_info src; + struct dst_info dst; + + /* private */ + unsigned short vid_fifo_ctl; + unsigned char vid_fmt; + unsigned char vid_disp_ctl1; + unsigned char vid_fifo_ctl1; + unsigned char vid_misc_ctl1; +}; + +struct framebuf { + unsigned int offset; /* mmap offset for this frame */ + unsigned int status; +#define FRAME_FREE 0 +#define FRAME_DONE 1 +#define FRAME_WAITING 2 +#define FRAME_GRABBING 3 + + /* + * Bus-Master DMA stuff. Note that we should + * probably use the kiovec stuff instead. + */ + unsigned long bus_addr[NR_PAGES]; /* list of pages */ + struct page *pages[NR_PAGES]; + void *buffer; + int dbg; +}; + +struct cyberpro_vidinfo { + struct video_device *dev; + struct i2c_bus *bus; + struct cyberpro_info info; /* host information */ + unsigned char *regs; + unsigned int irq; /* PCI interrupt number */ + + /* hardware configuration */ + unsigned int stream_fmt; /* format of stream from decoder*/ + + /* software settings */ + unsigned int decoder:1; /* decoder loaded */ + unsigned int interlace:1; /* interlace */ + unsigned int buf_set:1; /* VIDIOCSFBUF has been issued */ + unsigned int win_set:1; /* VIDIOCSWIN has been issued */ + unsigned int cap_active:1; /* capture is active */ + unsigned int ovl_active:1; /* overlay is active */ + unsigned int mmaped:1; /* buffer is mmap()d */ + unsigned int unused:25; + + unsigned int users; /* number of users */ + unsigned long cap_mem_offset; /* capture framebuffer offset */ + void * buffer; /* kernel capture buffer */ + unsigned int norm; /* video standard */ + + struct video_capability cap; /* capabilities */ + struct video_picture pic; /* current picture settings */ + struct video_buffer buf; /* display parameters */ + struct video_capture capt; /* video capture params */ + + struct win_info *ovl; /* overlay window set */ + struct win_info ext; /* "Extended" window info */ + struct win_info v2; /* "V2" window info */ + struct win_info x2; /* "X2" window info */ + + unsigned int bm_offset; /* Cap memory bus master offset */ + unsigned int bm_index; /* Cap page index */ + +#ifdef USE_MMAP + unsigned int frame_idx; /* currently grabbing frame */ + unsigned int frame_size; + struct framebuf frame[NR_FRAMES]; + wait_queue_head_t frame_wait; +#endif + + wait_queue_head_t vbl_wait; + + /* + * cyberpro registers + */ + unsigned char cap_mode1; + unsigned char cap_mode2; + unsigned char cap_miscctl; + unsigned char vfac1; + unsigned char vfac3; +}; + +/* + * Our access methods. + */ +#define cyberpro_writel(val,reg,dp) writel(val, (dp)->regs + (reg)) +#define cyberpro_writew(val,reg,dp) writew(val, (dp)->regs + (reg)) +#define cyberpro_writeb(val,reg,dp) writeb(val, (dp)->regs + (reg)) + +#define cyberpro_readb(reg,dp) readb((dp)->regs + (reg)) + +static inline void +cyberpro_grphw(unsigned int reg, unsigned int val, struct cyberpro_vidinfo *dp) +{ + cyberpro_writew((reg & 255) | val << 8, 0x3ce, dp); +} + +static void cyberpro_grphw8(unsigned int reg, unsigned int val, struct cyberpro_vidinfo *dp) +{ + cyberpro_grphw(reg, val, dp); +} + +static unsigned char cyberpro_grphr8(int reg, struct cyberpro_vidinfo *dp) +{ + cyberpro_writeb(reg, 0x3ce, dp); + return cyberpro_readb(0x3cf, dp); +} + +static void cyberpro_grphw16(int reg, unsigned int val, struct cyberpro_vidinfo *dp) +{ + cyberpro_grphw(reg, val, dp); + cyberpro_grphw(reg + 1, val >> 8, dp); +} + +static void cyberpro_grphw24(int reg, unsigned int val, struct cyberpro_vidinfo *dp) +{ + cyberpro_grphw(reg, val, dp); + cyberpro_grphw(reg + 1, val >> 8, dp); + cyberpro_grphw(reg + 2, val >> 16, dp); +} + +#if 0 +static void +cyberpro_dbg_dump(void) +{ + int i; + unsigned char idx[] = + { 0x30, 0x3e, 0x98, 0x99, 0x9a, 0x9b, 0x9c, 0x9d, + 0xa6, 0xa7, 0xa8, 0xa9, 0xaa, 0xab, 0xac, 0xad }; + printk(KERN_DEBUG); + for (i = 0; i < sizeof(idx); i++) + printk("%02x ", idx[i]); + printk("\n" KERN_DEBUG); + for (i = 0; i < sizeof(idx); i++) + printk("%02x ", cyberpro_grphr8(idx[i])); + printk("\n"); +} +#endif + +/* + * On the NetWinder, we can put the SAA7111 to sleep by holding + * it in reset. + * + * Note: once we have initialised the SAA7111, we can't put it back to + * sleep and expect it to keep its settings. Maybe a better solution + * is to register/de-register the i2c bus in open/release? + */ +static void +decoder_sleep(int sleep) +{ +#ifdef CONFIG_ARCH_NETWINDER + extern spinlock_t gpio_lock; + + spin_lock_irq(&gpio_lock); + cpld_modify(CPLD_7111_DISABLE, sleep ? CPLD_7111_DISABLE : 0); + spin_unlock_irq(&gpio_lock); + + if (!sleep) { + /* + * wait 20ms for device to wake up + */ + set_current_state(TASK_UNINTERRUPTIBLE); + schedule_timeout(HZ / 50); + } +#endif +} + +/* -------------------------------- I2C support ---------------------------- */ + +#define I2C_DELAY 100 + +static void +cyberpro_i2c_setlines(struct i2c_bus *bus, int ctrl, int data) +{ + struct cyberpro_vidinfo *dp = bus->data; + int v; + + v = (ctrl ? EXT_LATCH2_I2C_CLKEN : 0x00) | (data ? EXT_LATCH2_I2C_DATEN : 0x00); + cyberpro_grphw8(EXT_LATCH2, v, dp); + + udelay(I2C_DELAY); +} + +static int +cyberpro_i2c_getdataline(struct i2c_bus *bus) +{ + struct cyberpro_vidinfo *dp = bus->data; + unsigned long flags; + int v; + + save_flags(flags); + cli(); + + v = cyberpro_grphr8(EXT_LATCH2, dp); + + restore_flags(flags); + + return v & EXT_LATCH2_I2C_DAT ? 1 : 0; +} + +static void +cyberpro_i2c_attach(struct i2c_bus *bus, int id) +{ + struct cyberpro_vidinfo *dp = bus->data; + int zero = 0; + + if (id == I2C_DRIVERID_VIDEODECODER) { + __u16 norm = dp->norm; + i2c_control_device(bus, id, DECODER_SET_NORM, &norm); + i2c_control_device(bus, id, DECODER_SET_PICTURE, &dp->pic); + i2c_control_device(bus, id, DECODER_ENABLE_OUTPUT, &zero); + + dp->decoder = 1; + } +} + +static void +cyberpro_i2c_detach(struct i2c_bus *bus, int id) +{ + struct cyberpro_vidinfo *dp = bus->data; + + if (id == I2C_DRIVERID_VIDEODECODER) + dp->decoder = 0; +} + +static struct i2c_bus cyberpro_i2c_bus = { + name: "", + id: I2C_BUSID_CYBER2000, + bus_lock: SPIN_LOCK_UNLOCKED, + attach_inform: cyberpro_i2c_attach, + detach_inform: cyberpro_i2c_detach, + i2c_setlines: cyberpro_i2c_setlines, + i2c_getdataline: cyberpro_i2c_getdataline, +}; + +/*------------------------- Extended Overlay Window ------------------------- + * Initialise 1st overlay window (works) + */ +static void +cyberpro_ext_init(struct cyberpro_vidinfo *dp, struct win_info *wi) +{ + wi->vid_fifo_ctl = 0xf87c; + wi->vid_fmt = EXT_VID_FMT_YUV422; + wi->vid_disp_ctl1 = EXT_VID_DISP_CTL1_VINTERPOL_OFF | + EXT_VID_DISP_CTL1_NOCLIP; + wi->vid_fifo_ctl1 = EXT_VID_FIFO_CTL1_INTERLEAVE | + EXT_VID_FIFO_CTL1_OE_HIGH; + wi->vid_misc_ctl1 = 0; + + cyberpro_grphw8 (EXT_VID_DISP_CTL1, wi->vid_disp_ctl1, dp); + cyberpro_grphw16(EXT_DDA_X_INIT, 0x0800, dp); + cyberpro_grphw16(EXT_DDA_Y_INIT, 0x0800, dp); + cyberpro_grphw16(EXT_VID_FIFO_CTL, wi->vid_fifo_ctl, dp); + cyberpro_grphw8 (EXT_VID_FIFO_CTL1, wi->vid_fifo_ctl1, dp); +} + +/* + * Set the source parameters for the extended window + */ +static void +cyberpro_ext_set_src(struct cyberpro_vidinfo *dp, struct win_info *wi) +{ + unsigned int phase, pitch; + + pitch = (wi->src.width >> 2) & 0x0fff; + phase = (wi->src.width + 3) >> 2; + + wi->vid_fmt &= ~7; + switch (wi->src.format) { + case VIDEO_PALETTE_RGB565: wi->vid_fmt |= EXT_VID_FMT_RGB565; break; + case VIDEO_PALETTE_RGB24: wi->vid_fmt |= EXT_VID_FMT_RGB888_24; break; + case VIDEO_PALETTE_RGB32: wi->vid_fmt |= EXT_VID_FMT_RGB888_32; break; + case VIDEO_PALETTE_RGB555: wi->vid_fmt |= EXT_VID_FMT_RGB555; break; + case VIDEO_PALETTE_YUV422: wi->vid_fmt |= EXT_VID_FMT_YUV422; break; + } + + cyberpro_grphw24(EXT_MEM_START, wi->src.offset, dp); + cyberpro_grphw16(EXT_SRC_WIDTH, pitch | ((phase << 4) & 0xf000), dp); + cyberpro_grphw8 (EXT_SRC_WIN_WIDTH, phase, dp); + cyberpro_grphw8 (EXT_VID_FMT, wi->vid_fmt, dp); +} + +/* + * Set overlay1 window + */ +static void +cyberpro_ext_set_win(struct cyberpro_vidinfo *dp, struct win_info *wi) +{ + unsigned int xscale, yscale; + unsigned int xoff, yoff; + + /* + * Note: the offset does not appear to be influenced by + * hardware scrolling. + */ + xoff = yoff = 0; + + xoff += wi->dst.x; + yoff += wi->dst.y; + + xscale = wi->src.width; + + if (wi->dst.width >= wi->src.width * 2) { + wi->vid_fmt |= EXT_VID_FMT_DBL_H_PIX; + xscale *= 2; + } else { + wi->vid_fmt &= ~EXT_VID_FMT_DBL_H_PIX; + } + + xscale = ((xscale - /*2*/0) * 4096) / wi->dst.width; + yscale = ((wi->src.height - /*2*/0) * 4096) / wi->dst.height; + + cyberpro_grphw16(EXT_X_START, xoff, dp); + cyberpro_grphw16(EXT_X_END, xoff + wi->dst.width, dp); + cyberpro_grphw16(EXT_Y_START, yoff, dp); + cyberpro_grphw16(EXT_Y_END, yoff + wi->dst.height, dp); + cyberpro_grphw24(EXT_COLOUR_COMPARE, wi->dst.chromakey, dp); + cyberpro_grphw16(EXT_DDA_X_INC, xscale, dp); + cyberpro_grphw16(EXT_DDA_Y_INC, yscale, dp); + cyberpro_grphw8(EXT_VID_FMT, wi->vid_fmt, dp); + + if (wi->dst.flags & VIDEO_WINDOW_CHROMAKEY) + wi->vid_disp_ctl1 &= ~EXT_VID_DISP_CTL1_IGNORE_CCOMP; + else + wi->vid_disp_ctl1 |= EXT_VID_DISP_CTL1_IGNORE_CCOMP; +} + +/* + * Enable or disable the 1st overlay window. Note that for anything + * useful to be displayed, we must have capture enabled. + */ +static void +cyberpro_ext_ctl(struct cyberpro_vidinfo *dp, struct win_info *wi, int on) +{ + if (on) + wi->vid_disp_ctl1 |= EXT_VID_DISP_CTL1_ENABLE_WINDOW; + else + wi->vid_disp_ctl1 &= ~EXT_VID_DISP_CTL1_ENABLE_WINDOW; + + cyberpro_grphw8(EXT_VID_DISP_CTL1, wi->vid_disp_ctl1, dp); +} + +/*------------------------------- V2 Overlay Window ------------------------- + * Initialise 2nd overlay window (guesswork) + */ +static void +cyberpro_v2_init(struct cyberpro_vidinfo *dp, struct win_info *wi) +{ + wi->vid_fifo_ctl = 0xf87c; + wi->vid_fmt = EXT_VID_FMT_YUV422; + wi->vid_disp_ctl1 = EXT_VID_DISP_CTL1_VINTERPOL_OFF | + EXT_VID_DISP_CTL1_NOCLIP; + wi->vid_fifo_ctl1 = 0x06; + wi->vid_misc_ctl1 = 0; + + cyberpro_grphw8(REG_BANK, REG_BANK_Y, dp); + cyberpro_grphw8 (Y_V2_VID_DISP_CTL1, wi->vid_disp_ctl1, dp); + /* No DDA init values */ + cyberpro_grphw16(Y_V2_VID_FIFO_CTL, wi->vid_fifo_ctl, dp); + cyberpro_grphw8 (Y_V2_VID_FIFO_CTL1, wi->vid_fifo_ctl1, dp); +} + +/* + * Set the source parameters for the v2 window + */ +static void +cyberpro_v2_set_src(struct cyberpro_vidinfo *dp, struct win_info *wi) +{ + unsigned int phase, pitch; + + pitch = (wi->src.width >> 2) & 0x0fff; + phase = (wi->src.width + 3) >> 2; + + wi->vid_fmt &= ~7; + switch (wi->src.format) { + case VIDEO_PALETTE_RGB565: wi->vid_fmt |= EXT_VID_FMT_RGB565; break; + case VIDEO_PALETTE_RGB24: wi->vid_fmt |= EXT_VID_FMT_RGB888_24; break; + case VIDEO_PALETTE_RGB32: wi->vid_fmt |= EXT_VID_FMT_RGB888_32; break; + case VIDEO_PALETTE_RGB555: wi->vid_fmt |= EXT_VID_FMT_RGB555; break; + case VIDEO_PALETTE_YUV422: wi->vid_fmt |= EXT_VID_FMT_YUV422; break; + } + + cyberpro_grphw8(REG_BANK, REG_BANK_X, dp); + cyberpro_grphw24(X_V2_VID_MEM_START, wi->src.offset, dp); + cyberpro_grphw16(X_V2_VID_SRC_WIDTH, pitch | ((phase << 4) & 0xf000), dp); + cyberpro_grphw8 (X_V2_VID_SRC_WIN_WIDTH, phase, dp); + + cyberpro_grphw8(REG_BANK, REG_BANK_Y, dp); + cyberpro_grphw8(Y_V2_VID_FMT, wi->vid_fmt, dp); +} + +/* + * Set v2 window + */ +static void +cyberpro_v2_set_win(struct cyberpro_vidinfo *dp, struct win_info *wi) +{ + unsigned int xscale, yscale; + unsigned int xoff, yoff; + + /* + * Note: the offset does not appear to be influenced by + * hardware scrolling. + */ + xoff = yoff = 0; + + xoff += wi->dst.x; + yoff += wi->dst.y; + + xscale = (wi->src.width * 4096) / wi->dst.width; + yscale = (wi->src.height * 4096) / wi->dst.height; + + cyberpro_grphw8(REG_BANK, REG_BANK_X, dp); + cyberpro_grphw16(X_V2_X_START, xoff, dp); + cyberpro_grphw16(X_V2_X_END, xoff + wi->dst.width, dp); + cyberpro_grphw16(X_V2_Y_START, yoff, dp); + cyberpro_grphw16(X_V2_Y_END, yoff + wi->dst.height, dp); + + cyberpro_grphw8(REG_BANK, REG_BANK_Y, dp); + cyberpro_grphw16(Y_V2_DDA_X_INC, xscale, dp); + cyberpro_grphw16(Y_V2_DDA_Y_INC, yscale, dp); +} + +/* + * Enable or disable the 2nd overlay window. Note that for anything + * useful to be displayed, we must have capture enabled. + */ +static void +cyberpro_v2_ctl(struct cyberpro_vidinfo *dp, struct win_info *wi, int on) +{ + if (on) + wi->vid_disp_ctl1 |= EXT_VID_DISP_CTL1_ENABLE_WINDOW; + else + wi->vid_disp_ctl1 &= ~EXT_VID_DISP_CTL1_ENABLE_WINDOW; + + cyberpro_grphw8(REG_BANK, REG_BANK_Y, dp); + cyberpro_grphw8(Y_V2_VID_DISP_CTL1, wi->vid_disp_ctl1, dp); +} + +/*--------------------------- X2 Overlay Window ----------------------------- + * Initialise 3rd overlay window (guesswork) + */ +static void +cyberpro_x2_init(struct cyberpro_vidinfo *dp, struct win_info *wi) +{ + wi->vid_fmt = EXT_VID_FMT_YUV422; + wi->vid_disp_ctl1 = 0x40; + wi->vid_misc_ctl1 = 0; + + cyberpro_grphw8(REG_BANK, REG_BANK_K, dp); + cyberpro_grphw8 (K_X2_VID_DISP_CTL1, wi->vid_disp_ctl1, dp); + cyberpro_grphw16(K_X2_DDA_X_INIT, 0x0800, dp); + cyberpro_grphw16(K_X2_DDA_Y_INIT, 0x0800, dp); +} + +/* + * Set the source parameters for the x2 window + */ +static void +cyberpro_x2_set_src(struct cyberpro_vidinfo *dp, struct win_info *wi) +{ + unsigned int phase, pitch; + + pitch = (wi->src.width >> 2) & 0x0fff; + phase = (wi->src.width + 3) >> 2; + + wi->vid_fmt &= ~7; + switch (wi->src.format) { + case VIDEO_PALETTE_RGB565: wi->vid_fmt |= EXT_VID_FMT_RGB565; break; + case VIDEO_PALETTE_RGB24: wi->vid_fmt |= EXT_VID_FMT_RGB888_24; break; + case VIDEO_PALETTE_RGB32: wi->vid_fmt |= EXT_VID_FMT_RGB888_32; break; + case VIDEO_PALETTE_RGB555: wi->vid_fmt |= EXT_VID_FMT_RGB555; break; + case VIDEO_PALETTE_YUV422: wi->vid_fmt |= EXT_VID_FMT_YUV422; break; + } + + cyberpro_grphw8(REG_BANK, REG_BANK_J, dp); + cyberpro_grphw24(J_X2_VID_MEM_START, wi->src.offset, dp); + cyberpro_grphw16(J_X2_VID_SRC_WIDTH, pitch | ((phase << 4) & 0xf000), dp); + cyberpro_grphw8 (J_X2_VID_SRC_WIN_WIDTH, phase, dp); + + cyberpro_grphw8(REG_BANK, REG_BANK_K, dp); + cyberpro_grphw8(K_X2_VID_FMT, wi->vid_fmt, dp); +} + +/* + * Set x2 window + */ +static void +cyberpro_x2_set_win(struct cyberpro_vidinfo *dp, struct win_info *wi) +{ + unsigned int xscale, yscale; + unsigned int xoff, yoff; + + /* + * Note: the offset does not appear to be influenced by + * hardware scrolling. + */ + xoff = yoff = 0; + + xoff += wi->dst.x; + yoff += wi->dst.y; + + xscale = (wi->src.width * 4096) / wi->dst.width; + yscale = (wi->src.height * 4096) / wi->dst.height; + + cyberpro_grphw8(REG_BANK, REG_BANK_J, dp); + cyberpro_grphw16(J_X2_X_START, xoff, dp); + cyberpro_grphw16(J_X2_X_END, xoff + wi->dst.width, dp); + cyberpro_grphw16(J_X2_Y_START, yoff, dp); + cyberpro_grphw16(J_X2_Y_END, yoff + wi->dst.height, dp); + + cyberpro_grphw8(REG_BANK, REG_BANK_K, dp); + cyberpro_grphw16(K_X2_DDA_X_INC, xscale, dp); + cyberpro_grphw16(K_X2_DDA_Y_INC, yscale, dp); +} + +/* + * Enable or disable the 3rd overlay window. Note that for anything + * useful to be displayed, we must have capture enabled. + */ +static void +cyberpro_x2_ctl(struct cyberpro_vidinfo *dp, struct win_info *wi, int on) +{ + if (on) + wi->vid_disp_ctl1 |= EXT_VID_DISP_CTL1_ENABLE_WINDOW; + else + wi->vid_disp_ctl1 &= ~EXT_VID_DISP_CTL1_ENABLE_WINDOW; + + cyberpro_grphw8(REG_BANK, REG_BANK_K, dp); + cyberpro_grphw8(K_X2_VID_DISP_CTL1, wi->vid_disp_ctl1, dp); +} + +/* ------------------------------------------------------------------------- */ + +#if 0 +static void reset_seq(struct cyberpro_vidinfo *dp) +{ + unsigned char ext_mem_ctl = cyberpro_grphr8(0x70, dp); + + cyberpro_grphw8(ext_mem_ctl | 0x80, 0x70, dp); + cyberpro_grphw8(ext_mem_ctl, 0x70, dp); +} +#endif + +#ifdef USE_MMAP +/* + * Buffer support + */ +static int +cyberpro_alloc_frame_buffer(struct cyberpro_vidinfo *dp, + struct framebuf *frame) +{ + unsigned long addr; + void *buffer; + int pgidx; + + if (frame->buffer) + return 0; + + /* + * Allocate frame buffer + */ + buffer = vmalloc(NR_PAGES * PAGE_SIZE); + + if (frame->buffer) { + vfree(buffer); + return 0; + } + + if (!buffer) + return -ENOMEM; + + printk("Buffer allocated @ %p [", buffer); + + frame->buffer = buffer; + frame->dbg = 1; + + /* + * Don't leak information from the kernel. + */ + memset(buffer, 0x5a, NR_PAGES * PAGE_SIZE); + + /* + * Now, reserve all the pages, and calculate + * each pages' bus address. + */ + addr = (unsigned long)buffer; + for (pgidx = 0; pgidx < NR_PAGES; pgidx++, addr += PAGE_SIZE) { + struct page *page; + pgd_t *pgd; + pmd_t *pmd; + pte_t *pte; + + /* + * The page should be present. If not, + * vmalloc has gone nuts. + */ + pgd = pgd_offset_k(addr); + if (pgd_none(*pgd)) + BUG(); + pmd = pmd_offset(pgd, addr); + if (pmd_none(*pmd)) + BUG(); + pte = pte_offset(pmd, addr); + if (!pte_present(*pte)) + BUG(); + + page = pte_page(*pte); + + frame->bus_addr[pgidx] = virt_to_bus((void *)page_address(page)); + frame->pages[pgidx] = page; + SetPageReserved(page); + + printk("%08lx (%08lx) ", page_address(page), frame->bus_addr[pgidx]); + } + printk("\n"); + + return 0; +} + +static void +cyberpro_frames_free_one(struct cyberpro_vidinfo *dp, struct framebuf *frame) +{ + void *buffer; + int pgidx; + + frame->status = FRAME_FREE; + buffer = frame->buffer; + frame->buffer = NULL; + + if (buffer) { + for (pgidx = 0; pgidx < NR_PAGES; pgidx++) { + frame->bus_addr[pgidx] = 0; + ClearPageReserved(frame->pages[pgidx]); + frame->pages[pgidx] = NULL; + } + vfree(buffer); + } +} + +static void +cyberpro_busmaster_frame(struct cyberpro_vidinfo *dp, struct framebuf *frame) +{ + unsigned long bus_addr; + + bus_addr = frame->bus_addr[dp->bm_index]; + + if (frame->dbg) { + printk("Frame%d: %06x -> %08lx\n", + dp->frame_idx, + dp->bm_offset, + bus_addr); + } + + cyber2000_outw(dp->bm_offset, BM_VID_ADDR_LOW); + cyber2000_outw(dp->bm_offset >> 16, BM_VID_ADDR_HIGH); + + cyber2000_outw(bus_addr, BM_ADDRESS_LOW); + cyber2000_outw(bus_addr >> 16, BM_ADDRESS_HIGH); + + /* + * One page-full only + */ + cyber2000_outw(1023, BM_LENGTH); + + /* + * Load length + */ + cyber2000_outw(BM_CONTROL_INIT, BM_CONTROL); + + /* + * Enable transfer + */ + cyber2000_outw(BM_CONTROL_ENABLE|BM_CONTROL_IRQEN, BM_CONTROL); + + dp->bm_offset += 1024; + dp->bm_index += 1; +} + +static void cyberpro_busmaster_interrupt(struct cyberpro_vidinfo *dp) +{ + struct framebuf *frame = dp->frame + dp->frame_idx; + + /* + * Disable Busmaster operations + */ + cyber2000_outw(0, BM_CONTROL); + + if (frame->status == FRAME_GRABBING) { + /* + * We are still grabbing this frame to system + * memory. Transfer next page if there are + * more, or else flag this frame as complete. + */ + if (dp->bm_index < NR_PAGES) + cyberpro_busmaster_frame(dp); + else { + unsigned int idx; + + frame->status = FRAME_DONE; + frame->dbg = 0; + + idx = dp->frame_idx + 1; + if (idx >= NR_FRAMES) + idx = 0; + + dp->frame_idx = idx; + + wake_up(&dp->frame_wait); + } + } +} + +static void cyberpro_frames_vbl(struct cyberpro_vidinfo *dp, unsigned int stat) +{ + struct framebuf *frame = dp->frame + dp->frame_idx; + + /* + * No point capturing frames if the grabber isn't active. + */ + if (stat & EXT_ROM_UCB4GH_FREEZE) + return; + + /* + * If the next buffer is ready for grabbing, + * set up the bus master registers for the + * transfer. + */ + if (frame->status == FRAME_WAITING) { + frame->status = FRAME_GRABBING; + + dp->bm_offset = dp->cap_mem_offset; + dp->bm_index = 0; + + cyberpro_busmaster_frame(dp, frame); + } +} + +static void __init cyberpro_frames_init(struct cyberpro_vidinfo *dp) +{ + unsigned int offset, maxsize; + int i; + + init_waitqueue_head(&dp->frame_wait); + + maxsize = 2 * dp->cap.maxwidth * dp->cap.maxheight; + dp->frame_size = PAGE_ALIGN(maxsize); + dp->frame_idx = 0; + + for (i = offset = 0; i < NR_FRAMES; i++) { + dp->frame[i].offset = offset; + dp->frame[i].status = FRAME_FREE; + offset += dp->frame_size; + } +} + +static void cyberpro_frames_free(struct cyberpro_vidinfo *dp) +{ + int i; + + dp->mmaped = 0; + + /* + * Free all frame buffers + */ + for (i = 0; i < NR_FRAMES; i++) + cyberpro_frames_free_one(dp, dp->frame + i); +} + +#else +#define cyberpro_frames_vbl(dp,stat) do { } while (0) +#define cyberpro_frames_init(dp) do { } while (0) +#define cyberpro_frames_free(dp) do { } while (0) +#endif + +/* + * CyberPro Interrupts + * ------------------- + * + * We don't really know how to signal an IRQ clear to the chip. However, + * disabling and re-enabling the capture interrupt enable seems to do what + * we want. + */ +static void cyberpro_interrupt(int nr, void *dev_id, struct pt_regs *regs) +{ + struct cyberpro_vidinfo *dp = dev_id; + unsigned char old_grphidx; + unsigned int status; + + /* + * Save old graphics index register + */ + old_grphidx = cyberpro_readb(0x3ce, dp); + + status = cyberpro_grphr8(EXT_ROM_UCB4GH, dp); + + /* + * Was it due to the Capture VSYNC? + */ + if (status & EXT_ROM_UCB4GH_INTSTAT) { + /* + * Frob the IRQ enable bit to drop the request. + */ + cyberpro_grphw8(VFAC_CTL3, dp->vfac3 & ~VFAC_CTL3_CAP_IRQ, dp); + cyberpro_grphw8(VFAC_CTL3, dp->vfac3, dp); + + cyberpro_frames_vbl(dp, status); + wake_up(&dp->vbl_wait); + } + + /* + * Restore graphics controller index + */ + cyberpro_writeb(old_grphidx, 0x3ce, dp); + +#ifdef USE_MMAP + /* + * Do Bus-Master IRQ stuff + */ + if (cyber2000_inb(BM_CONTROL) & (1 << 7)) + cyberpro_busmaster_interrupt(dp); +#endif +} + +static void cyberpro_capture(struct cyberpro_vidinfo *dp, int on) +{ + DECLARE_WAITQUEUE(wait, current); + unsigned int status; + + status = cyberpro_grphr8(EXT_ROM_UCB4GH, dp); + + add_wait_queue(&dp->vbl_wait, &wait); + set_current_state(TASK_UNINTERRUPTIBLE); + + if (!!on ^ !(status & EXT_ROM_UCB4GH_FREEZE)) { + if (on) { + schedule_timeout(40 * HZ / 1000); + dp->vfac1 &= ~(VFAC_CTL1_FREEZE_CAPTURE|VFAC_CTL1_FREEZE_CAPTURE_SYNC); + cyberpro_grphw8(VFAC_CTL1, dp->vfac1, dp); + + status = cyberpro_grphr8(EXT_ROM_UCB4GH, dp); + } else { + dp->vfac1 |= VFAC_CTL1_FREEZE_CAPTURE_SYNC; + cyberpro_grphw8(VFAC_CTL1, dp->vfac1, dp); + + status = cyberpro_grphr8(EXT_ROM_UCB4GH, dp); + if (!(status & EXT_ROM_UCB4GH_FREEZE)) + schedule_timeout(40 * HZ / 1000); + } + } + + current->state = TASK_RUNNING; + remove_wait_queue(&dp->vbl_wait, &wait); +} + +static void cyberpro_capture_one(struct cyberpro_vidinfo *dp) +{ + struct task_struct *tsk = current; + DECLARE_WAITQUEUE(wait, tsk); + unsigned int status; + unsigned long policy, rt_priority; + + policy = tsk->policy; + rt_priority = tsk->rt_priority; + + tsk->policy = SCHED_FIFO; + tsk->rt_priority = 1; + + status = cyberpro_grphr8(EXT_ROM_UCB4GH, dp); + + add_wait_queue(&dp->vbl_wait, &wait); + set_current_state(TASK_UNINTERRUPTIBLE); + + schedule_timeout(40 * HZ / 1000); + dp->vfac1 &= ~(VFAC_CTL1_FREEZE_CAPTURE|VFAC_CTL1_FREEZE_CAPTURE_SYNC); + cyberpro_grphw8(VFAC_CTL1, dp->vfac1, dp); + + status = cyberpro_grphr8(EXT_ROM_UCB4GH, dp); + set_current_state(TASK_UNINTERRUPTIBLE); + schedule_timeout(40 * HZ / 1000); + set_current_state(TASK_UNINTERRUPTIBLE); + schedule_timeout(40 * HZ / 1000); + + dp->vfac1 |= VFAC_CTL1_FREEZE_CAPTURE_SYNC; + cyberpro_grphw8(VFAC_CTL1, dp->vfac1, dp); + + set_current_state(TASK_UNINTERRUPTIBLE); + status = cyberpro_grphr8(EXT_ROM_UCB4GH, dp); + + current->state = TASK_RUNNING; + remove_wait_queue(&dp->vbl_wait, &wait); + + tsk->policy = policy; + tsk->rt_priority = rt_priority; +} + +static void cyberpro_capture_set_win(struct cyberpro_vidinfo *dp) +{ + unsigned int xstart, xend, ystart, yend; + + xstart = 4 + dp->capt.x; + xend = xstart + dp->capt.width; + + if (dp->cap_mode1 & EXT_CAP_MODE1_8BIT) { + /* 8-bit capture */ + xstart *= 2; + xend *= 2; + } + + xstart -= 1; + xend -= 1; + + ystart = 18 + dp->capt.y; + yend = ystart + dp->capt.height / 2; + + cyberpro_grphw16(CAP_X_START, xstart, dp); + cyberpro_grphw16(CAP_X_END, xend + 1, dp); + cyberpro_grphw16(CAP_Y_START, ystart, dp); + cyberpro_grphw16(CAP_Y_END, yend + 2, dp); + + /* + * This should take account of capt.decimation + */ + cyberpro_grphw16(CAP_DDA_X_INIT, 0x0800, dp); + cyberpro_grphw16(CAP_DDA_X_INC, 0x1000, dp); + cyberpro_grphw16(CAP_DDA_Y_INIT, 0x0800, dp); + cyberpro_grphw16(CAP_DDA_Y_INC, 0x1000, dp); + + cyberpro_grphw8(CAP_PITCH, dp->capt.width >> 2, dp); +} + +static void cyberpro_set_interlace(struct cyberpro_vidinfo *dp) +{ + /* + * set interlace mode + */ + if (dp->interlace) { + dp->vfac3 |= VFAC_CTL3_CAP_INTERLACE; + dp->cap_miscctl &= ~CAP_CTL_MISC_ODDEVEN; + dp->ovl->src.height = dp->capt.height; + } else { + dp->vfac3 &= ~VFAC_CTL3_CAP_INTERLACE; + dp->cap_miscctl |= CAP_CTL_MISC_ODDEVEN; + dp->ovl->src.height = dp->capt.height / 2; + } + + cyberpro_grphw8(VFAC_CTL3, dp->vfac3, dp); + cyberpro_grphw8(CAP_CTL_MISC, dp->cap_miscctl, dp); + + dp->ovl->set_src(dp, dp->ovl); + + if (dp->win_set) + dp->ovl->set_win(dp, dp->ovl); +} + +/* + * Calculate and set the address of the capture buffer. Note we + * also update the extended memory buffer for the overlay window. + * + * base: phys base address of display + * width: pixel width of display + * height: height of display + * depth: depth of display (8/16/24) + * bytesperline: number of bytes on a line + * + * We place the capture buffer 16K after the screen. + */ +static int +cyberpro_set_buffer(struct cyberpro_vidinfo *dp, struct video_buffer *b) +{ + unsigned long screensize, maxbufsz; + + if (b->height <= 0 || b->width <= 0 || b->bytesperline <= 0) + return -EINVAL; + + maxbufsz = dp->cap.maxwidth * dp->cap.maxheight * 2; + screensize = b->height * b->bytesperline + 16384; + + if ((screensize + maxbufsz) >= dp->info.fb_size) + return -EINVAL; + + dp->buf.base = b->base; + dp->buf.width = b->width; + dp->buf.height = b->height; + dp->buf.depth = b->depth; + dp->buf.bytesperline = b->bytesperline; + dp->cap_mem_offset = screensize >> 2; + + cyberpro_grphw24(CAP_MEM_START, dp->cap_mem_offset, dp); + + /* + * Setup the overlay source information. + */ + dp->ovl->src.offset = dp->cap_mem_offset; + dp->ovl->set_src(dp, dp->ovl); + + return 0; +} + +static void cyberpro_hw_init(struct cyberpro_vidinfo *dp) +{ + unsigned char old; + + /* + * Enable access to bus-master registers + */ + dp->info.enable_extregs(dp->info.info); + + dp->vfac1 = VFAC_CTL1_PHILIPS | + VFAC_CTL1_FREEZE_CAPTURE | + VFAC_CTL1_FREEZE_CAPTURE_SYNC; + dp->vfac3 = VFAC_CTL3_CAP_IRQ; + + dp->cap_miscctl = CAP_CTL_MISC_DISPUSED | + CAP_CTL_MISC_SYNCTZOR | + CAP_CTL_MISC_SYNCTZHIGH; + + /* + * Setup bus-master mode + */ + cyberpro_grphw8(BM_CTRL1, 0x88, dp); + cyberpro_grphw8(PCI_BM_CTL, PCI_BM_CTL_ENABLE, dp); + cyberpro_grphw8(BM_CTRL0, 0x44, dp); + cyberpro_grphw8(BM_CTRL1, 0x84, dp); + + cyberpro_grphw24(CAP_MEM_START, 0, dp); + + cyberpro_grphw8(VFAC_CTL1, dp->vfac1, dp); + cyberpro_grphw8(VFAC_CTL3, dp->vfac3, dp); + cyberpro_grphw8(VFAC_CTL2, 0, dp); + + cyberpro_grphw8(REG_BANK, REG_BANK_Y, dp); + cyberpro_grphw8(EXT_TV_CTL, 0x80, dp); + + cyberpro_grphw8(EXT_CAP_CTL1, 0x3f, dp); /* disable PIP */ + cyberpro_grphw8(EXT_CAP_CTL2, 0xc0 | EXT_CAP_CTL2_ODDFRAMEIRQ, dp); + + /* + * Configure capture mode to match the + * external video processor format + */ + cyberpro_grphw8(EXT_CAP_MODE1, dp->cap_mode1, dp); + cyberpro_grphw8(EXT_CAP_MODE2, dp->cap_mode2, dp); + + /* setup overlay */ + cyberpro_grphw16(EXT_FIFO_CTL, 0x1010, dp); +// cyberpro_grphw16(EXT_FIFO_CTL, 0x1b0f, dp); + + /* + * Always reset the capture parameters on each open. + */ + dp->capt.x = 0; + dp->capt.y = 0; + dp->capt.width = dp->cap.maxwidth; + dp->capt.height = dp->cap.maxheight; + dp->capt.decimation = 0; + dp->capt.flags = 0; + + cyberpro_capture_set_win(dp); + + /* + * Enable VAFC + */ + old = cyberpro_grphr8(EXT_LATCH1, dp); + cyberpro_grphw8(EXT_LATCH1, old | EXT_LATCH1_VAFC_EN, dp); + + /* + * Enable capture (we hope that VSYNC=1) + */ + dp->vfac1 |= VFAC_CTL1_CAPTURE; + cyberpro_grphw8(VFAC_CTL1, dp->vfac1, dp); + + /* + * The overlay source format is always the + * same as the capture stream format. + */ + dp->ovl->src.width = dp->capt.width; + dp->ovl->src.height = dp->capt.height; + dp->ovl->src.format = dp->stream_fmt; + + /* + * Initialise the overlay windows + */ + dp->ext.init(dp, &dp->ext); + dp->v2.init(dp, &dp->v2); + dp->x2.init(dp, &dp->x2); +} + +static void cyberpro_deinit(struct cyberpro_vidinfo *dp) +{ + unsigned char old; + + /* + * Stop any bus-master activity + */ + cyberpro_writew(0, BM_CONTROL, dp); + + /* + * Shut down overlay + */ + if (dp->ovl_active) + dp->ovl->ctl(dp, dp->ovl, 0); + dp->ovl_active = 0; + + /* + * Shut down capture + */ + if (dp->cap_active) + cyberpro_capture(dp, 0); + dp->cap_active = 0; + + /* + * Disable all capture + */ + cyberpro_grphw8(VFAC_CTL1, 0, dp); + + /* + * Disable VAFC + */ + old = cyberpro_grphr8(EXT_LATCH1, dp); + cyberpro_grphw8(EXT_LATCH1, old & ~EXT_LATCH1_VAFC_EN, dp); + + /* + * Disable interrupt (this allows it to float) + */ + dp->vfac3 &= ~VFAC_CTL3_CAP_IRQ; + cyberpro_grphw8(VFAC_CTL3, dp->vfac3, dp); + + /* + * Switch off bus-master mode + */ + cyberpro_grphw8(PCI_BM_CTL, 0, dp); + + /* + * Disable access to bus-master registers + */ + dp->info.disable_extregs(dp->info.info); +} + +static int cyberpro_grabber_open(struct video_device *dev, int flags) +{ + struct cyberpro_vidinfo *dp = dev->priv; + int ret, one = 1; + + MOD_INC_USE_COUNT; + + ret = -EBUSY; + if (flags || dp->users) + goto out; + + dp->users += 1; + + if (dp->users == 1) { + ret = request_irq(dp->irq, cyberpro_interrupt, SA_SHIRQ, + dp->info.dev_name, dp); + + if (ret) { + dp->users -= 1; + goto out; + } + + /* + * Initialise the VGA chip + */ + cyberpro_hw_init(dp); + + /* + * Enable the IRQ. This allows the IRQ to work as expected + * even if the IRQ line is missing the pull-up resistor. + */ + enable_irq(dp->irq); + + i2c_control_device(dp->bus, I2C_DRIVERID_VIDEODECODER, + DECODER_ENABLE_OUTPUT, &one); + } + + ret = 0; +out: + if (ret) + MOD_DEC_USE_COUNT; + return ret; +} + +static void cyberpro_grabber_close(struct video_device *dev) +{ + struct cyberpro_vidinfo *dp = dev->priv; + + if (dp->users == 1) { + int zero = 0; + + /* + * Disable the IRQ. This prevents problems with missing + * pull-up resistors on the PCI interrupt line. + */ + disable_irq(dp->irq); + + cyberpro_frames_free(dp); + + /* + * Turn off the SAA7111 decoder + */ + i2c_control_device(dp->bus, I2C_DRIVERID_VIDEODECODER, + DECODER_ENABLE_OUTPUT, &zero); + + /* + * Disable grabber + */ + cyberpro_deinit(dp); + + free_irq(dp->irq, dp); + } + + dp->users -= 1; + + MOD_DEC_USE_COUNT; +} + +/* + * Our general plan here is: + * 1. Set the CyberPro to perform a BM-DMA of one frame to this memory + * 2. Copy the frame to the userspace + * + * However, BM-DMA seems to be unreliable at the moment, especially on + * rev. 4 NetWinders. + */ +static long +cyberpro_grabber_read(struct video_device *dev, char *buf, + unsigned long count, int noblock) +{ + struct cyberpro_vidinfo *dp = dev->priv; + int ret = -EINVAL; + +#ifdef USE_MMIO + unsigned long maxbufsz = dp->capt.width * dp->capt.height * 2; + char *disp = dp->info.fb + (dp->cap_mem_offset << 2); + + /* + * If the buffer is mmap()'d, we shouldn't be using read() + */ + if (dp->mmaped) + return -EINVAL; + + if (count > maxbufsz) + count = maxbufsz; + + if (dp->cap_active) + cyberpro_capture(dp, 0); + else + cyberpro_capture_one(dp); + + ret = (int)count; + if (copy_to_user(buf, disp, count)) + ret = -EFAULT; + + /* + * unfreeze capture + */ + if (dp->cap_active) + cyberpro_capture(dp, 1); +#endif + + return ret; +} + +/* + * We don't support writing to the grabber + * (In theory, we could allow writing to a separate region of VGA memory, + * and display this using the second overlay window. This would allow us + * to do video conferencing for example). + */ +static long +cyberpro_grabber_write(struct video_device *dev, const char *buf, + unsigned long count, int noblock) +{ + return -EINVAL; +} + +static int +cyberpro_grabber_ioctl(struct video_device *dev, unsigned int cmd, void *arg) +{ + struct cyberpro_vidinfo *dp = dev->priv; + + switch (cmd) { + case VIDIOCGCAP: + return copy_to_user(arg, &dp->cap, sizeof(dp->cap)) + ? -EFAULT : 0; + + case VIDIOCGCHAN: + { + struct video_channel chan; + + chan.channel = 0; + strcpy(chan.name, "Composite"); + chan.tuners = 0; + chan.flags = 0; + chan.type = VIDEO_TYPE_CAMERA; + chan.norm = dp->norm; + + return copy_to_user(arg, &chan, sizeof(chan)) ? -EFAULT : 0; + } + + case VIDIOCGPICT: + return copy_to_user(arg, &dp->pic, sizeof(dp->pic)) + ? -EINVAL : 0; + + case VIDIOCGWIN: + { + struct video_window win; + + win.x = dp->ovl->dst.x; + win.y = dp->ovl->dst.y; + win.width = dp->ovl->dst.width; + win.height = dp->ovl->dst.height; + win.chromakey = dp->ovl->dst.chromakey; + win.flags = VIDEO_WINDOW_CHROMAKEY | + (dp->interlace ? VIDEO_WINDOW_INTERLACE : 0); + win.clips = NULL; + win.clipcount = 0; + + return copy_to_user(arg, &win, sizeof(win)) + ? -EINVAL : 0; + } + + case VIDIOCGFBUF: + return copy_to_user(arg, &dp->buf, sizeof(dp->buf)) + ? -EINVAL : 0; + + case VIDIOCGUNIT: + { + struct video_unit unit; + + unit.video = dev->minor; + unit.vbi = VIDEO_NO_UNIT; + unit.radio = VIDEO_NO_UNIT; + unit.audio = VIDEO_NO_UNIT; + unit.teletext = VIDEO_NO_UNIT; + + return copy_to_user(arg, &unit, sizeof(unit)) + ? -EINVAL : 0; + } + + case VIDIOCGCAPTURE: + return copy_to_user(arg, &dp->capt, sizeof(dp->capt)) + ? -EFAULT : 0; + + case VIDIOCSCHAN: + { + struct video_decoder_capability vdc; + struct video_channel v; + int ok; + + if (copy_from_user(&v, arg, sizeof(v))) + return -EFAULT; + + if (v.channel != 0) + return -EINVAL; + + i2c_control_device(dp->bus, I2C_DRIVERID_VIDEODECODER, + DECODER_GET_CAPABILITIES, &vdc); + + switch (v.norm) { + case VIDEO_MODE_PAL: + ok = vdc.flags & VIDEO_DECODER_PAL; + break; + case VIDEO_MODE_NTSC: + ok = vdc.flags & VIDEO_DECODER_NTSC; + break; + case VIDEO_MODE_AUTO: + ok = vdc.flags & VIDEO_DECODER_AUTO; + break; + default: + ok = 0; + } + if (!ok) + return -EINVAL; + + dp->norm = v.norm; + + i2c_control_device(dp->bus, I2C_DRIVERID_VIDEODECODER, + DECODER_SET_NORM, &v.norm); + + return 0; + } + + case VIDIOCSPICT: + { + struct video_picture p; + + if (copy_from_user(&p, arg, sizeof(p))) + return -EFAULT; + + if (p.palette != dp->stream_fmt || + p.depth != 8) + return -EINVAL; + + dp->pic = p; + + /* p.depth sets the capture depth */ + /* p.palette sets the capture palette */ + + i2c_control_device(dp->bus, I2C_DRIVERID_VIDEODECODER, + DECODER_SET_PICTURE, &p); + + return 0; + } + + case VIDIOCSWIN: /* set the size & position of the overlay window */ + { + struct video_window w; + int diff; + + if (!dp->buf_set) + return -EINVAL; + + if (copy_from_user(&w, arg, sizeof(w))) + return -EFAULT; + + if (w.clipcount) + return -EINVAL; + + /* + * Bound the overlay window by the size of the screen + */ + if (w.x < 0) + w.x = 0; + if (w.y < 0) + w.y = 0; + + if (w.x > dp->buf.width) + w.x = dp->buf.width; + if (w.y > dp->buf.height) + w.y = dp->buf.height; + + if (w.width < dp->capt.width) + w.width = dp->capt.width; + if (w.height < dp->capt.height) + w.height = dp->capt.height; + + if (w.x + w.width > dp->buf.width) + w.width = dp->buf.width - w.x; + if (w.y + w.height > dp->buf.height) + w.height = dp->buf.height - w.y; + + /* + * We've tried to make the values fit, but + * they just won't. + */ + if (w.width < dp->capt.width || w.height < dp->capt.height) + return -EINVAL; + + diff = dp->ovl->dst.x != w.x || + dp->ovl->dst.y != w.y || + dp->ovl->dst.width != w.width || + dp->ovl->dst.height != w.height || + dp->ovl->dst.chromakey != w.chromakey || + dp->ovl->dst.flags != w.flags; + + if (!dp->win_set || diff) { + dp->ovl->dst.x = w.x; + dp->ovl->dst.y = w.y; + dp->ovl->dst.width = w.width; + dp->ovl->dst.height = w.height; + dp->ovl->dst.chromakey = w.chromakey; + dp->ovl->dst.flags = w.flags; + + if (dp->ovl_active) + dp->ovl->ctl(dp, dp->ovl, 0); + + dp->ovl->set_win(dp, dp->ovl); + + if (dp->ovl_active) + dp->ovl->ctl(dp, dp->ovl, 1); + + diff = w.flags & VIDEO_WINDOW_INTERLACE ? 1 : 0; + if (!dp->win_set || dp->interlace != diff) { + dp->interlace = diff; + cyberpro_set_interlace(dp); + } + } + + dp->win_set = 1; + + return 0; + } + + case VIDIOCSFBUF: /* set frame buffer info */ + { + struct video_buffer b; + int ret; + + if (!capable(CAP_SYS_ADMIN) || !capable(CAP_SYS_RAWIO)) + return -EPERM; + + if (dp->cap_active) + return -EINVAL; + + if (copy_from_user(&b, arg, sizeof(b))) + return -EFAULT; + + ret = cyberpro_set_buffer(dp, &b); + if (ret == 0) { + dp->buf_set = 1; + dp->win_set = 0; + } + + return ret; + } + + case VIDIOCCAPTURE: + { + int on; + + if (get_user(on, (int *)arg)) + return -EFAULT; + + if (( on && dp->ovl_active) || + (!on && !dp->ovl_active)) + return 0; + + if (on && (!dp->buf_set || !dp->win_set)) + return -EINVAL; + + cyberpro_capture(dp, on); + dp->cap_active = on; + dp->ovl->ctl(dp, dp->ovl, on); + dp->ovl_active = on; + + return 0; + } + +#ifdef USE_MMAP + case VIDIOCSYNC: + { + DECLARE_WAITQUEUE(wait, current); + int buf; + + /* + * The buffer must have been mmaped + * for this call to work. + */ + if (!dp->mmaped) + return -EINVAL; + + if (get_user(buf, (int *)arg)) + return -EFAULT; + + if (buf < 0 || buf >= NR_FRAMES) + return -EINVAL; + + switch (dp->frame[buf].status) { + case FRAME_FREE: + return -EINVAL; + + case FRAME_WAITING: + case FRAME_GRABBING: + add_wait_queue(&dp->frame_wait, &wait); + while (1) { + set_current_state(TASK_INTERRUPTIBLE); + if (signal_pending(current)) + break; + if (dp->frame[buf].status == FRAME_DONE) + break; + schedule(); + } + set_current_state(TASK_RUNNING); + remove_wait_queue(&dp->frame_wait, &wait); + if (signal_pending(current)) + return -EINTR; + /*FALLTHROUGH*/ + case FRAME_DONE: + dp->frame[buf].status = FRAME_FREE; + break; + } + return 0; + } + + case VIDIOCMCAPTURE: + { + struct video_mmap vmap; + + /* + * The buffer must have been mmaped + * for this call to work. + */ + if (!dp->mmaped) + return -EINVAL; + + if (copy_from_user(&vmap, arg, sizeof(vmap))) + return -EFAULT; + + /* + * We can only capture in our source format/size. + */ + if (vmap.frame >= NR_FRAMES || + vmap.format != dp->stream_fmt || + vmap.width != dp->capt.width || + vmap.height != dp->capt.height) + return -EINVAL; + + if (dp->frame[vmap.frame].status == FRAME_WAITING || + dp->frame[vmap.frame].status == FRAME_GRABBING) + return -EBUSY; + + dp->frame[vmap.frame].status = FRAME_WAITING; + return 0; + } + + case VIDIOCGMBUF: + { + struct video_mbuf vmb; + unsigned int i; + + vmb.frames = NR_FRAMES; + vmb.size = dp->frame_size * NR_FRAMES; + + for (i = 0; i < NR_FRAMES; i++) + vmb.offsets[i] = dp->frame[i].offset; + + return copy_to_user(arg, &vmb, sizeof(vmb)) ? -EFAULT : 0; + } +#endif + + case VIDIOCSCAPTURE: + { + struct video_capture capt; + +#ifndef ALLOW_SCAPTURE_WHILE_CAP + if (dp->cap_active) + return -EINVAL; +#endif + + if (copy_from_user(&capt, arg, sizeof(capt))) + return -EFAULT; + + if (capt.x < 0 || capt.width < 0 || + capt.y < 0 || capt.height < 0 || + capt.x + capt.width > dp->cap.maxwidth || + capt.y + capt.height > dp->cap.maxheight) + return -EINVAL; + + /* + * The capture width must be a multiple of 4 + */ + if (dp->capt.width & 3) + return -EINVAL; + + dp->capt.x = capt.x; + dp->capt.y = capt.y; + dp->capt.width = capt.width; + dp->capt.height = capt.height; +#ifdef ALLOW_SCAPTURE_WHILE_CAP + if (dp->ovl_active) + dp->ovl->ctl(dp, dp->ovl, 0); + if (dp->cap_active) + cyberpro_capture(dp, 0); +#endif + cyberpro_capture_set_win(dp); + + /* + * Update the overlay window information + */ + dp->ovl->src.width = capt.width; + dp->ovl->src.height = capt.height; + + dp->ovl->set_src(dp, dp->ovl); + if (dp->win_set) + dp->ovl->set_win(dp, dp->ovl); + +#ifdef ALLOW_SCAPTURE_WHILE_CAP + if (dp->cap_active) + cyberpro_capture(dp, 1); + if (dp->ovl_active) + dp->ovl->ctl(dp, dp->ovl, 1); +#endif + return 0; + } + + case VIDIOCGTUNER: /* no tuner */ + case VIDIOCSTUNER: + return -EINVAL; + } + + return -EINVAL; +} + +#ifdef USE_MMAP +static int +cyberpro_grabber_mmap(struct video_device *dev, const char *addr, unsigned long size) +{ + struct cyberpro_vidinfo *dp = dev->priv; + unsigned long vaddr = (unsigned long)addr; + pgprot_t prot; + int frame_idx, ret = -EINVAL; + +#if defined(__arm__) + prot = __pgprot(L_PTE_PRESENT | L_PTE_YOUNG | L_PTE_USER | L_PTE_WRITE | L_PTE_DIRTY); +#elif defined(__i386__) + prot = __pgprot(_PAGE_PRESENT | _PAGE_RW | _PAGE_USER | _PAGE_ACCESSED); + if (boot_cpu_data.x86 > 3) + pgprot_val(prot) |= _PAGE_PCD; +#else +#error "Unsupported architecture" +#endif + + /* + * The mmap() request must have the correct size. + */ + if (size != NR_FRAMES * dp->frame_size) + goto out; + + /* + * If it's already mapped, don't re-do + */ + if (dp->mmaped) + goto out; + dp->mmaped = 1; + + /* + * Map in each frame + */ + for (frame_idx = 0; frame_idx < NR_FRAMES; frame_idx++) { + struct framebuf *frame; + int pgidx; + + frame = dp->frame + frame_idx; + + ret = cyberpro_alloc_frame_buffer(dp, frame); + + /* + * If an error occurs, we can be lazy and leave what we've + * been able to do. Our release function will free any + * allocated buffers, and do_mmap_pgoff() will zap any + * inserted mappings. + */ + if (ret) + goto out2; + + /* + * Map in each page on a page by page basis. This is just + * a little on the inefficient side, but it's only run once. + */ + for (pgidx = 0; pgidx < NR_PAGES; pgidx++) { + unsigned long virt; + + virt = page_address(frame->pages[pgidx]); + + ret = remap_page_range(vaddr, virt_to_phys((void *)virt), + PAGE_SIZE, prot); + + if (ret) + goto out2; + + vaddr += PAGE_SIZE; + } + } + + out2: + if (ret) + dp->mmaped = 0; + out: + return ret; +} +#endif + +static int __init cyberpro_grabber_init_done(struct video_device *dev) +{ + struct cyberpro_vidinfo *dp; + struct cyberpro_info *info = dev->priv; + int ret; + + dp = kmalloc(sizeof(*dp), GFP_KERNEL); + if (!dp) + return -ENOMEM; + + memset(dp, 0, sizeof(*dp)); + + dev->priv = dp; + dp->info = *info; + dp->dev = dev; + dp->bus = &cyberpro_i2c_bus; + dp->regs = info->regs; + dp->irq = info->dev->irq; + + strcpy(dp->cap.name, dev->name); + dp->cap.type = dev->type; + dp->cap.channels = 1; + dp->cap.audios = 0; + dp->cap.minwidth = 32; + dp->cap.maxwidth = 716; + dp->cap.minheight = 32; + dp->cap.maxheight = 576; + + dp->pic.brightness = 32768; + dp->pic.hue = 32768; + dp->pic.colour = 32768; + dp->pic.contrast = 32768; + dp->pic.whiteness = 0; + dp->pic.depth = 8; + dp->pic.palette = VIDEO_PALETTE_YUV422; + + /* dp->buf is setup by the user */ + /* dp->cap_mem_offset setup by dp->buf */ + + dp->norm = VIDEO_MODE_AUTO; + + /* + * The extended overlay window + */ + dp->ext.init = cyberpro_ext_init; + dp->ext.set_src = cyberpro_ext_set_src; + dp->ext.set_win = cyberpro_ext_set_win; + dp->ext.ctl = cyberpro_ext_ctl; + + /* + * The V2 overlay window + */ + dp->v2.init = cyberpro_v2_init; + dp->v2.set_src = cyberpro_v2_set_src; + dp->v2.set_win = cyberpro_v2_set_win; + dp->v2.ctl = cyberpro_v2_ctl; + + /* + * The X2 overlay window + */ + dp->x2.init = cyberpro_x2_init; + dp->x2.set_src = cyberpro_x2_set_src; + dp->x2.set_win = cyberpro_x2_set_win; + dp->x2.ctl = cyberpro_x2_ctl; + + /* + * Set the overlay window which we shall be using + */ + dp->ovl = &dp->ext; + + dp->cap_mode1 = EXT_CAP_MODE1_ALTFIFO; + + /* + * Initialise hardware specific values. + * - CCIR656 8bit mode (YUV422 data) + * - Ignore Hgood signal + * - Invert Odd/Even field signal + */ + dp->cap_mode1 |= EXT_CAP_MODE1_CCIR656 | EXT_CAP_MODE1_8BIT; + dp->cap_mode2 = EXT_CAP_MODE2_FIXSONY | EXT_CAP_MODE2_DATEND | + EXT_CAP_MODE2_CCIRINVOE; + dp->stream_fmt = VIDEO_PALETTE_YUV422; + + + init_waitqueue_head(&dp->vbl_wait); + cyberpro_frames_init(dp); + + /* + * wake up the decoder + */ + decoder_sleep(0); + + dp->bus->data = dp; + strncpy(dp->bus->name, dev->name, sizeof(dp->bus->name)); + + pci_set_master(dp->info.dev); + + ret = i2c_register_bus(dp->bus); + + /* + * If we successfully registered the bus, but didn't initialise + * the decoder (because its driver is not present), request + * that it is loaded. + */ + if (ret == 0 && !dp->decoder) + request_module("saa7111"); + + /* + * If that didn't work, then we're out of luck. + */ + if (ret == 0 && !dp->decoder) { + i2c_unregister_bus(dp->bus); + ret = -ENXIO; + } + + if (ret) { + kfree(dp); + + /* + * put the decoder back to sleep + */ + decoder_sleep(1); + } + + return ret; +} + +static struct video_device cyberpro_grabber = { + name: "", + type: VID_TYPE_CAPTURE | VID_TYPE_OVERLAY | + VID_TYPE_CHROMAKEY | VID_TYPE_SCALES | + VID_TYPE_SUBCAPTURE, + hardware: 0, + open: cyberpro_grabber_open, + close: cyberpro_grabber_close, + read: cyberpro_grabber_read, + write: cyberpro_grabber_write, + ioctl: cyberpro_grabber_ioctl, +#ifdef USE_MMAP + mmap: cyberpro_grabber_mmap, +#endif + initialize: cyberpro_grabber_init_done, +}; + +int init_cyber2000fb_viddev(void) +{ + struct cyberpro_info info; + + if (!cyber2000fb_attach(&info, 0)) + return -ENXIO; + + strncpy(cyberpro_grabber.name, info.dev_name, sizeof(cyberpro_grabber.name)); + + cyberpro_grabber.priv = &info; + + return video_register_device(&cyberpro_grabber, VFL_TYPE_GRABBER, -1); +} + +/* + * This can be cleaned up when the SAA7111 code is fixed. + */ +#ifdef MODULE +static int __init cyberpro_init(void) +{ + disable_irq(35); + return init_cyber2000fb_viddev(); +} + +static void __exit cyberpro_exit(void) +{ + video_unregister_device(&cyberpro_grabber); + kfree(cyberpro_grabber.priv); + i2c_unregister_bus(&cyberpro_i2c_bus); + + /* + * put the decoder back to sleep + */ + decoder_sleep(1); + + cyber2000fb_detach(0); +} + +module_init(cyberpro_init); +module_exit(cyberpro_exit); +#endif diff -urN orig/drivers/media/video/i2c-old.c linux/drivers/media/video/i2c-old.c --- orig/drivers/media/video/i2c-old.c Sun Oct 14 20:53:08 2001 +++ linux/drivers/media/video/i2c-old.c Fri Sep 28 21:43:57 2001 @@ -36,22 +36,20 @@ static struct i2c_driver *drivers[I2C_DRIVER_MAX]; static int bus_count = 0, driver_count = 0; -#ifdef CONFIG_VIDEO_BUZ extern int saa7111_init(void); extern int saa7185_init(void); -#endif -#ifdef CONFIG_VIDEO_LML33 extern int bt819_init(void); extern int bt856_init(void); -#endif int i2c_init(void) { printk(KERN_INFO "i2c: initialized%s\n", scan ? " (i2c bus scan enabled)" : ""); /* anything to do here ? */ -#ifdef CONFIG_VIDEO_BUZ +#if defined(CONFIG_VIDEO_BUZ) || defined(CONFIG_VIDEO_CYBERPRO) saa7111_init(); +#endif +#ifdef CONFIG_VIDEO_BUZ saa7185_init(); #endif #ifdef CONFIG_VIDEO_LML33 diff -urN orig/drivers/media/video/saa7111.c linux/drivers/media/video/saa7111.c --- orig/drivers/media/video/saa7111.c Sun Oct 14 20:53:08 2001 +++ linux/drivers/media/video/saa7111.c Thu Jun 20 23:22:47 2002 @@ -20,9 +20,9 @@ along with this program; if not, write to the Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ - -#include +#include #include +#include #include #include #include @@ -149,7 +149,11 @@ 0x0d, 0x00, /* 0d - HUE=0 */ 0x0e, 0x01, /* 0e - CDTO=0, CSTD=0, DCCF=0, FCTC=0, CHBW=1 */ 0x0f, 0x00, /* 0f - reserved */ +#ifndef CONFIG_ARCH_NETWINDER 0x10, 0x48, /* 10 - OFTS=1, HDEL=0, VRLN=1, YDEL=0 */ +#else + 0x10, 0xc8, /* 10 - OFTS=YUV-CCIR656, HDEL=0, VLRN=1, YDEL=0 */ +#endif 0x11, 0x1c, /* 11 - GPSW=0, CM99=0, FECO=0, COMPO=1, OEYC=1, OEHV=1, VIPB=0, COLO=0 */ 0x12, 0x00, /* 12 - output control 2 */ 0x13, 0x00, /* 13 - output control 3 */ diff -urN orig/drivers/message/i2o/i2o_core.c linux/drivers/message/i2o/i2o_core.c --- orig/drivers/message/i2o/i2o_core.c Mon Aug 5 13:31:00 2002 +++ linux/drivers/message/i2o/i2o_core.c Mon Aug 5 15:02:24 2002 @@ -1665,14 +1665,14 @@ } memset(status, 0, 4); - msg[0]=EIGHT_WORD_MSG_SIZE|SGL_OFFSET_0; - msg[1]=I2O_CMD_ADAPTER_RESET<<24|HOST_TID<<12|ADAPTER_TID; - msg[2]=core_context; - msg[3]=0; - msg[4]=0; - msg[5]=0; - msg[6]=virt_to_bus(status); - msg[7]=0; /* 64bit host FIXME */ + writel(EIGHT_WORD_MSG_SIZE|SGL_OFFSET_0, msg + 0); + writel(I2O_CMD_ADAPTER_RESET<<24|HOST_TID<<12|ADAPTER_TID, msg + 1); + writel(core_context, msg + 2); + writel(0, msg + 3); + writel(0, msg + 4); + writel(0, msg + 5); + writel(virt_to_bus(status), msg + 6); + writel(0, msg + 7); /* 64bit host FIXME */ i2o_post_message(c,m); @@ -1781,15 +1781,15 @@ return -ETIMEDOUT; msg=(u32 *)(c->mem_offset+m); - msg[0]=NINE_WORD_MSG_SIZE|SGL_OFFSET_0; - msg[1]=I2O_CMD_STATUS_GET<<24|HOST_TID<<12|ADAPTER_TID; - msg[2]=core_context; - msg[3]=0; - msg[4]=0; - msg[5]=0; - msg[6]=virt_to_bus(c->status_block); - msg[7]=0; /* 64bit host FIXME */ - msg[8]=sizeof(i2o_status_block); /* always 88 bytes */ + writel(NINE_WORD_MSG_SIZE|SGL_OFFSET_0, msg + 0); + writel(I2O_CMD_STATUS_GET<<24|HOST_TID<<12|ADAPTER_TID, msg + 1); + writel(core_context, msg + 2); + writel(0, msg + 3); + writel(0, msg + 4); + writel(0, msg + 5); + writel(virt_to_bus(c->status_block), msg + 6); + writel(0, msg + 7); /* 64bit host FIXME */ + writel(sizeof(i2o_status_block), msg + 8); /* always 88 bytes */ i2o_post_message(c,m); @@ -2193,15 +2193,15 @@ } memset(status, 0, 4); - msg[0]= EIGHT_WORD_MSG_SIZE| TRL_OFFSET_6; - msg[1]= I2O_CMD_OUTBOUND_INIT<<24 | HOST_TID<<12 | ADAPTER_TID; - msg[2]= core_context; - msg[3]= 0x0106; /* Transaction context */ - msg[4]= 4096; /* Host page frame size */ + writel(EIGHT_WORD_MSG_SIZE| TRL_OFFSET_6, msg + 0); + writel(I2O_CMD_OUTBOUND_INIT<<24 | HOST_TID<<12 | ADAPTER_TID, msg + 1); + writel(core_context, msg + 2); + writel(0x0106, msg + 3); /* Transaction context */ + writel(PAGE_SIZE, msg + 4); /* Host page frame size */ /* Frame size is in words. 256 bytes a frame for now */ - msg[5]= MSG_FRAME_SIZE<<16|0x80; /* Outbound msg frame size in words and Initcode */ - msg[6]= 0xD0000004; /* Simple SG LE, EOB */ - msg[7]= virt_to_bus(status); + writel(MSG_FRAME_SIZE<<16|0x80, msg + 5);/* Outbound msg frame size in words and Initcode */ + writel(0xD0000004, msg + 6); /* Simple SG LE, EOB */ + writel(virt_to_bus(status), msg + 7); i2o_post_message(c,m); diff -urN orig/drivers/message/i2o/i2o_pci.c linux/drivers/message/i2o/i2o_pci.c --- orig/drivers/message/i2o/i2o_pci.c Mon Aug 5 13:31:00 2002 +++ linux/drivers/message/i2o/i2o_pci.c Mon Aug 5 15:03:25 2002 @@ -384,4 +384,4 @@ MODULE_PARM_DESC(dpt, "Set this if you want to drive DPT cards normally handled by dpt_i2o"); module_init(i2o_pci_core_attach); module_exit(i2o_pci_core_detach); - \ No newline at end of file + diff -urN orig/drivers/misc/Config.in linux/drivers/misc/Config.in --- orig/drivers/misc/Config.in Wed Dec 29 22:27:08 1999 +++ linux/drivers/misc/Config.in Wed Oct 3 14:57:35 2001 @@ -1,7 +1,17 @@ # -# Misc strange devices +# MCP drivers # mainmenu_option next_comment -comment 'Misc devices' +comment 'Multimedia Capabilities Port drivers' + +bool 'Multimedia drivers' CONFIG_MCP + +# Interface drivers +dep_bool 'Support SA1100 MCP interface' CONFIG_MCP_SA1100 $CONFIG_MCP $CONFIG_ARCH_SA1100 + +# Chip drivers +dep_tristate 'Support for UCB1200 / UCB1300' CONFIG_MCP_UCB1200 $CONFIG_MCP +dep_tristate ' Audio / Telephony interface support' CONFIG_MCP_UCB1200_AUDIO $CONFIG_MCP_UCB1200 $CONFIG_SOUND +dep_tristate ' Touchscreen interface support' CONFIG_MCP_UCB1200_TS $CONFIG_MCP_UCB1200 endmenu diff -urN orig/drivers/misc/Makefile linux/drivers/misc/Makefile --- orig/drivers/misc/Makefile Sun Dec 31 22:12:25 2000 +++ linux/drivers/misc/Makefile Sat Sep 22 23:30:40 2001 @@ -11,6 +11,14 @@ O_TARGET := misc.o +export-objs := mcp-core.o mcp-sa1100.o ucb1x00-core.o + +obj-$(CONFIG_MCP) += mcp-core.o +obj-$(CONFIG_MCP_SA1100) += mcp-sa1100.o +obj-$(CONFIG_MCP_UCB1200) += ucb1x00-core.o +obj-$(CONFIG_MCP_UCB1200_AUDIO) += ucb1x00-audio.o +obj-$(CONFIG_MCP_UCB1200_TS) += ucb1x00-ts.o + include $(TOPDIR)/Rules.make fastdep: diff -urN orig/drivers/misc/mcp-core.c linux/drivers/misc/mcp-core.c --- orig/drivers/misc/mcp-core.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/misc/mcp-core.c Mon Feb 11 09:39:42 2002 @@ -0,0 +1,155 @@ +/* + * linux/drivers/misc/mcp-core.c + * + * Copyright (C) 2001 Russell King + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License. + * + * Generic MCP (Multimedia Communications Port) layer. All MCP locking + * is solely held within this file. + */ +#include +#include +#include +#include + +#include +#include + +#include "mcp.h" + +/** + * mcp_set_telecom_divisor - set the telecom divisor + * @mcp: MCP interface structure + * @div: SIB clock divisor + * + * Set the telecom divisor on the MCP interface. The resulting + * sample rate is SIBCLOCK/div. + */ +void mcp_set_telecom_divisor(struct mcp *mcp, unsigned int div) +{ + spin_lock_irq(&mcp->lock); + mcp->set_telecom_divisor(mcp, div); + spin_unlock_irq(&mcp->lock); +} + +/** + * mcp_set_audio_divisor - set the audio divisor + * @mcp: MCP interface structure + * @div: SIB clock divisor + * + * Set the audio divisor on the MCP interface. + */ +void mcp_set_audio_divisor(struct mcp *mcp, unsigned int div) +{ + spin_lock_irq(&mcp->lock); + mcp->set_audio_divisor(mcp, div); + spin_unlock_irq(&mcp->lock); +} + +/** + * mcp_reg_write - write a device register + * @mcp: MCP interface structure + * @reg: 4-bit register index + * @val: 16-bit data value + * + * Write a device register. The MCP interface must be enabled + * to prevent this function hanging. + */ +void mcp_reg_write(struct mcp *mcp, unsigned int reg, unsigned int val) +{ + unsigned long flags; + + spin_lock_irqsave(&mcp->lock, flags); + mcp->reg_write(mcp, reg, val); + spin_unlock_irqrestore(&mcp->lock, flags); +} + +/** + * mcp_reg_read - read a device register + * @mcp: MCP interface structure + * @reg: 4-bit register index + * + * Read a device register and return its value. The MCP interface + * must be enabled to prevent this function hanging. + */ +unsigned int mcp_reg_read(struct mcp *mcp, unsigned int reg) +{ + unsigned long flags; + unsigned int val; + + spin_lock_irqsave(&mcp->lock, flags); + val = mcp->reg_read(mcp, reg); + spin_unlock_irqrestore(&mcp->lock, flags); + + return val; +} + +/** + * mcp_enable - enable the MCP interface + * @mcp: MCP interface to enable + * + * Enable the MCP interface. Each call to mcp_enable will need + * a corresponding call to mcp_disable to disable the interface. + */ +void mcp_enable(struct mcp *mcp) +{ + spin_lock_irq(&mcp->lock); + if (mcp->use_count++ == 0) + mcp->enable(mcp); + spin_unlock_irq(&mcp->lock); +} + +/** + * mcp_disable - disable the MCP interface + * @mcp: MCP interface to disable + * + * Disable the MCP interface. The MCP interface will only be + * disabled once the number of calls to mcp_enable matches the + * number of calls to mcp_disable. + */ +void mcp_disable(struct mcp *mcp) +{ + unsigned long flags; + + spin_lock_irqsave(&mcp->lock, flags); + if (--mcp->use_count == 0) + mcp->disable(mcp); + spin_unlock_irqrestore(&mcp->lock, flags); +} + + +/* + * This needs re-working + */ +static struct mcp *mcp_if; + +struct mcp *mcp_get(void) +{ + return mcp_if; +} + +int mcp_register(struct mcp *mcp) +{ + if (mcp_if) + return -EBUSY; + if (mcp->owner) + __MOD_INC_USE_COUNT(mcp->owner); + mcp_if = mcp; + return 0; +} + +EXPORT_SYMBOL(mcp_set_telecom_divisor); +EXPORT_SYMBOL(mcp_set_audio_divisor); +EXPORT_SYMBOL(mcp_reg_write); +EXPORT_SYMBOL(mcp_reg_read); +EXPORT_SYMBOL(mcp_enable); +EXPORT_SYMBOL(mcp_disable); +EXPORT_SYMBOL(mcp_get); +EXPORT_SYMBOL(mcp_register); + +MODULE_AUTHOR("Russell King "); +MODULE_DESCRIPTION("Core multimedia communications port driver"); +MODULE_LICENSE("GPL"); diff -urN orig/drivers/misc/mcp-sa1100.c linux/drivers/misc/mcp-sa1100.c --- orig/drivers/misc/mcp-sa1100.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/misc/mcp-sa1100.c Fri Dec 20 17:17:15 2002 @@ -0,0 +1,180 @@ +/* + * linux/drivers/misc/mcp-sa1100.c + * + * Copyright (C) 2001 Russell King + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License. + * + * SA1100 MCP (Multimedia Communications Port) driver. + * + * MCP read/write timeouts from Jordi Colomer, rehacked by rmk. + */ +#include +#include +#include +#include +#include +#include + +#include +#include +#include + +#include "mcp.h" + +static void +mcp_sa1100_set_telecom_divisor(struct mcp *mcp, unsigned int divisor) +{ + unsigned int mccr0; + + divisor /= 32; + + mccr0 = Ser4MCCR0 & ~0x00007f00; + mccr0 |= divisor << 8; + Ser4MCCR0 = mccr0; +} + +static void +mcp_sa1100_set_audio_divisor(struct mcp *mcp, unsigned int divisor) +{ + unsigned int mccr0; + + divisor /= 32; + + mccr0 = Ser4MCCR0 & ~0x0000007f; + mccr0 |= divisor; + Ser4MCCR0 = mccr0; +} + +/* + * Write data to the device. The bit should be set after 3 subframe + * times (each frame is 64 clocks). We wait a maximum of 6 subframes. + * We really should try doing something more productive while we + * wait. + */ +static void +mcp_sa1100_write(struct mcp *mcp, unsigned int reg, unsigned int val) +{ + int ret = -ETIME; + int i; + + Ser4MCDR2 = reg << 17 | MCDR2_Wr | (val & 0xffff); + + for (i = 0; i < 2; i++) { + udelay(mcp->rw_timeout); + if (Ser4MCSR & MCSR_CWC) { + ret = 0; + break; + } + } + + if (ret < 0) + printk(KERN_WARNING "mcp: write timed out\n"); +} + +/* + * Read data from the device. The bit should be set after 3 subframe + * times (each frame is 64 clocks). We wait a maximum of 6 subframes. + * We really should try doing something more productive while we + * wait. + */ +static unsigned int +mcp_sa1100_read(struct mcp *mcp, unsigned int reg) +{ + int ret = -ETIME; + int i; + + Ser4MCDR2 = reg << 17 | MCDR2_Rd; + + for (i = 0; i < 2; i++) { + udelay(mcp->rw_timeout); + if (Ser4MCSR & MCSR_CRC) { + ret = Ser4MCDR2 & 0xffff; + break; + } + } + + if (ret < 0) + printk(KERN_WARNING "mcp: read timed out\n"); + + return ret; +} + +static void mcp_sa1100_enable(struct mcp *mcp) +{ + Ser4MCSR = -1; + Ser4MCCR0 |= MCCR0_MCE; +} + +static void mcp_sa1100_disable(struct mcp *mcp) +{ + Ser4MCCR0 &= ~MCCR0_MCE; +} + +struct mcp mcp_sa1100 = { + owner: THIS_MODULE, + lock: SPIN_LOCK_UNLOCKED, + sclk_rate: 11981000, + dma_audio_rd: DMA_Ser4MCP0Rd, + dma_audio_wr: DMA_Ser4MCP0Wr, + dma_telco_rd: DMA_Ser4MCP1Rd, + dma_telco_wr: DMA_Ser4MCP1Wr, + set_telecom_divisor: mcp_sa1100_set_telecom_divisor, + set_audio_divisor: mcp_sa1100_set_audio_divisor, + reg_write: mcp_sa1100_write, + reg_read: mcp_sa1100_read, + enable: mcp_sa1100_enable, + disable: mcp_sa1100_disable, +}; + +/* + * This needs re-working + */ +static int mcp_sa1100_init(void) +{ + struct mcp *mcp = &mcp_sa1100; + int ret = -ENODEV; + + if (machine_is_accelent_sa() || + machine_is_adsbitsy() || machine_is_assabet() || + machine_is_cerf() || machine_is_flexanet() || + machine_is_freebird() || machine_is_graphicsclient() || + machine_is_graphicsmaster() || machine_is_lart() || + machine_is_omnimeter() || machine_is_pfs168() || + machine_is_shannon() || machine_is_simpad() || + machine_is_simputer() || machine_is_yopy()) { + /* + * Setup the PPC unit correctly. + */ + PPDR &= ~PPC_RXD4; + PPDR |= PPC_TXD4 | PPC_SCLK | PPC_SFRM; + PSDR |= PPC_RXD4; + PSDR &= ~(PPC_TXD4 | PPC_SCLK | PPC_SFRM); + PPSR &= ~(PPC_TXD4 | PPC_SCLK | PPC_SFRM); + + Ser4MCSR = -1; + Ser4MCCR1 = 0; + Ser4MCCR0 = 0x00007f7f | MCCR0_ADM; + + /* + * Calculate the read/write timeout (us) from the bit clock + * rate. This is the period for 3 64-bit frames. Always + * round this time up. + */ + mcp->rw_timeout = (64 * 3 * 1000000 + mcp->sclk_rate - 1) / + mcp->sclk_rate; + + ret = mcp_register(mcp); + } + + return ret; +} + +module_init(mcp_sa1100_init); +EXPORT_SYMBOL(mcp_sa1100_init); + +MODULE_AUTHOR("Russell King "); +MODULE_DESCRIPTION("SA11x0 multimedia communications port driver"); +MODULE_LICENSE("GPL"); diff -urN orig/drivers/misc/mcp.h linux/drivers/misc/mcp.h --- orig/drivers/misc/mcp.h Thu Jan 1 01:00:00 1970 +++ linux/drivers/misc/mcp.h Sun Jan 27 22:51:28 2002 @@ -0,0 +1,44 @@ +/* + * linux/drivers/misc/mcp.h + * + * Copyright (C) 2001 Russell King, All Rights Reserved. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License. + */ +#ifndef MCP_H +#define MCP_H + +struct mcp { + struct module *owner; + spinlock_t lock; + int use_count; + unsigned int sclk_rate; + unsigned int rw_timeout; + dma_device_t dma_audio_rd; + dma_device_t dma_audio_wr; + dma_device_t dma_telco_rd; + dma_device_t dma_telco_wr; + void (*set_telecom_divisor)(struct mcp *, unsigned int); + void (*set_audio_divisor)(struct mcp *, unsigned int); + void (*reg_write)(struct mcp *, unsigned int, unsigned int); + unsigned int (*reg_read)(struct mcp *, unsigned int); + void (*enable)(struct mcp *); + void (*disable)(struct mcp *); +}; + +void mcp_set_telecom_divisor(struct mcp *, unsigned int); +void mcp_set_audio_divisor(struct mcp *, unsigned int); +void mcp_reg_write(struct mcp *, unsigned int, unsigned int); +unsigned int mcp_reg_read(struct mcp *, unsigned int); +void mcp_enable(struct mcp *); +void mcp_disable(struct mcp *); + +/* noddy implementation alert! */ +struct mcp *mcp_get(void); +int mcp_register(struct mcp *); + +#define mcp_get_sclk_rate(mcp) ((mcp)->sclk_rate) + +#endif diff -urN orig/drivers/misc/ucb1x00-audio.c linux/drivers/misc/ucb1x00-audio.c --- orig/drivers/misc/ucb1x00-audio.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/misc/ucb1x00-audio.c Mon Aug 5 22:46:24 2002 @@ -0,0 +1,378 @@ +/* + * linux/drivers/misc/ucb1x00-audio.c + * + * Copyright (C) 2001 Russell King, All Rights Reserved. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + */ +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include + +#include "ucb1x00.h" + +#include "../drivers/sound/sa1100-audio.h" + +#define MAGIC 0x41544154 + +struct ucb1x00_audio { + struct file_operations fops; + struct file_operations mops; + struct ucb1x00 *ucb; + audio_stream_t output_stream; + audio_stream_t input_stream; + audio_state_t state; + unsigned int rate; + int dev_id; + int mix_id; + unsigned int daa_oh_bit; + unsigned int telecom; + unsigned int magic; + unsigned int ctrl_a; + unsigned int ctrl_b; + + /* mixer info */ + unsigned int mod_cnt; + unsigned short output_level; + unsigned short input_level; +}; + +#define REC_MASK (SOUND_MASK_VOLUME | SOUND_MASK_MIC) +#define DEV_MASK REC_MASK + +static int +ucb1x00_mixer_ioctl(struct inode *ino, struct file *filp, uint cmd, ulong arg) +{ + struct ucb1x00_audio *ucba; + unsigned int val, gain; + int ret = 0; + + ucba = list_entry(filp->f_op, struct ucb1x00_audio, mops); + + if (_IOC_TYPE(cmd) != 'M') + return -EINVAL; + + if (cmd == SOUND_MIXER_INFO) { + struct mixer_info mi; + + strncpy(mi.id, "UCB1x00", sizeof(mi.id)); + strncpy(mi.name, "Philips UCB1x00", sizeof(mi.name)); + mi.modify_counter = ucba->mod_cnt; + return copy_to_user((void *)arg, &mi, sizeof(mi)) ? -EFAULT : 0; + } + + if (_IOC_DIR(cmd) & _IOC_WRITE) { + unsigned int left, right; + + ret = get_user(val, (unsigned int *)arg); + if (ret) + goto out; + + left = val & 255; + right = val >> 8; + + if (left > 100) + left = 100; + if (right > 100) + right = 100; + + gain = (left + right) / 2; + + ret = -EINVAL; + if (!ucba->telecom) { + switch(_IOC_NR(cmd)) { + case SOUND_MIXER_VOLUME: + ucba->output_level = gain | gain << 8; + ucba->mod_cnt++; + ucba->ctrl_b = (ucba->ctrl_b & 0xff00) | + ((gain * 31) / 100); + ucb1x00_reg_write(ucba->ucb, UCB_AC_B, + ucba->ctrl_b); + ret = 0; + break; + + case SOUND_MIXER_MIC: + ucba->input_level = gain | gain << 8; + ucba->mod_cnt++; + ucba->ctrl_a = (ucba->ctrl_a & 0x7f) | + (((gain * 31) / 100) << 7); + ucb1x00_reg_write(ucba->ucb, UCB_AC_A, + ucba->ctrl_a); + ret = 0; + break; + } + } + } + + if (ret == 0 && _IOC_DIR(cmd) & _IOC_READ) { + switch (_IOC_NR(cmd)) { + case SOUND_MIXER_VOLUME: + val = ucba->output_level; + break; + + case SOUND_MIXER_MIC: + val = ucba->input_level; + break; + + case SOUND_MIXER_RECSRC: + case SOUND_MIXER_RECMASK: + val = ucba->telecom ? 0 : REC_MASK; + break; + + case SOUND_MIXER_DEVMASK: + val = ucba->telecom ? 0 : DEV_MASK; + break; + + case SOUND_MIXER_CAPS: + case SOUND_MIXER_STEREODEVS: + val = 0; + break; + + default: + val = 0; + ret = -EINVAL; + break; + } + + if (ret == 0) + ret = put_user(val, (int *)arg); + } + out: + return ret; +} + +static int ucb1x00_audio_setrate(struct ucb1x00_audio *ucba, int rate) +{ + unsigned int div_rate = ucb1x00_clkrate(ucba->ucb) / 32; + unsigned int div; + + div = (div_rate + (rate / 2)) / rate; + if (div < 6) + div = 6; + if (div > 127) + div = 127; + + ucba->ctrl_a = (ucba->ctrl_a & ~0x7f) | div; + + if (ucba->telecom) { + ucb1x00_reg_write(ucba->ucb, UCB_TC_B, 0); + ucb1x00_set_telecom_divisor(ucba->ucb, div * 32); + ucb1x00_reg_write(ucba->ucb, UCB_TC_A, ucba->ctrl_a); + ucb1x00_reg_write(ucba->ucb, UCB_TC_B, ucba->ctrl_b); + } else { + ucb1x00_reg_write(ucba->ucb, UCB_AC_B, 0); + ucb1x00_set_audio_divisor(ucba->ucb, div * 32); + ucb1x00_reg_write(ucba->ucb, UCB_AC_A, ucba->ctrl_a); + ucb1x00_reg_write(ucba->ucb, UCB_AC_B, ucba->ctrl_b); + } + + ucba->rate = div_rate / div; + + return ucba->rate; +} + +static int ucb1x00_audio_getrate(struct ucb1x00_audio *ucba) +{ + return ucba->rate; +} + +static void ucb1x00_audio_startup(void *data) +{ + struct ucb1x00_audio *ucba = data; + + ucb1x00_enable(ucba->ucb); + ucb1x00_audio_setrate(ucba, ucba->rate); + + ucb1x00_reg_write(ucba->ucb, UCB_MODE, UCB_MODE_DYN_VFLAG_ENA); + + /* + * Take off-hook + */ + if (ucba->daa_oh_bit) + ucb1x00_io_write(ucba->ucb, 0, ucba->daa_oh_bit); +} + +static void ucb1x00_audio_shutdown(void *data) +{ + struct ucb1x00_audio *ucba = data; + + /* + * Place on-hook + */ + if (ucba->daa_oh_bit) + ucb1x00_io_write(ucba->ucb, ucba->daa_oh_bit, 0); + + ucb1x00_reg_write(ucba->ucb, ucba->telecom ? UCB_TC_B : UCB_AC_B, 0); + ucb1x00_disable(ucba->ucb); +} + +static int +ucb1x00_audio_ioctl(struct inode *inode, struct file *file, uint cmd, ulong arg) +{ + struct ucb1x00_audio *ucba; + int val, ret = 0; + + ucba = list_entry(file->f_op, struct ucb1x00_audio, fops); + + /* + * Make sure we have our magic number + */ + if (ucba->magic != MAGIC) + return -ENODEV; + + switch (cmd) { + case SNDCTL_DSP_STEREO: + ret = get_user(val, (int *)arg); + if (ret) + return ret; + if (val != 0) + return -EINVAL; + val = 0; + break; + + case SNDCTL_DSP_CHANNELS: + case SOUND_PCM_READ_CHANNELS: + val = 1; + break; + + case SNDCTL_DSP_SPEED: + ret = get_user(val, (int *)arg); + if (ret) + return ret; + val = ucb1x00_audio_setrate(ucba, val); + break; + + case SOUND_PCM_READ_RATE: + val = ucb1x00_audio_getrate(ucba); + break; + + case SNDCTL_DSP_SETFMT: + case SNDCTL_DSP_GETFMTS: + val = AFMT_S16_LE; + break; + + default: + return ucb1x00_mixer_ioctl(inode, file, cmd, arg); + } + + return put_user(val, (int *)arg); +} + +static int ucb1x00_audio_open(struct inode *inode, struct file *file) +{ + struct ucb1x00_audio *ucba; + + ucba = list_entry(file->f_op, struct ucb1x00_audio, fops); + + return sa1100_audio_attach(inode, file, &ucba->state); +} + +static struct ucb1x00_audio *ucb1x00_audio_alloc(struct ucb1x00 *ucb) +{ + struct ucb1x00_audio *ucba; + + ucba = kmalloc(sizeof(*ucba), GFP_KERNEL); + if (ucba) { + memset(ucba, 0, sizeof(*ucba)); + + ucba->magic = MAGIC; + ucba->ucb = ucb; + ucba->fops.owner = THIS_MODULE; + ucba->fops.open = ucb1x00_audio_open; + ucba->mops.owner = THIS_MODULE; + ucba->mops.ioctl = ucb1x00_mixer_ioctl; + ucba->state.output_stream = &ucba->output_stream; + ucba->state.input_stream = &ucba->input_stream; + ucba->state.data = ucba; + ucba->state.hw_init = ucb1x00_audio_startup; + ucba->state.hw_shutdown = ucb1x00_audio_shutdown; + ucba->state.client_ioctl = ucb1x00_audio_ioctl; + + /* There is a bug in the StrongARM causes corrupt MCP data to be sent to + * the codec when the FIFOs are empty and writes are made to the OS timer + * match register 0. To avoid this we must make sure that data is always + * sent to the codec. + */ + ucba->state.need_tx_for_rx = 1; + + init_MUTEX(&ucba->state.sem); + ucba->rate = 8000; + } + return ucba; +} + +static struct ucb1x00_audio *audio, *telecom; + +static int __init ucb1x00_audio_init(void) +{ + struct ucb1x00 *ucb = ucb1x00_get(); + struct ucb1x00_audio *a; + + if (!ucb) + return -ENODEV; + + a = ucb1x00_audio_alloc(ucb); + if (a) { + a->state.input_dma = ucb->mcp->dma_audio_rd; + a->state.input_id = "UCB1x00 audio in"; + a->state.output_dma = ucb->mcp->dma_audio_wr; + a->state.output_id = "UCB1x00 audio out"; + a->dev_id = register_sound_dsp(&a->fops, -1); + a->mix_id = register_sound_mixer(&a->mops, -1); + a->ctrl_a = 0; + a->ctrl_b = UCB_AC_B_IN_ENA|UCB_AC_B_OUT_ENA; + audio = a; + } + + a = ucb1x00_audio_alloc(ucb); + if (a) { +#if 0 + a->daa_oh_bit = UCB_IO_8; + + ucb1x00_enable(ucb); + ucb1x00_io_write(ucb, a->daa_oh_bit, 0); + ucb1x00_io_set_dir(ucb, UCB_IO_7 | UCB_IO_6, a->daa_oh_bit); + ucb1x00_disable(ucb); +#endif + + a->telecom = 1; + a->state.input_dma = ucb->mcp->dma_telco_rd; + a->state.input_id = "UCB1x00 telco in"; + a->state.output_dma = ucb->mcp->dma_telco_wr; + a->state.output_id = "UCB1x00 telco out"; + a->dev_id = register_sound_dsp(&a->fops, -1); + a->mix_id = register_sound_mixer(&a->mops, -1); + a->ctrl_a = 0; + a->ctrl_b = UCB_TC_B_IN_ENA|UCB_TC_B_OUT_ENA; + telecom = a; + } + + return 0; +} + +static void __exit ucb1x00_audio_exit(void) +{ + unregister_sound_dsp(telecom->dev_id); + unregister_sound_dsp(audio->dev_id); + unregister_sound_mixer(telecom->mix_id); + unregister_sound_mixer(audio->mix_id); +} + +module_init(ucb1x00_audio_init); +module_exit(ucb1x00_audio_exit); + +MODULE_AUTHOR("Russell King "); +MODULE_DESCRIPTION("UCB1x00 telecom/audio driver"); +MODULE_LICENSE("GPL"); diff -urN orig/drivers/misc/ucb1x00-core.c linux/drivers/misc/ucb1x00-core.c --- orig/drivers/misc/ucb1x00-core.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/misc/ucb1x00-core.c Thu Oct 24 14:40:02 2002 @@ -0,0 +1,647 @@ +/* + * linux/drivers/misc/ucb1x00-core.c + * + * Copyright (C) 2001 Russell King, All Rights Reserved. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License. + * + * The UCB1x00 core driver provides basic services for handling IO, + * the ADC, interrupts, and accessing registers. It is designed + * such that everything goes through this layer, thereby providing + * a consistent locking methodology, as well as allowing the drivers + * to be used on other non-MCP-enabled hardware platforms. + * + * Note that all locks are private to this file. Nothing else may + * touch them. + */ +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include + +#include "ucb1x00.h" + +/** + * ucb1x00_io_set_dir - set IO direction + * @ucb: UCB1x00 structure describing chip + * @in: bitfield of IO pins to be set as inputs + * @out: bitfield of IO pins to be set as outputs + * + * Set the IO direction of the ten general purpose IO pins on + * the UCB1x00 chip. The @in bitfield has priority over the + * @out bitfield, in that if you specify a pin as both input + * and output, it will end up as an input. + * + * ucb1x00_enable must have been called to enable the comms + * before using this function. + * + * This function takes a spinlock, disabling interrupts. + */ +void ucb1x00_io_set_dir(struct ucb1x00 *ucb, unsigned int in, unsigned int out) +{ + unsigned long flags; + + spin_lock_irqsave(&ucb->io_lock, flags); + ucb->io_dir |= out; + ucb->io_dir &= ~in; + + ucb1x00_reg_write(ucb, UCB_IO_DIR, ucb->io_dir); + spin_unlock_irqrestore(&ucb->io_lock, flags); +} + +/** + * ucb1x00_io_write - set or clear IO outputs + * @ucb: UCB1x00 structure describing chip + * @set: bitfield of IO pins to set to logic '1' + * @clear: bitfield of IO pins to set to logic '0' + * + * Set the IO output state of the specified IO pins. The value + * is retained if the pins are subsequently configured as inputs. + * The @clear bitfield has priority over the @set bitfield - + * outputs will be cleared. + * + * ucb1x00_enable must have been called to enable the comms + * before using this function. + * + * This function takes a spinlock, disabling interrupts. + */ +void ucb1x00_io_write(struct ucb1x00 *ucb, unsigned int set, unsigned int clear) +{ + unsigned long flags; + + spin_lock_irqsave(&ucb->io_lock, flags); + ucb->io_out |= set; + ucb->io_out &= ~clear; + + ucb1x00_reg_write(ucb, UCB_IO_DATA, ucb->io_out); + spin_unlock_irqrestore(&ucb->io_lock, flags); +} + +/** + * ucb1x00_io_read - read the current state of the IO pins + * @ucb: UCB1x00 structure describing chip + * + * Return a bitfield describing the logic state of the ten + * general purpose IO pins. + * + * ucb1x00_enable must have been called to enable the comms + * before using this function. + * + * This function does not take any semaphores or spinlocks. + */ +unsigned int ucb1x00_io_read(struct ucb1x00 *ucb) +{ + return ucb1x00_reg_read(ucb, UCB_IO_DATA); +} + +/* + * UCB1300 data sheet says we must: + * 1. enable ADC => 5us (including reference startup time) + * 2. select input => 51*tsibclk => 4.3us + * 3. start conversion => 102*tsibclk => 8.5us + * (tsibclk = 1/11981000) + * Period between SIB 128-bit frames = 10.7us + */ + +/** + * ucb1x00_adc_enable - enable the ADC converter + * @ucb: UCB1x00 structure describing chip + * + * Enable the ucb1x00 and ADC converter on the UCB1x00 for use. + * Any code wishing to use the ADC converter must call this + * function prior to using it. + * + * This function takes the ADC semaphore to prevent two or more + * concurrent uses, and therefore may sleep. As a result, it + * can only be called from process context, not interrupt + * context. + * + * You should release the ADC as soon as possible using + * ucb1x00_adc_disable. + */ +void ucb1x00_adc_enable(struct ucb1x00 *ucb) +{ + down(&ucb->adc_sem); + + ucb->adc_cr |= UCB_ADC_ENA; + + ucb1x00_enable(ucb); + ucb1x00_reg_write(ucb, UCB_ADC_CR, ucb->adc_cr); +} + +/** + * ucb1x00_adc_read - read the specified ADC channel + * @ucb: UCB1x00 structure describing chip + * @adc_channel: ADC channel mask + * @sync: wait for syncronisation pulse. + * + * Start an ADC conversion and wait for the result. Note that + * synchronised ADC conversions (via the ADCSYNC pin) must wait + * until the trigger is asserted and the conversion is finished. + * + * This function currently spins waiting for the conversion to + * complete (2 frames max without sync). + * + * If called for a synchronised ADC conversion, it may sleep + * with the ADC semaphore held. + */ +unsigned int ucb1x00_adc_read(struct ucb1x00 *ucb, int adc_channel, int sync) +{ + unsigned int val; + + if (sync) + adc_channel |= UCB_ADC_SYNC_ENA; + + ucb1x00_reg_write(ucb, UCB_ADC_CR, ucb->adc_cr | adc_channel); + ucb1x00_reg_write(ucb, UCB_ADC_CR, ucb->adc_cr | adc_channel | UCB_ADC_START); + + for (;;) { + val = ucb1x00_reg_read(ucb, UCB_ADC_DATA); + if (val & UCB_ADC_DAT_VAL) + break; + /* yield to other processes */ + set_current_state(TASK_INTERRUPTIBLE); + schedule_timeout(1); + } + + return UCB_ADC_DAT(val); +} + +/** + * ucb1x00_adc_disable - disable the ADC converter + * @ucb: UCB1x00 structure describing chip + * + * Disable the ADC converter and release the ADC semaphore. + */ +void ucb1x00_adc_disable(struct ucb1x00 *ucb) +{ + ucb->adc_cr &= ~UCB_ADC_ENA; + ucb1x00_reg_write(ucb, UCB_ADC_CR, ucb->adc_cr); + ucb1x00_disable(ucb); + + up(&ucb->adc_sem); +} + +#ifdef CONFIG_PM +static int ucb1x00_pm (struct pm_dev *dev, pm_request_t rqst, void *data) +{ + struct ucb1x00 *ucb = (struct ucb1x00 *)dev->data; + unsigned int isr; + + if (rqst == PM_RESUME) { + ucb1x00_enable(ucb); + isr = ucb1x00_reg_read(ucb, UCB_IE_STATUS); + ucb1x00_reg_write(ucb, UCB_IE_CLEAR, isr); + ucb1x00_reg_write(ucb, UCB_IE_CLEAR, 0); + ucb1x00_disable(ucb); + } + + return 0; +} +#endif + +/* + * UCB1x00 Interrupt handling. + * + * The UCB1x00 can generate interrupts when the SIBCLK is stopped. + * Since we need to read an internal register, we must re-enable + * SIBCLK to talk to the chip. We leave the clock running until + * we have finished processing all interrupts from the chip. + */ +static void ucb1x00_irq(int irqnr, void *devid, struct pt_regs *regs) +{ + struct ucb1x00 *ucb = devid; + struct ucb1x00_irq *irq; + unsigned int isr, i; + + ucb1x00_enable(ucb); + isr = ucb1x00_reg_read(ucb, UCB_IE_STATUS); + ucb1x00_reg_write(ucb, UCB_IE_CLEAR, isr); + ucb1x00_reg_write(ucb, UCB_IE_CLEAR, 0); + + for (i = 0, irq = ucb->irq_handler; i < 16 && isr; i++, isr >>= 1, irq++) + if (isr & 1 && irq->fn) + irq->fn(i, irq->devid); + ucb1x00_disable(ucb); +} + +/** + * ucb1x00_hook_irq - hook a UCB1x00 interrupt + * @ucb: UCB1x00 structure describing chip + * @idx: interrupt index + * @fn: function to call when interrupt is triggered + * @devid: device id to pass to interrupt handler + * + * Hook the specified interrupt. You can only register one handler + * for each interrupt source. The interrupt source is not enabled + * by this function; use ucb1x00_enable_irq instead. + * + * Interrupt handlers will be called with other interrupts enabled. + * + * Returns zero on success, or one of the following errors: + * -EINVAL if the interrupt index is invalid + * -EBUSY if the interrupt has already been hooked + */ +int ucb1x00_hook_irq(struct ucb1x00 *ucb, unsigned int idx, void (*fn)(int, void *), void *devid) +{ + struct ucb1x00_irq *irq; + int ret = -EINVAL; + + if (idx < 16) { + irq = ucb->irq_handler + idx; + ret = -EBUSY; + + spin_lock_irq(&ucb->lock); + if (irq->fn == NULL) { + irq->devid = devid; + irq->fn = fn; + ret = 0; + } + spin_unlock_irq(&ucb->lock); + } + return ret; +} + +/** + * ucb1x00_enable_irq - enable an UCB1x00 interrupt source + * @ucb: UCB1x00 structure describing chip + * @idx: interrupt index + * @edges: interrupt edges to enable + * + * Enable the specified interrupt to trigger on %UCB_RISING, + * %UCB_FALLING or both edges. The interrupt should have been + * hooked by ucb1x00_hook_irq. + */ +void ucb1x00_enable_irq(struct ucb1x00 *ucb, unsigned int idx, int edges) +{ + unsigned long flags; + + if (idx < 16) { + spin_lock_irqsave(&ucb->lock, flags); + + ucb1x00_enable(ucb); + if (edges & UCB_RISING) { + ucb->irq_ris_enbl |= 1 << idx; + ucb1x00_reg_write(ucb, UCB_IE_RIS, ucb->irq_ris_enbl); + } + if (edges & UCB_FALLING) { + ucb->irq_fal_enbl |= 1 << idx; + ucb1x00_reg_write(ucb, UCB_IE_FAL, ucb->irq_fal_enbl); + } + ucb1x00_disable(ucb); + spin_unlock_irqrestore(&ucb->lock, flags); + } +} + +/** + * ucb1x00_disable_irq - disable an UCB1x00 interrupt source + * @ucb: UCB1x00 structure describing chip + * @edges: interrupt edges to disable + * + * Disable the specified interrupt triggering on the specified + * (%UCB_RISING, %UCB_FALLING or both) edges. + */ +void ucb1x00_disable_irq(struct ucb1x00 *ucb, unsigned int idx, int edges) +{ + unsigned long flags; + + if (idx < 16) { + spin_lock_irqsave(&ucb->lock, flags); + + ucb1x00_enable(ucb); + if (edges & UCB_RISING) { + ucb->irq_ris_enbl &= ~(1 << idx); + ucb1x00_reg_write(ucb, UCB_IE_RIS, ucb->irq_ris_enbl); + } + if (edges & UCB_FALLING) { + ucb->irq_fal_enbl &= ~(1 << idx); + ucb1x00_reg_write(ucb, UCB_IE_FAL, ucb->irq_fal_enbl); + } + ucb1x00_disable(ucb); + spin_unlock_irqrestore(&ucb->lock, flags); + } +} + +/** + * ucb1x00_free_irq - disable and free the specified UCB1x00 interrupt + * @ucb: UCB1x00 structure describing chip + * @idx: interrupt index + * @devid: device id. + * + * Disable the interrupt source and remove the handler. devid must + * match the devid passed when hooking the interrupt. + * + * Returns zero on success, or one of the following errors: + * -EINVAL if the interrupt index is invalid + * -ENOENT if devid does not match + */ +int ucb1x00_free_irq(struct ucb1x00 *ucb, unsigned int idx, void *devid) +{ + struct ucb1x00_irq *irq; + int ret; + + if (idx >= 16) + goto bad; + + irq = ucb->irq_handler + idx; + ret = -ENOENT; + + spin_lock_irq(&ucb->lock); + if (irq->devid == devid) { + ucb->irq_ris_enbl &= ~(1 << idx); + ucb->irq_fal_enbl &= ~(1 << idx); + + ucb1x00_enable(ucb); + ucb1x00_reg_write(ucb, UCB_IE_RIS, ucb->irq_ris_enbl); + ucb1x00_reg_write(ucb, UCB_IE_FAL, ucb->irq_fal_enbl); + ucb1x00_disable(ucb); + + irq->fn = NULL; + irq->devid = NULL; + ret = 0; + } + spin_unlock_irq(&ucb->lock); + return ret; + +bad: + printk(KERN_ERR __FUNCTION__ ": freeing bad irq %d\n", idx); + return -EINVAL; +} + +/* + * Try to probe our interrupt, rather than relying on lots of + * hard-coded machine dependencies. For reference, the expected + * IRQ mappings are: + * + * Machine Default IRQ + * adsbitsy IRQ_GPCIN4 + * cerf IRQ_GPIO_UCB1200_IRQ + * flexanet IRQ_GPIO_GUI + * freebird IRQ_GPIO_FREEBIRD_UCB1300_IRQ + * graphicsclient ADS_EXT_IRQ(8) + * graphicsmaster ADS_EXT_IRQ(8) + * lart LART_IRQ_UCB1200 + * omnimeter IRQ_GPIO23 + * pfs168 IRQ_GPIO_UCB1300_IRQ + * simpad IRQ_GPIO_UCB1300_IRQ + * shannon SHANNON_IRQ_GPIO_IRQ_CODEC + * yopy IRQ_GPIO_UCB1200_IRQ + */ +static int __init ucb1x00_detect_irq(struct ucb1x00 *ucb) +{ + unsigned long mask; + + mask = probe_irq_on(); + if (!mask) + return NO_IRQ; + + /* + * Enable the ADC interrupt. + */ + ucb1x00_reg_write(ucb, UCB_IE_RIS, UCB_IE_ADC); + ucb1x00_reg_write(ucb, UCB_IE_FAL, UCB_IE_ADC); + ucb1x00_reg_write(ucb, UCB_IE_CLEAR, 0xffff); + ucb1x00_reg_write(ucb, UCB_IE_CLEAR, 0); + + /* + * Cause an ADC interrupt. + */ + ucb1x00_reg_write(ucb, UCB_ADC_CR, UCB_ADC_ENA); + ucb1x00_reg_write(ucb, UCB_ADC_CR, UCB_ADC_ENA | UCB_ADC_START); + + /* + * Wait for the conversion to complete. + */ + while ((ucb1x00_reg_read(ucb, UCB_ADC_DATA) & UCB_ADC_DAT_VAL) == 0); + ucb1x00_reg_write(ucb, UCB_ADC_CR, 0); + + /* + * Disable and clear interrupt. + */ + ucb1x00_reg_write(ucb, UCB_IE_RIS, 0); + ucb1x00_reg_write(ucb, UCB_IE_FAL, 0); + ucb1x00_reg_write(ucb, UCB_IE_CLEAR, 0xffff); + ucb1x00_reg_write(ucb, UCB_IE_CLEAR, 0); + + /* + * Read triggered interrupt. + */ + return probe_irq_off(mask); +} + +/* + * This configures the UCB1x00 layer depending on the machine type + * we're running on. The UCB1x00 drivers should not contain any + * machine dependencies. + * + * We can get rid of some of these dependencies by using existing + * facilities provided by the kernel - namely IRQ probing. The + * machine specific files are expected to setup the IRQ levels on + * initialisation. With any luck, we'll get rid of all the + * machine dependencies here. + */ +static int __init ucb1x00_configure(struct ucb1x00 *ucb) +{ + unsigned int irq_gpio_pin = 0; + int irq, default_irq = NO_IRQ; + + if (machine_is_adsbitsy()) + default_irq = IRQ_GPCIN4; + +// if (machine_is_assabet()) +// default_irq = IRQ_GPIO23; + +#ifdef CONFIG_SA1100_CERF + if (machine_is_cerf()) + default_irq = IRQ_GPIO_UCB1200_IRQ; +#endif +#ifdef CONFIG_SA1100_FREEBIRD + if (machine_is_freebird()) + default_irq = IRQ_GPIO_FREEBIRD_UCB1300_IRQ; +#endif +#if defined(CONFIG_SA1100_GRAPHICSCLIENT) || defined(CONFIG_SA1100_GRAPICSMASTER) + if (machine_is_graphicsclient() || machine_is_graphicsmaster()) + default_irq = ADS_EXT_IRQ(8); +#endif +#ifdef CONFIG_SA1100_LART + if (machine_is_lart()) { + default_irq = LART_IRQ_UCB1200; + irq_gpio_pin = LART_GPIO_UCB1200; + } +#endif + if (machine_is_omnimeter()) + default_irq = IRQ_GPIO23; + +#ifdef CONFIG_SA1100_PFS168 + if (machine_is_pfs168()) + default_irq = IRQ_GPIO_UCB1300_IRQ; +#endif +#ifdef CONFIG_SA1100_SIMPAD + if (machine_is_simpad()) + default_irq = IRQ_GPIO_UCB1300_IRQ; +#endif +#ifdef CONFIG_SA1100_SIMPUTER + if (machine_is_simputer()) { + default_irq = IRQ_GPIO_UCB1300_IRQ; + irq_gpio_pin = GPIO_UCB1300_IRQ; + } +#endif + if (machine_is_shannon()) + default_irq = SHANNON_IRQ_GPIO_IRQ_CODEC; +#ifdef CONFIG_SA1100_YOPY + if (machine_is_yopy()) + default_irq = IRQ_GPIO_UCB1200_IRQ; +#endif +#ifdef CONFIG_SA1100_ACCELENT + if (machine_is_accelent_sa()) { + ucb->irq = IRQ_GPIO_UCB1200_IRQ; + irq_gpio_pin = GPIO_UCB1200_IRQ; + } +#endif + + /* + * Eventually, this will disappear. + */ + if (irq_gpio_pin) + set_GPIO_IRQ_edge(irq_gpio_pin, GPIO_RISING_EDGE); + + irq = ucb1x00_detect_irq(ucb); + if (irq != NO_IRQ) { + if (default_irq != NO_IRQ && irq != default_irq) + printk(KERN_ERR "UCB1x00: probed IRQ%d != default IRQ%d\n", + irq, default_irq); + if (irq == default_irq) + printk(KERN_ERR "UCB1x00: probed IRQ%d correctly. " + "Please remove machine dependencies from " + "ucb1x00-core.c\n", irq); + ucb->irq = irq; + } else { + printk(KERN_ERR "UCB1x00: IRQ probe failed, using IRQ%d\n", + default_irq); + ucb->irq = default_irq; + } + + return ucb->irq == NO_IRQ ? -ENODEV : 0; +} + +struct ucb1x00 *my_ucb; + +/** + * ucb1x00_get - get the UCB1x00 structure describing a chip + * @ucb: UCB1x00 structure describing chip + * + * Return the UCB1x00 structure describing a chip. + * + * FIXME: Currently very noddy indeed, which currently doesn't + * matter since we only support one chip. + */ +struct ucb1x00 *ucb1x00_get(void) +{ + return my_ucb; +} + +static int __init ucb1x00_init(void) +{ + struct mcp *mcp; + unsigned int id; + int ret = -ENODEV; + + mcp = mcp_get(); + if (!mcp) + goto no_mcp; + + mcp_enable(mcp); + id = mcp_reg_read(mcp, UCB_ID); + + if (id != UCB_ID_1200 && id != UCB_ID_1300) { + printk(KERN_WARNING "UCB1x00 ID not found: %04x\n", id); + goto out; + } + + my_ucb = kmalloc(sizeof(struct ucb1x00), GFP_KERNEL); + ret = -ENOMEM; + if (!my_ucb) + goto out; + + if (machine_is_shannon()) { + /* reset the codec */ + GPDR |= SHANNON_GPIO_CODEC_RESET; + GPCR = SHANNON_GPIO_CODEC_RESET; + GPSR = SHANNON_GPIO_CODEC_RESET; + + } + + memset(my_ucb, 0, sizeof(struct ucb1x00)); + + spin_lock_init(&my_ucb->lock); + spin_lock_init(&my_ucb->io_lock); + sema_init(&my_ucb->adc_sem, 1); + + my_ucb->id = id; + my_ucb->mcp = mcp; + + ret = ucb1x00_configure(my_ucb); + if (ret) + goto out; + + ret = request_irq(my_ucb->irq, ucb1x00_irq, 0, "UCB1x00", my_ucb); + if (ret) { + printk(KERN_ERR "ucb1x00: unable to grab irq%d: %d\n", + my_ucb->irq, ret); + kfree(my_ucb); + my_ucb = NULL; + goto out; + } + +#ifdef CONFIG_PM + my_ucb->pmdev = pm_register(PM_SYS_DEV, PM_SYS_UNKNOWN, ucb1x00_pm); + if (my_ucb->pmdev == NULL) + printk("ucb1x00: unable to register in PM.\n"); + else + my_ucb->pmdev->data = my_ucb; +#endif + +out: + mcp_disable(mcp); +no_mcp: + return ret; +} + +static void __exit ucb1x00_exit(void) +{ + free_irq(my_ucb->irq, my_ucb); + kfree(my_ucb); +} + +module_init(ucb1x00_init); +module_exit(ucb1x00_exit); + +EXPORT_SYMBOL(ucb1x00_get); + +EXPORT_SYMBOL(ucb1x00_io_set_dir); +EXPORT_SYMBOL(ucb1x00_io_write); +EXPORT_SYMBOL(ucb1x00_io_read); + +EXPORT_SYMBOL(ucb1x00_adc_enable); +EXPORT_SYMBOL(ucb1x00_adc_read); +EXPORT_SYMBOL(ucb1x00_adc_disable); + +EXPORT_SYMBOL(ucb1x00_hook_irq); +EXPORT_SYMBOL(ucb1x00_free_irq); +EXPORT_SYMBOL(ucb1x00_enable_irq); +EXPORT_SYMBOL(ucb1x00_disable_irq); + +MODULE_AUTHOR("Russell King "); +MODULE_DESCRIPTION("UCB1x00 core driver"); +MODULE_LICENSE("GPL"); diff -urN orig/drivers/misc/ucb1x00-ts.c linux/drivers/misc/ucb1x00-ts.c --- orig/drivers/misc/ucb1x00-ts.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/misc/ucb1x00-ts.c Fri Jun 21 15:49:47 2002 @@ -0,0 +1,664 @@ +/* + * linux/drivers/misc/ucb1x00-ts.c + * + * Copyright (C) 2001 Russell King, All Rights Reserved. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + * + * 21-Jan-2002 : + * + * Added support for synchronous A/D mode. This mode is useful to + * avoid noise induced in the touchpanel by the LCD, provided that + * the UCB1x00 has a valid LCD sync signal routed to its ADCSYNC pin. + * It is important to note that the signal connected to the ADCSYNC + * pin should provide pulses even when the LCD is blanked, otherwise + * a pen touch needed to unblank the LCD will never be read. + */ +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include + +#include "ucb1x00.h" + +/* + * Define this if you want the UCB1x00 stuff to talk to the input layer + */ +#undef USE_INPUT + +#ifndef USE_INPUT + +#include +#include +#include + +/* + * This structure is nonsense - millisecs is not very useful + * since the field size is too small. Also, we SHOULD NOT + * be exposing jiffies to user space directly. + */ +struct ts_event { + u16 pressure; + u16 x; + u16 y; + u16 pad; + struct timeval stamp; +}; + +#define NR_EVENTS 16 + +#else + +#include + +#endif + +struct ucb1x00_ts { +#ifdef USE_INPUT + struct input_dev idev; +#endif + struct ucb1x00 *ucb; +#ifdef CONFIG_PM + struct pm_dev *pmdev; +#endif + + wait_queue_head_t irq_wait; + struct semaphore sem; + struct completion init_exit; + struct task_struct *rtask; + int use_count; + u16 x_res; + u16 y_res; + +#ifndef USE_INPUT + struct fasync_struct *fasync; + wait_queue_head_t read_wait; + u8 evt_head; + u8 evt_tail; + struct ts_event events[NR_EVENTS]; +#endif + int restart:1; + int adcsync:1; +}; + +static struct ucb1x00_ts ucbts; +static int adcsync = UCB_NOSYNC; + +static int ucb1x00_ts_startup(struct ucb1x00_ts *ts); +static void ucb1x00_ts_shutdown(struct ucb1x00_ts *ts); + +#ifndef USE_INPUT + +#define ucb1x00_ts_evt_pending(ts) ((volatile u8)(ts)->evt_head != (ts)->evt_tail) +#define ucb1x00_ts_evt_get(ts) ((ts)->events + (ts)->evt_tail) +#define ucb1x00_ts_evt_pull(ts) ((ts)->evt_tail = ((ts)->evt_tail + 1) & (NR_EVENTS - 1)) +#define ucb1x00_ts_evt_clear(ts) ((ts)->evt_head = (ts)->evt_tail = 0) + +static inline void ucb1x00_ts_evt_add(struct ucb1x00_ts *ts, u16 pressure, u16 x, u16 y) +{ + int next_head; + + next_head = (ts->evt_head + 1) & (NR_EVENTS - 1); + if (next_head != ts->evt_tail) { + ts->events[ts->evt_head].pressure = pressure; + ts->events[ts->evt_head].x = x; + ts->events[ts->evt_head].y = y; + do_gettimeofday(&ts->events[ts->evt_head].stamp); + ts->evt_head = next_head; + + if (ts->fasync) + kill_fasync(&ts->fasync, SIGIO, POLL_IN); + wake_up_interruptible(&ts->read_wait); + } +} + +static inline void ucb1x00_ts_event_release(struct ucb1x00_ts *ts) +{ + ucb1x00_ts_evt_add(ts, 0, 0, 0); +} + +/* + * User space driver interface. + */ +static ssize_t +ucb1x00_ts_read(struct file *filp, char *buffer, size_t count, loff_t *ppos) +{ + DECLARE_WAITQUEUE(wait, current); + struct ucb1x00_ts *ts = filp->private_data; + char *ptr = buffer; + int err = 0; + + add_wait_queue(&ts->read_wait, &wait); + while (count >= sizeof(struct ts_event)) { + err = -ERESTARTSYS; + if (signal_pending(current)) + break; + + if (ucb1x00_ts_evt_pending(ts)) { + struct ts_event *evt = ucb1x00_ts_evt_get(ts); + + err = copy_to_user(ptr, evt, sizeof(struct ts_event)); + ucb1x00_ts_evt_pull(ts); + + if (err) + break; + + ptr += sizeof(struct ts_event); + count -= sizeof(struct ts_event); + continue; + } + + set_current_state(TASK_INTERRUPTIBLE); + err = -EAGAIN; + if (filp->f_flags & O_NONBLOCK) + break; + schedule(); + } + current->state = TASK_RUNNING; + remove_wait_queue(&ts->read_wait, &wait); + + return ptr == buffer ? err : ptr - buffer; +} + +static unsigned int ucb1x00_ts_poll(struct file *filp, poll_table *wait) +{ + struct ucb1x00_ts *ts = filp->private_data; + int ret = 0; + + poll_wait(filp, &ts->read_wait, wait); + if (ucb1x00_ts_evt_pending(ts)) + ret = POLLIN | POLLRDNORM; + + return ret; +} + +static int ucb1x00_ts_fasync(int fd, struct file *filp, int on) +{ + struct ucb1x00_ts *ts = filp->private_data; + + return fasync_helper(fd, filp, on, &ts->fasync); +} + +static int ucb1x00_ts_open(struct inode *inode, struct file *filp) +{ + struct ucb1x00_ts *ts = &ucbts; + int ret = 0; + + ret = ucb1x00_ts_startup(ts); + if (ret == 0) + filp->private_data = ts; + + return ret; +} + +/* + * Release touchscreen resources. Disable IRQs. + */ +static int ucb1x00_ts_release(struct inode *inode, struct file *filp) +{ + struct ucb1x00_ts *ts = filp->private_data; + + down(&ts->sem); + ucb1x00_ts_fasync(-1, filp, 0); + ucb1x00_ts_shutdown(ts); + up(&ts->sem); + + return 0; +} + +static struct file_operations ucb1x00_fops = { + owner: THIS_MODULE, + read: ucb1x00_ts_read, + poll: ucb1x00_ts_poll, + open: ucb1x00_ts_open, + release: ucb1x00_ts_release, + fasync: ucb1x00_ts_fasync, +}; + +/* + * The official UCB1x00 touchscreen is a miscdevice: + * 10 char Non-serial mice, misc features + * 14 = /dev/touchscreen/ucb1x00 UCB 1x00 touchscreen + */ +static struct miscdevice ucb1x00_ts_dev = { + minor: 14, + name: "touchscreen/ucb1x00", + fops: &ucb1x00_fops, +}; + +static inline int ucb1x00_ts_register(struct ucb1x00_ts *ts) +{ + init_waitqueue_head(&ts->read_wait); + return misc_register(&ucb1x00_ts_dev); +} + +static inline void ucb1x00_ts_deregister(struct ucb1x00_ts *ts) +{ + misc_deregister(&ucb1x00_ts_dev); +} + +#else + +#define ucb1x00_ts_evt_clear(ts) do { } while (0) + +static inline void ucb1x00_ts_evt_add(struct ucb1x00_ts *ts, u16 pressure, u16 x, u16 y) +{ + input_report_abs(&ts->idev, ABS_X, x); + input_report_abs(&ts->idev, ABS_Y, y); + input_report_abs(&ts->idev, ABS_PRESSURE, pressure); +} + +static int ucb1x00_ts_open(struct input_dev *idev) +{ + struct ucb1x00_ts *ts = (struct ucb1x00_ts *)idev; + + return ucb1x00_ts_startup(ts); +} + +static void ucb1x00_ts_close(struct input_dev *idev) +{ + struct ucb1x00_ts *ts = (struct ucb1x00_ts *)idev; + + down(&ts->sem); + ucb1x00_ts_shutdown(ts); + up(&ts->sem); +} + +static inline int ucb1x00_ts_register(struct ucb1x00_ts *ts) +{ + ts->idev.name = "Touchscreen panel"; + ts->idev.idproduct = ts->ucb->id; + ts->idev.open = ucb1x00_ts_open; + ts->idev.close = ucb1x00_ts_close; + + __set_bit(EV_ABS, ts->idev.evbit); + __set_bit(ABS_X, ts->idev.absbit); + __set_bit(ABS_Y, ts->idev.absbit); + __set_bit(ABS_PRESSURE, ts->idev.absbit); + + input_register_device(&ts->idev); + + return 0; +} + +static inline void ucb1x00_ts_deregister(struct ucb1x00_ts *ts) +{ + input_unregister_device(&ts->idev); +} + +#endif + +/* + * Switch to interrupt mode. + */ +static inline void ucb1x00_ts_mode_int(struct ucb1x00_ts *ts) +{ + ucb1x00_reg_write(ts->ucb, UCB_TS_CR, + UCB_TS_CR_TSMX_POW | UCB_TS_CR_TSPX_POW | + UCB_TS_CR_TSMY_GND | UCB_TS_CR_TSPY_GND | + UCB_TS_CR_MODE_INT); +} + +/* + * Switch to pressure mode, and read pressure. We don't need to wait + * here, since both plates are being driven. + */ +static inline unsigned int ucb1x00_ts_read_pressure(struct ucb1x00_ts *ts) +{ + ucb1x00_reg_write(ts->ucb, UCB_TS_CR, + UCB_TS_CR_TSMX_POW | UCB_TS_CR_TSPX_POW | + UCB_TS_CR_TSMY_GND | UCB_TS_CR_TSPY_GND | + UCB_TS_CR_MODE_PRES | UCB_TS_CR_BIAS_ENA); + + return ucb1x00_adc_read(ts->ucb, UCB_ADC_INP_TSPY, ts->adcsync); +} + +/* + * Switch to X position mode and measure Y plate. We switch the plate + * configuration in pressure mode, then switch to position mode. This + * gives a faster response time. Even so, we need to wait about 55us + * for things to stabilise. + */ +static inline unsigned int ucb1x00_ts_read_xpos(struct ucb1x00_ts *ts) +{ + ucb1x00_reg_write(ts->ucb, UCB_TS_CR, + UCB_TS_CR_TSMX_GND | UCB_TS_CR_TSPX_POW | + UCB_TS_CR_MODE_PRES | UCB_TS_CR_BIAS_ENA); + ucb1x00_reg_write(ts->ucb, UCB_TS_CR, + UCB_TS_CR_TSMX_GND | UCB_TS_CR_TSPX_POW | + UCB_TS_CR_MODE_PRES | UCB_TS_CR_BIAS_ENA); + ucb1x00_reg_write(ts->ucb, UCB_TS_CR, + UCB_TS_CR_TSMX_GND | UCB_TS_CR_TSPX_POW | + UCB_TS_CR_MODE_POS | UCB_TS_CR_BIAS_ENA); + + udelay(55); + + return ucb1x00_adc_read(ts->ucb, UCB_ADC_INP_TSPY, ts->adcsync); +} + +/* + * Switch to Y position mode and measure X plate. We switch the plate + * configuration in pressure mode, then switch to position mode. This + * gives a faster response time. Even so, we need to wait about 55us + * for things to stabilise. + */ +static inline unsigned int ucb1x00_ts_read_ypos(struct ucb1x00_ts *ts) +{ + ucb1x00_reg_write(ts->ucb, UCB_TS_CR, + UCB_TS_CR_TSMY_GND | UCB_TS_CR_TSPY_POW | + UCB_TS_CR_MODE_PRES | UCB_TS_CR_BIAS_ENA); + ucb1x00_reg_write(ts->ucb, UCB_TS_CR, + UCB_TS_CR_TSMY_GND | UCB_TS_CR_TSPY_POW | + UCB_TS_CR_MODE_PRES | UCB_TS_CR_BIAS_ENA); + ucb1x00_reg_write(ts->ucb, UCB_TS_CR, + UCB_TS_CR_TSMY_GND | UCB_TS_CR_TSPY_POW | + UCB_TS_CR_MODE_POS | UCB_TS_CR_BIAS_ENA); + + udelay(55); + + return ucb1x00_adc_read(ts->ucb, UCB_ADC_INP_TSPX, ts->adcsync); +} + +/* + * Switch to X plate resistance mode. Set MX to ground, PX to + * supply. Measure current. + */ +static inline unsigned int ucb1x00_ts_read_xres(struct ucb1x00_ts *ts) +{ + ucb1x00_reg_write(ts->ucb, UCB_TS_CR, + UCB_TS_CR_TSMX_GND | UCB_TS_CR_TSPX_POW | + UCB_TS_CR_MODE_PRES | UCB_TS_CR_BIAS_ENA); + return ucb1x00_adc_read(ts->ucb, 0, ts->adcsync); +} + +/* + * Switch to Y plate resistance mode. Set MY to ground, PY to + * supply. Measure current. + */ +static inline unsigned int ucb1x00_ts_read_yres(struct ucb1x00_ts *ts) +{ + ucb1x00_reg_write(ts->ucb, UCB_TS_CR, + UCB_TS_CR_TSMY_GND | UCB_TS_CR_TSPY_POW | + UCB_TS_CR_MODE_PRES | UCB_TS_CR_BIAS_ENA); + return ucb1x00_adc_read(ts->ucb, 0, ts->adcsync); +} + +/* + * This is a RT kernel thread that handles the ADC accesses + * (mainly so we can use semaphores in the UCB1200 core code + * to serialise accesses to the ADC). + */ +static int ucb1x00_thread(void *_ts) +{ + struct ucb1x00_ts *ts = _ts; + struct task_struct *tsk = current; + DECLARE_WAITQUEUE(wait, tsk); + int valid; + + ts->rtask = tsk; + + daemonize(); + reparent_to_init(); + strcpy(tsk->comm, "ktsd"); + tsk->tty = NULL; + /* + * We could run as a real-time thread. However, thus far + * this doesn't seem to be necessary. + */ +// tsk->policy = SCHED_FIFO; +// tsk->rt_priority = 1; + + /* only want to receive SIGKILL */ + spin_lock_irq(&tsk->sigmask_lock); + siginitsetinv(&tsk->blocked, sigmask(SIGKILL)); + recalc_sigpending(tsk); + spin_unlock_irq(&tsk->sigmask_lock); + + complete(&ts->init_exit); + + valid = 0; + + add_wait_queue(&ts->irq_wait, &wait); + for (;;) { + unsigned int x, y, p, val; + signed long timeout; + + ts->restart = 0; + + ucb1x00_adc_enable(ts->ucb); + + x = ucb1x00_ts_read_xpos(ts); + y = ucb1x00_ts_read_ypos(ts); + p = ucb1x00_ts_read_pressure(ts); + + /* + * Switch back to interrupt mode. + */ + ucb1x00_ts_mode_int(ts); + ucb1x00_adc_disable(ts->ucb); + + set_task_state(tsk, TASK_UNINTERRUPTIBLE); + schedule_timeout(HZ / 100); + if (signal_pending(tsk)) + break; + + ucb1x00_enable(ts->ucb); + val = ucb1x00_reg_read(ts->ucb, UCB_TS_CR); + + if (val & (UCB_TS_CR_TSPX_LOW | UCB_TS_CR_TSMX_LOW)) { + set_task_state(tsk, TASK_INTERRUPTIBLE); + + ucb1x00_enable_irq(ts->ucb, UCB_IRQ_TSPX, UCB_FALLING); + ucb1x00_disable(ts->ucb); + + /* + * If we spat out a valid sample set last time, + * spit out a "pen off" sample here. + */ + if (valid) { + ucb1x00_ts_event_release(ts); + valid = 0; + } + + timeout = MAX_SCHEDULE_TIMEOUT; + } else { + ucb1x00_disable(ts->ucb); + + /* + * Filtering is policy. Policy belongs in user + * space. We therefore leave it to user space + * to do any filtering they please. + */ + if (!ts->restart) { + ucb1x00_ts_evt_add(ts, p, x, y); + valid = 1; + } + + set_task_state(tsk, TASK_INTERRUPTIBLE); + timeout = HZ / 100; + } + + schedule_timeout(timeout); + if (signal_pending(tsk)) + break; + } + + remove_wait_queue(&ts->irq_wait, &wait); + + ts->rtask = NULL; + ucb1x00_ts_evt_clear(ts); + complete_and_exit(&ts->init_exit, 0); +} + +/* + * We only detect touch screen _touches_ with this interrupt + * handler, and even then we just schedule our task. + */ +static void ucb1x00_ts_irq(int idx, void *id) +{ + struct ucb1x00_ts *ts = id; + ucb1x00_disable_irq(ts->ucb, UCB_IRQ_TSPX, UCB_FALLING); + wake_up(&ts->irq_wait); +} + +static int ucb1x00_ts_startup(struct ucb1x00_ts *ts) +{ + int ret = 0; + + if (down_interruptible(&ts->sem)) + return -EINTR; + + if (ts->use_count++ != 0) + goto out; + + if (ts->rtask) + panic("ucb1x00: rtask running?"); + + init_waitqueue_head(&ts->irq_wait); + ret = ucb1x00_hook_irq(ts->ucb, UCB_IRQ_TSPX, ucb1x00_ts_irq, ts); + if (ret < 0) + goto out; + + /* + * If we do this at all, we should allow the user to + * measure and read the X and Y resistance at any time. + */ + ucb1x00_adc_enable(ts->ucb); + ts->x_res = ucb1x00_ts_read_xres(ts); + ts->y_res = ucb1x00_ts_read_yres(ts); + ucb1x00_adc_disable(ts->ucb); + + init_completion(&ts->init_exit); + ret = kernel_thread(ucb1x00_thread, ts, 0); + if (ret >= 0) { + wait_for_completion(&ts->init_exit); + ret = 0; + } else { + ucb1x00_free_irq(ts->ucb, UCB_IRQ_TSPX, ts); + } + + out: + if (ret) + ts->use_count--; + up(&ts->sem); + return ret; +} + +/* + * Release touchscreen resources. Disable IRQs. + */ +static void ucb1x00_ts_shutdown(struct ucb1x00_ts *ts) +{ + if (--ts->use_count == 0) { + if (ts->rtask) { + send_sig(SIGKILL, ts->rtask, 1); + wait_for_completion(&ts->init_exit); + } + + ucb1x00_enable(ts->ucb); + ucb1x00_free_irq(ts->ucb, UCB_IRQ_TSPX, ts); + ucb1x00_reg_write(ts->ucb, UCB_TS_CR, 0); + ucb1x00_disable(ts->ucb); + } +} + +#ifdef CONFIG_PM +static int ucb1x00_ts_pm (struct pm_dev *dev, pm_request_t rqst, void *data) +{ + struct ucb1x00_ts *ts = (struct ucb1x00_ts *) (dev->data); + + if (rqst == PM_RESUME && ts->rtask != NULL) { + /* + * Restart the TS thread to ensure the + * TS interrupt mode is set up again + * after sleep. + */ + ts->restart = 1; + wake_up(&ts->irq_wait); + } + return 0; +} +#endif + + +/* + * Initialisation. + */ +static int __init ucb1x00_ts_init(void) +{ + struct ucb1x00_ts *ts = &ucbts; + + ts->ucb = ucb1x00_get(); + if (!ts->ucb) + return -ENODEV; + + ts->adcsync = adcsync; + init_MUTEX(&ts->sem); + +#ifdef CONFIG_PM + ts->pmdev = pm_register(PM_SYS_DEV, PM_SYS_UNKNOWN, ucb1x00_ts_pm); + if (ts->pmdev == NULL) + printk("ucb1x00_ts: unable to register in PM.\n"); + else + ts->pmdev->data = ts; +#endif + return ucb1x00_ts_register(ts); +} + +static void __exit ucb1x00_ts_exit(void) +{ + struct ucb1x00_ts *ts = &ucbts; + + ucb1x00_ts_deregister(ts); + +#ifdef CONFIG_PM + if (ts->pmdev) + pm_unregister(ts->pmdev); +#endif +} + +#ifndef MODULE + +/* + * Parse kernel command-line options. + * + * syntax : ucbts=[sync|nosync],... + */ +static int __init ucb1x00_ts_setup(char *str) +{ + char *p; + + while ((p = strsep(&str, ",")) != NULL) { + if (strcmp(p, "sync") == 0) + adcsync = UCB_SYNC; + } + + return 1; +} + +__setup("ucbts=", ucb1x00_ts_setup); + +#else + +MODULE_PARM(adcsync, "i"); +MODULE_PARM_DESC(adcsync, "Enable use of ADCSYNC signal"); + +#endif + +module_init(ucb1x00_ts_init); +module_exit(ucb1x00_ts_exit); + +MODULE_AUTHOR("Russell King "); +MODULE_DESCRIPTION("UCB1x00 touchscreen driver"); +MODULE_LICENSE("GPL"); diff -urN orig/drivers/misc/ucb1x00.h linux/drivers/misc/ucb1x00.h --- orig/drivers/misc/ucb1x00.h Thu Jan 1 01:00:00 1970 +++ linux/drivers/misc/ucb1x00.h Thu Oct 24 15:24:52 2002 @@ -0,0 +1,232 @@ +/* + * linux/drivers/misc/ucb1x00.h + * + * Copyright (C) 2001 Russell King, All Rights Reserved. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License. + */ +#ifndef UCB1200_H +#define UCB1200_H + +#define UCB_IO_DATA 0x00 +#define UCB_IO_DIR 0x01 + +#define UCB_IO_0 (1 << 0) +#define UCB_IO_1 (1 << 1) +#define UCB_IO_2 (1 << 2) +#define UCB_IO_3 (1 << 3) +#define UCB_IO_4 (1 << 4) +#define UCB_IO_5 (1 << 5) +#define UCB_IO_6 (1 << 6) +#define UCB_IO_7 (1 << 7) +#define UCB_IO_8 (1 << 8) +#define UCB_IO_9 (1 << 9) + +#define UCB_IE_RIS 0x02 +#define UCB_IE_FAL 0x03 +#define UCB_IE_STATUS 0x04 +#define UCB_IE_CLEAR 0x04 +#define UCB_IE_ADC (1 << 11) +#define UCB_IE_TSPX (1 << 12) +#define UCB_IE_TSMX (1 << 13) +#define UCB_IE_TCLIP (1 << 14) +#define UCB_IE_ACLIP (1 << 15) + +#define UCB_IRQ_TSPX 12 + +#define UCB_TC_A 0x05 +#define UCB_TC_A_LOOP (1 << 7) /* UCB1200 */ +#define UCB_TC_A_AMPL (1 << 7) /* UCB1300 */ + +#define UCB_TC_B 0x06 +#define UCB_TC_B_VOICE_ENA (1 << 3) +#define UCB_TC_B_CLIP (1 << 4) +#define UCB_TC_B_ATT (1 << 6) +#define UCB_TC_B_SIDE_ENA (1 << 11) +#define UCB_TC_B_MUTE (1 << 13) +#define UCB_TC_B_IN_ENA (1 << 14) +#define UCB_TC_B_OUT_ENA (1 << 15) + +#define UCB_AC_A 0x07 +#define UCB_AC_B 0x08 +#define UCB_AC_B_LOOP (1 << 8) +#define UCB_AC_B_MUTE (1 << 13) +#define UCB_AC_B_IN_ENA (1 << 14) +#define UCB_AC_B_OUT_ENA (1 << 15) + +#define UCB_TS_CR 0x09 +#define UCB_TS_CR_TSMX_POW (1 << 0) +#define UCB_TS_CR_TSPX_POW (1 << 1) +#define UCB_TS_CR_TSMY_POW (1 << 2) +#define UCB_TS_CR_TSPY_POW (1 << 3) +#define UCB_TS_CR_TSMX_GND (1 << 4) +#define UCB_TS_CR_TSPX_GND (1 << 5) +#define UCB_TS_CR_TSMY_GND (1 << 6) +#define UCB_TS_CR_TSPY_GND (1 << 7) +#define UCB_TS_CR_MODE_INT (0 << 8) +#define UCB_TS_CR_MODE_PRES (1 << 8) +#define UCB_TS_CR_MODE_POS (2 << 8) +#define UCB_TS_CR_BIAS_ENA (1 << 11) +#define UCB_TS_CR_TSPX_LOW (1 << 12) +#define UCB_TS_CR_TSMX_LOW (1 << 13) + +#define UCB_ADC_CR 0x0a +#define UCB_ADC_SYNC_ENA (1 << 0) +#define UCB_ADC_VREFBYP_CON (1 << 1) +#define UCB_ADC_INP_TSPX (0 << 2) +#define UCB_ADC_INP_TSMX (1 << 2) +#define UCB_ADC_INP_TSPY (2 << 2) +#define UCB_ADC_INP_TSMY (3 << 2) +#define UCB_ADC_INP_AD0 (4 << 2) +#define UCB_ADC_INP_AD1 (5 << 2) +#define UCB_ADC_INP_AD2 (6 << 2) +#define UCB_ADC_INP_AD3 (7 << 2) +#define UCB_ADC_EXT_REF (1 << 5) +#define UCB_ADC_START (1 << 7) +#define UCB_ADC_ENA (1 << 15) + +#define UCB_ADC_DATA 0x0b +#define UCB_ADC_DAT_VAL (1 << 15) +#define UCB_ADC_DAT(x) (((x) & 0x7fe0) >> 5) + +#define UCB_ID 0x0c +#define UCB_ID_1200 0x1004 +#define UCB_ID_1300 0x1005 + +#define UCB_MODE 0x0d +#define UCB_MODE_DYN_VFLAG_ENA (1 << 12) +#define UCB_MODE_AUD_OFF_CAN (1 << 13) + +#include "mcp.h" + +struct ucb1x00; + +struct ucb1x00_irq { + void *devid; + void (*fn)(int, void *); +}; + +struct ucb1x00 { + spinlock_t lock; + struct mcp *mcp; + struct pm_dev *pmdev; + unsigned int irq; + struct semaphore adc_sem; + spinlock_t io_lock; + u16 id; + u16 io_dir; + u16 io_out; + u16 adc_cr; + u16 irq_fal_enbl; + u16 irq_ris_enbl; + struct ucb1x00_irq irq_handler[16]; +}; + +/** + * ucb1x00_clkrate - return the UCB1x00 SIB clock rate + * @ucb: UCB1x00 structure describing chip + * + * Return the SIB clock rate in Hz. + */ +static inline unsigned int ucb1x00_clkrate(struct ucb1x00 *ucb) +{ + return mcp_get_sclk_rate(ucb->mcp); +} + +/** + * ucb1x00_enable - enable the UCB1x00 SIB clock + * @ucb: UCB1x00 structure describing chip + * + * Enable the SIB clock. This can be called multiple times. + */ +static inline void ucb1x00_enable(struct ucb1x00 *ucb) +{ + mcp_enable(ucb->mcp); +} + +/** + * ucb1x00_disable - disable the UCB1x00 SIB clock + * @ucb: UCB1x00 structure describing chip + * + * Disable the SIB clock. The SIB clock will only be disabled + * when the number of ucb1x00_enable calls match the number of + * ucb1x00_disable calls. + */ +static inline void ucb1x00_disable(struct ucb1x00 *ucb) +{ + mcp_disable(ucb->mcp); +} + +/** + * ucb1x00_reg_write - write a UCB1x00 register + * @ucb: UCB1x00 structure describing chip + * @reg: UCB1x00 4-bit register index to write + * @val: UCB1x00 16-bit value to write + * + * Write the UCB1x00 register @reg with value @val. The SIB + * clock must be running for this function to return. + */ +static inline void ucb1x00_reg_write(struct ucb1x00 *ucb, unsigned int reg, unsigned int val) +{ + mcp_reg_write(ucb->mcp, reg, val); +} + +/** + * ucb1x00_reg_read - read a UCB1x00 register + * @ucb: UCB1x00 structure describing chip + * @reg: UCB1x00 4-bit register index to write + * + * Read the UCB1x00 register @reg and return its value. The SIB + * clock must be running for this function to return. + */ +static inline unsigned int ucb1x00_reg_read(struct ucb1x00 *ucb, unsigned int reg) +{ + return mcp_reg_read(ucb->mcp, reg); +} +/** + * ucb1x00_set_audio_divisor - + * @ucb: UCB1x00 structure describing chip + * @div: SIB clock divisor + */ +static inline void ucb1x00_set_audio_divisor(struct ucb1x00 *ucb, unsigned int div) +{ + mcp_set_audio_divisor(ucb->mcp, div); +} + +/** + * ucb1x00_set_telecom_divisor - + * @ucb: UCB1x00 structure describing chip + * @div: SIB clock divisor + */ +static inline void ucb1x00_set_telecom_divisor(struct ucb1x00 *ucb, unsigned int div) +{ + mcp_set_telecom_divisor(ucb->mcp, div); +} + +struct ucb1x00 *ucb1x00_get(void); + +void ucb1x00_io_set_dir(struct ucb1x00 *ucb, unsigned int, unsigned int); +void ucb1x00_io_write(struct ucb1x00 *ucb, unsigned int, unsigned int); +unsigned int ucb1x00_io_read(struct ucb1x00 *ucb); + +#define UCB_NOSYNC (0) +#define UCB_SYNC (1) + +unsigned int ucb1x00_adc_read(struct ucb1x00 *ucb, int adc_channel, int sync); +void ucb1x00_adc_enable(struct ucb1x00 *ucb); +void ucb1x00_adc_disable(struct ucb1x00 *ucb); + +/* + * Which edges of the IRQ do you want to control today? + */ +#define UCB_RISING (1 << 0) +#define UCB_FALLING (1 << 1) + +int ucb1x00_hook_irq(struct ucb1x00 *ucb, unsigned int idx, void (*fn)(int, void *), void *devid); +void ucb1x00_enable_irq(struct ucb1x00 *ucb, unsigned int idx, int edges); +void ucb1x00_disable_irq(struct ucb1x00 *ucb, unsigned int idx, int edges); +int ucb1x00_free_irq(struct ucb1x00 *ucb, unsigned int idx, void *devid); + +#endif diff -urN orig/drivers/mtd/Config.in linux/drivers/mtd/Config.in --- orig/drivers/mtd/Config.in Mon Aug 5 13:31:01 2002 +++ linux/drivers/mtd/Config.in Mon Aug 5 13:44:02 2002 @@ -1,5 +1,5 @@ -# $Id: Config.in,v 1.73 2002/03/08 16:34:35 rkaiser Exp $ +# $Id: Config.in,v 1.74 2002/04/23 13:52:14 mag Exp $ mainmenu_option next_comment comment 'Memory Technology Devices (MTD)' @@ -14,8 +14,8 @@ dep_tristate ' MTD partitioning support' CONFIG_MTD_PARTITIONS $CONFIG_MTD dep_tristate ' MTD concatenating support' CONFIG_MTD_CONCAT $CONFIG_MTD dep_tristate ' RedBoot partition table parsing' CONFIG_MTD_REDBOOT_PARTS $CONFIG_MTD_PARTITIONS + dep_tristate ' Command line partition table parsing' CONFIG_MTD_CMDLINE_PARTS $CONFIG_MTD_PARTITIONS if [ "$CONFIG_ARM" = "y" ]; then - dep_tristate ' Compaq bootldr partition table parsing' CONFIG_MTD_BOOTLDR_PARTS $CONFIG_MTD_PARTITIONS dep_tristate ' ARM Firmware Suite partition parsing' CONFIG_MTD_AFS_PARTS $CONFIG_MTD_PARTITIONS fi diff -urN orig/drivers/mtd/Makefile linux/drivers/mtd/Makefile --- orig/drivers/mtd/Makefile Mon Aug 5 13:31:01 2002 +++ linux/drivers/mtd/Makefile Mon Aug 5 13:44:03 2002 @@ -8,7 +8,7 @@ # Note 2! The CFLAGS definitions are now inherited from the # parent makes.. # -# $Id: Makefile,v 1.65 2002/03/22 07:10:34 dwmw2 Exp $ +# $Id: Makefile,v 1.66 2002/04/23 13:52:14 mag Exp $ obj-y += chips/chipslink.o maps/mapslink.o \ @@ -19,7 +19,7 @@ O_TARGET := mtdlink.o -export-objs := mtdcore.o mtdpart.o redboot.o bootldr.o afs.o mtdconcat.o +export-objs := mtdcore.o mtdpart.o redboot.o cmdline.o afs.o mtdconcat.o list-multi := nftl.o mod-subdirs := @@ -44,10 +44,10 @@ # Core functionality. obj-$(CONFIG_MTD) += mtdcore.o -obj-$(CONFIG_MTD_CONCAT) += mtdconcat.o +obj-$(CONFIG_MTD_CONCAT) += mtdconcat.o obj-$(CONFIG_MTD_PARTITIONS) += mtdpart.o obj-$(CONFIG_MTD_REDBOOT_PARTS) += redboot.o -obj-$(CONFIG_MTD_BOOTLDR_PARTS) += bootldr.o +obj-$(CONFIG_MTD_CMDLINE_PARTS) += cmdline.o obj-$(CONFIG_MTD_AFS_PARTS) += afs.o # 'Users' - code which presents functionality to userspace. diff -urN orig/drivers/mtd/afs.c linux/drivers/mtd/afs.c --- orig/drivers/mtd/afs.c Mon Aug 5 13:31:01 2002 +++ linux/drivers/mtd/afs.c Mon Aug 5 13:44:03 2002 @@ -21,7 +21,7 @@ This is access code for flashes using ARM's flash partitioning standards. - $Id: afs.c,v 1.7 2001/11/01 20:04:54 rmk Exp $ + $Id: afs.c,v 1.8 2002/05/04 08:49:09 rmk Exp $ ======================================================================*/ @@ -169,6 +169,7 @@ if (!parts) return -ENOMEM; + memset(parts, 0, sz); str = (char *)(parts + idx); /* diff -urN orig/drivers/mtd/bootldr.c linux/drivers/mtd/bootldr.c --- orig/drivers/mtd/bootldr.c Sun Oct 14 20:53:08 2001 +++ linux/drivers/mtd/bootldr.c Thu Jan 1 01:00:00 1970 @@ -1,214 +0,0 @@ -/* - * Read flash partition table from Compaq Bootloader - * - * Copyright 2001 Compaq Computer Corporation. - * - * $Id: bootldr.c,v 1.6 2001/10/02 15:05:11 dwmw2 Exp $ - * - * Use consistent with the GNU GPL is permitted, - * provided that this copyright notice is - * preserved in its entirety in all copies and derived works. - * - * COMPAQ COMPUTER CORPORATION MAKES NO WARRANTIES, EXPRESSED OR IMPLIED, - * AS TO THE USEFULNESS OR CORRECTNESS OF THIS CODE OR ITS - * FITNESS FOR ANY PARTICULAR PURPOSE. - * - */ - -/* - * Maintainer: Jamey Hicks (jamey.hicks@compaq.com) - */ - -#include -#include - -#include -#include -#include -#include - -#define FLASH_PARTITION_NAMELEN 32 -enum LFR_FLAGS { - LFR_SIZE_PREFIX = 1, /* prefix data with 4-byte size */ - LFR_PATCH_BOOTLDR = 2, /* patch bootloader's 0th instruction */ - LFR_KERNEL = 4, /* add BOOTIMG_MAGIC, imgsize and VKERNEL_BASE to head of programmed region (see bootldr.c) */ - LFR_EXPAND = 8 /* expand partition size to fit rest of flash */ -}; - -// the tags are parsed too early to malloc or alloc_bootmem so we'll fix it -// for now -#define MAX_NUM_PARTITIONS 8 -typedef struct FlashRegion { - char name[FLASH_PARTITION_NAMELEN]; - unsigned long base; - unsigned long size; - enum LFR_FLAGS flags; -} FlashRegion; - -typedef struct BootldrFlashPartitionTable { - int magic; /* should be filled with 0x646c7470 (btlp) BOOTLDR_PARTITION_MAGIC */ - int npartitions; - struct FlashRegion partition[8]; -} BootldrFlashPartitionTable; - -#define BOOTLDR_MAGIC 0x646c7462 /* btld: marks a valid bootldr image */ -#define BOOTLDR_PARTITION_MAGIC 0x646c7470 /* btlp: marks a valid bootldr partition table in params sector */ - -#define BOOTLDR_MAGIC_OFFSET 0x20 /* offset 0x20 into the bootldr */ -#define BOOTCAP_OFFSET 0X30 /* offset 0x30 into the bootldr */ - -#define BOOTCAP_WAKEUP (1<<0) -#define BOOTCAP_PARTITIONS (1<<1) /* partition table stored in params sector */ -#define BOOTCAP_PARAMS_AFTER_BOOTLDR (1<<2) /* params sector right after bootldr sector(s), else in last sector */ - -static struct BootldrFlashPartitionTable Table; -static struct BootldrFlashPartitionTable *partition_table = NULL; - - -int parse_bootldr_partitions(struct mtd_info *master, struct mtd_partition **pparts) -{ - struct mtd_partition *parts; - int ret, retlen, i; - int npartitions = 0; - long partition_table_offset; - long bootmagic = 0; - long bootcap = 0; - int namelen = 0; - - char *names; - -#if 0 - /* verify bootldr magic */ - ret = master->read(master, BOOTLDR_MAGIC_OFFSET, sizeof(long), &retlen, (void *)&bootmagic); - if (ret) - goto out; - if (bootmagic != BOOTLDR_MAGIC) - goto out; - /* see if bootldr supports partition tables and where to find the partition table */ - ret = master->read(master, BOOTCAP_OFFSET, sizeof(long), &retlen, (void *)&bootcap); - if (ret) - goto out; - - if (!(bootcap & BOOTCAP_PARTITIONS)) - goto out; - if (bootcap & BOOTCAP_PARAMS_AFTER_BOOTLDR) - partition_table_offset = master->erasesize; - else - partition_table_offset = master->size - master->erasesize; - - printk(__FUNCTION__ ": partition_table_offset=%#lx\n", partition_table_offset); - printk(__FUNCTION__ ": ptable_addr=%#lx\n", ptable_addr); - - - /* Read the partition table */ - partition_table = (struct BootldrFlashPartitionTable *)kmalloc(PAGE_SIZE, GFP_KERNEL); - if (!partition_table) - return -ENOMEM; - - ret = master->read(master, partition_table_offset, - PAGE_SIZE, &retlen, (void *)partition_table); - if (ret) - goto out; - -#endif - if (!partition_table) - return -ENOMEM; - - - printk(__FUNCTION__ ": magic=%#x\n", partition_table->magic); - printk(__FUNCTION__ ": numPartitions=%#x\n", partition_table->npartitions); - - - /* check for partition table magic number */ - if (partition_table->magic != BOOTLDR_PARTITION_MAGIC) - goto out; - npartitions = (partition_table->npartitions > MAX_NUM_PARTITIONS)? - MAX_NUM_PARTITIONS:partition_table->npartitions; - - printk(__FUNCTION__ ": npartitions=%#x\n", npartitions); - - for (i = 0; i < npartitions; i++) { - namelen += strlen(partition_table->partition[i].name) + 1; - } - - parts = kmalloc(sizeof(*parts)*npartitions + namelen, GFP_KERNEL); - if (!parts) { - ret = -ENOMEM; - goto out; - } - names = (char *)&parts[npartitions]; - memset(parts, 0, sizeof(*parts)*npartitions + namelen); - - - - // from here we use the partition table - for (i = 0; i < npartitions; i++) { - struct FlashRegion *partition = &partition_table->partition[i]; - const char *name = partition->name; - parts[i].name = names; - names += strlen(name) + 1; - strcpy(parts[i].name, name); - - if (partition->flags & LFR_EXPAND) - parts[i].size = MTDPART_SIZ_FULL; - else - parts[i].size = partition->size; - parts[i].offset = partition->base; - parts[i].mask_flags = 0; - - printk(" partition %s o=%x s=%x\n", - parts[i].name, parts[i].offset, parts[i].size); - - } - - ret = npartitions; - *pparts = parts; - - out: -#if 0 - if (partition_table) - kfree(partition_table); -#endif - - return ret; -} - - -static int __init parse_tag_ptable(const struct tag *tag) -{ - char buf[128]; - int i; - int j; - - partition_table = &Table; - -#ifdef CONFIG_DEBUG_LL - sprintf(buf,"ptable: magic = = 0x%lx npartitions= %d \n", - tag->u.ptable.magic,tag->u.ptable.npartitions); - printascii(buf); - - for (i=0; iu.ptable.npartitions; i++){ - sprintf(buf,"ptable: partition name = %s base= 0x%lx size= 0x%lx flags= 0x%lx\n", - (char *) (&tag->u.ptable.partition[i].name[0]), - tag->u.ptable.partition[i].base, - tag->u.ptable.partition[i].size, - tag->u.ptable.partition[i].flags); - printascii(buf); - } -#endif - - memcpy((void *)partition_table,(void *) (&(tag->u.ptable)),sizeof(partition_table) + - sizeof(struct FlashRegion)*tag->u.ptable.npartitions); - - - return 0; -} - -__tagtable(ATAG_PTABLE, parse_tag_ptable); - -EXPORT_SYMBOL(parse_bootldr_partitions); - - -MODULE_LICENSE("GPL"); -MODULE_AUTHOR("Compaq Computer Corporation"); -MODULE_DESCRIPTION("Parsing code for Compaq bootldr partitions"); diff -urN orig/drivers/mtd/chips/Config.in linux/drivers/mtd/chips/Config.in --- orig/drivers/mtd/chips/Config.in Sun Oct 14 20:53:08 2001 +++ linux/drivers/mtd/chips/Config.in Fri Mar 1 10:38:39 2002 @@ -1,19 +1,18 @@ # drivers/mtd/chips/Config.in -# $Id: Config.in,v 1.12 2001/09/23 15:35:21 dwmw2 Exp $ +# $Id: Config.in,v 1.15 2002/02/13 16:29:15 dwmw2 Exp $ mainmenu_option next_comment comment 'RAM/ROM/Flash chip drivers' dep_tristate ' Detect flash chips by Common Flash Interface (CFI) probe' CONFIG_MTD_CFI $CONFIG_MTD -#dep_tristate ' Detect non-CFI Intel-compatible flash chips' CONFIG_MTD_INTELPROBE $CONFIG_MTD -dep_tristate ' Detect non-CFI AMD/JEDEC-compatible flash chips' CONFIG_MTD_JEDECPROBE $CONFIG_MTD +dep_tristate ' Detect JEDEC JESD21c compatible flash chips' CONFIG_MTD_JEDECPROBE $CONFIG_MTD -if [ "$CONFIG_MTD_CFI" = "y" -o "$CONFIG_MTD_INTELPROBE" = "y" -o "$CONFIG_MTD_JEDECPROBE" = "y" ]; then +if [ "$CONFIG_MTD_CFI" = "y" -o "$CONFIG_MTD_JEDECPROBE" = "y" ]; then define_bool CONFIG_MTD_GEN_PROBE y else - if [ "$CONFIG_MTD_CFI" = "m" -o "$CONFIG_MTD_INTELPROBE" = "m" -o "$CONFIG_MTD_JEDECPROBE" = "m" ]; then + if [ "$CONFIG_MTD_CFI" = "m" -o "$CONFIG_MTD_JEDECPROBE" = "m" ]; then define_bool CONFIG_MTD_GEN_PROBE m else define_bool CONFIG_MTD_GEN_PROBE n @@ -31,6 +30,7 @@ bool ' Support 8-bit buswidth' CONFIG_MTD_CFI_B1 bool ' Support 16-bit buswidth' CONFIG_MTD_CFI_B2 bool ' Support 32-bit buswidth' CONFIG_MTD_CFI_B4 + bool ' Support 64-bit buswidth' CONFIG_MTD_CFI_B8 if [ "$CONFIG_MTD_CFI_B1" = "y" ]; then define_bool CONFIG_MTD_CFI_I1 y else @@ -38,6 +38,7 @@ fi bool ' Support 2-chip flash interleave' CONFIG_MTD_CFI_I2 bool ' Support 4-chip flash interleave' CONFIG_MTD_CFI_I4 + bool ' Support 8-chip flash interleave' CONFIG_MTD_CFI_I8 fi fi fi diff -urN orig/drivers/mtd/chips/cfi_cmdset_0001.c linux/drivers/mtd/chips/cfi_cmdset_0001.c --- orig/drivers/mtd/chips/cfi_cmdset_0001.c Sun Oct 14 20:53:08 2001 +++ linux/drivers/mtd/chips/cfi_cmdset_0001.c Thu Jun 20 20:16:54 2002 @@ -4,7 +4,7 @@ * * (C) 2000 Red Hat. GPL'd * - * $Id: cfi_cmdset_0001.c,v 1.87 2001/10/02 15:05:11 dwmw2 Exp $ + * $Id: cfi_cmdset_0001.c,v 1.99 2002/06/19 09:19:57 jocke Exp $ * * * 10/10/2000 Nicolas Pitre @@ -13,6 +13,8 @@ * - scalability vs code size is completely set at compile-time * (see include/linux/mtd/cfi.h for selection) * - optimized write buffer method + * 02/05/2002 Christopher Hoover / + * - reworked lock/unlock/erase support for var size flash */ #include @@ -30,7 +32,11 @@ #include #include +// debugging, turns off buffer write mode #define FORCE_WORD_WRITE + static int cfi_intelext_read (struct mtd_info *, loff_t, size_t, size_t *, u_char *); +static int cfi_intelext_read_user_prot_reg (struct mtd_info *, loff_t, size_t, size_t *, u_char *); +static int cfi_intelext_read_fact_prot_reg (struct mtd_info *, loff_t, size_t, size_t *, u_char *); static int cfi_intelext_write_words(struct mtd_info *, loff_t, size_t, size_t *, const u_char *); static int cfi_intelext_write_buffers(struct mtd_info *, loff_t, size_t, size_t *, const u_char *); static int cfi_intelext_erase_varsize(struct mtd_info *, struct erase_info *); @@ -59,6 +65,7 @@ #ifdef DEBUG_CFI_FEATURES static void cfi_tell_features(struct cfi_pri_intelext *extp) { + int i; printk(" Feature/Command Support: %4.4X\n", extp->FeatureSupport); printk(" - Chip Erase: %s\n", extp->FeatureSupport&1?"supported":"unsupported"); printk(" - Suspend Erase: %s\n", extp->FeatureSupport&2?"supported":"unsupported"); @@ -110,7 +117,7 @@ int i; __u32 base = cfi->chips[0].start; - if (cfi->cfi_mode) { + if (cfi->cfi_mode == CFI_MODE_CFI) { /* * It's a real CFI chip, not one for which the probe * routine faked a CFI structure. So we read the feature @@ -149,17 +156,18 @@ } /* Do some byteswapping if necessary */ - extp->FeatureSupport = cfi32_to_cpu(extp->FeatureSupport); - extp->BlkStatusRegMask = cfi32_to_cpu(extp->BlkStatusRegMask); - + extp->FeatureSupport = le32_to_cpu(extp->FeatureSupport); + extp->BlkStatusRegMask = le16_to_cpu(extp->BlkStatusRegMask); + extp->ProtRegAddr = le16_to_cpu(extp->ProtRegAddr); + #ifdef DEBUG_CFI_FEATURES /* Tell the user about it in lots of lovely detail */ cfi_tell_features(extp); #endif /* Install our own private info structure */ - cfi->cmdset_priv = extp; - } + cfi->cmdset_priv = extp; + } for (i=0; i< cfi->numchips; i++) { cfi->chips[i].word_write_time = 128; @@ -240,13 +248,19 @@ /* Also select the correct geometry setup too */ mtd->erase = cfi_intelext_erase_varsize; mtd->read = cfi_intelext_read; +#ifndef FORCE_WORD_WRITE if ( cfi->cfiq->BufWriteTimeoutTyp ) { - //printk(KERN_INFO "Using buffer write method\n" ); + printk("Using buffer write method\n" ); mtd->write = cfi_intelext_write_buffers; } else { - //printk(KERN_INFO "Using word write method\n" ); +#else + { +#endif + printk("Using word write method\n" ); mtd->write = cfi_intelext_write_words; } + mtd->read_user_prot_reg = cfi_intelext_read_user_prot_reg; + mtd->read_fact_prot_reg = cfi_intelext_read_fact_prot_reg; mtd->sync = cfi_intelext_sync; mtd->lock = cfi_intelext_lock; mtd->unlock = cfi_intelext_unlock; @@ -262,7 +276,7 @@ static inline int do_read_onechip(struct map_info *map, struct flchip *chip, loff_t adr, size_t len, u_char *buf) { - __u32 status, status_OK; + cfi_word status, status_OK; unsigned long timeo; DECLARE_WAITQUEUE(wait, current); int suspended = 0; @@ -286,7 +300,7 @@ */ switch (chip->state) { case FL_ERASING: - if (!((struct cfi_pri_intelext *)cfi->cmdset_priv)->FeatureSupport & 2) + if (!(((struct cfi_pri_intelext *)cfi->cmdset_priv)->FeatureSupport & 2)) goto sleep; /* We don't support erase suspend */ cfi_write (map, CMD(0xb0), cmd_addr); @@ -312,7 +326,7 @@ chip->state = FL_ERASING; spin_unlock_bh(chip->mutex); printk(KERN_ERR "Chip not ready after erase " - "suspended: status = 0x%x\n", status); + "suspended: status = 0x%llx\n", (__u64)status); return -EIO; } @@ -350,7 +364,7 @@ /* Urgh. Chip not yet ready to talk to us. */ if (time_after(jiffies, timeo)) { spin_unlock_bh(chip->mutex); - printk(KERN_ERR "waiting for chip to be ready timed out in read. WSM status = %x\n", status); + printk(KERN_ERR "waiting for chip to be ready timed out in read. WSM status = %llx\n", (__u64)status); return -EIO; } @@ -433,10 +447,120 @@ return ret; } -static int do_write_oneword(struct map_info *map, struct flchip *chip, unsigned long adr, __u32 datum) +static int cfi_intelext_read_prot_reg (struct mtd_info *mtd, loff_t from, size_t len, size_t *retlen, u_char *buf, int base_offst, int reg_sz) { + struct map_info *map = mtd->priv; struct cfi_private *cfi = map->fldrv_priv; - __u32 status, status_OK; + struct cfi_pri_intelext *extp=cfi->cmdset_priv; + int ofs_factor = cfi->interleave * cfi->device_type; + int count=len; + struct flchip *chip; + int chip_num,offst; + unsigned long timeo; + DECLARE_WAITQUEUE(wait, current); + + chip=0; + /* Calculate which chip & protection register offset we need */ + chip_num=((unsigned int)from/reg_sz); + offst=from-(reg_sz*chip_num)+base_offst; + + while(count){ + + if(chip_num>=cfi->numchips) + goto out; + + /* Make sure that the chip is in the right state */ + + timeo = jiffies + HZ; + chip=&cfi->chips[chip_num]; + retry: + spin_lock_bh(chip->mutex); + + switch (chip->state) { + case FL_READY: + case FL_STATUS: + case FL_CFI_QUERY: + case FL_JEDEC_QUERY: + break; + + default: + /* Stick ourselves on a wait queue to be woken when + someone changes the status */ + set_current_state(TASK_UNINTERRUPTIBLE); + add_wait_queue(&chip->wq, &wait); + spin_unlock_bh(chip->mutex); + schedule(); + remove_wait_queue(&chip->wq, &wait); + timeo = jiffies + HZ; + goto retry; + } + + /* Now read the data required from this flash */ + + cfi_send_gen_cmd(0x90, 0x55,chip->start, map, cfi, cfi->device_type, NULL); + while(count && ((offst-base_offst)read8(map,(chip->start+(extp->ProtRegAddr*ofs_factor)+offst)); + buf++; + offst++; + count--; + } + + chip->state=FL_CFI_QUERY; + spin_unlock_bh(chip->mutex); + /* Move on to the next chip */ + chip_num++; + offst=base_offst; + + } + + out: + wake_up(&chip->wq); + return len-count; +} + +static int cfi_intelext_read_user_prot_reg (struct mtd_info *mtd, loff_t from, size_t len, size_t *retlen, u_char *buf) +{ + struct map_info *map = mtd->priv; + struct cfi_private *cfi = map->fldrv_priv; + struct cfi_pri_intelext *extp=cfi->cmdset_priv; + int base_offst,reg_sz; + + /* Check that we actually have some protection registers */ + if(!(extp->FeatureSupport&64)){ + printk(KERN_WARNING "%s: This flash device has no protection data to read!\n",map->name); + return 0; + } + + base_offst=(1<FactProtRegSize); + reg_sz=(1<UserProtRegSize); + + return cfi_intelext_read_prot_reg(mtd, from, len, retlen, buf, base_offst, reg_sz); +} + +static int cfi_intelext_read_fact_prot_reg (struct mtd_info *mtd, loff_t from, size_t len, size_t *retlen, u_char *buf) +{ + struct map_info *map = mtd->priv; + struct cfi_private *cfi = map->fldrv_priv; + struct cfi_pri_intelext *extp=cfi->cmdset_priv; + int base_offst,reg_sz; + + /* Check that we actually have some protection registers */ + if(!(extp->FeatureSupport&64)){ + printk(KERN_WARNING "%s: This flash device has no protection data to read!\n",map->name); + return 0; + } + + base_offst=0; + reg_sz=(1<FactProtRegSize); + + return cfi_intelext_read_prot_reg(mtd, from, len, retlen, buf, base_offst, reg_sz); +} + + +static int do_write_oneword(struct map_info *map, struct flchip *chip, unsigned long adr, cfi_word datum) +{ + struct cfi_private *cfi = map->fldrv_priv; + cfi_word status, status_OK; unsigned long timeo; DECLARE_WAITQUEUE(wait, current); int z; @@ -583,8 +707,8 @@ unsigned long bus_ofs = ofs & ~(CFIDEV_BUSWIDTH-1); int gap = ofs - bus_ofs; int i = 0, n = 0; - u_char tmp_buf[4]; - __u32 datum; + u_char tmp_buf[8]; + cfi_word datum; while (gap--) tmp_buf[i++] = 0xff; @@ -597,6 +721,8 @@ datum = *(__u16*)tmp_buf; } else if (cfi_buswidth_is_4()) { datum = *(__u32*)tmp_buf; + } else if (cfi_buswidth_is_8()) { + datum = *(__u64*)tmp_buf; } else { return -EINVAL; /* should never happen, but be safe */ } @@ -619,7 +745,7 @@ } while(len >= CFIDEV_BUSWIDTH) { - __u32 datum; + cfi_word datum; if (cfi_buswidth_is_1()) { datum = *(__u8*)buf; @@ -627,6 +753,8 @@ datum = *(__u16*)buf; } else if (cfi_buswidth_is_4()) { datum = *(__u32*)buf; + } else if (cfi_buswidth_is_8()) { + datum = *(__u64*)buf; } else { return -EINVAL; } @@ -651,8 +779,8 @@ if (len & (CFIDEV_BUSWIDTH-1)) { int i = 0, n = 0; - u_char tmp_buf[4]; - __u32 datum; + u_char tmp_buf[8]; + cfi_word datum; while (len--) tmp_buf[i++] = buf[n++]; @@ -663,6 +791,8 @@ datum = *(__u16*)tmp_buf; } else if (cfi_buswidth_is_4()) { datum = *(__u32*)tmp_buf; + } else if (cfi_buswidth_is_8()) { + datum = *(__u64*)tmp_buf; } else { return -EINVAL; /* should never happen, but be safe */ } @@ -683,7 +813,7 @@ unsigned long adr, const u_char *buf, int len) { struct cfi_private *cfi = map->fldrv_priv; - __u32 status, status_OK; + cfi_word status, status_OK; unsigned long cmd_adr, timeo; DECLARE_WAITQUEUE(wait, current); int wbufsize, z; @@ -706,8 +836,6 @@ */ switch (chip->state) { case FL_READY: - break; - case FL_CFI_QUERY: case FL_JEDEC_QUERY: cfi_write(map, CMD(0x70), cmd_adr); @@ -740,13 +868,23 @@ timeo = jiffies + HZ; goto retry; } - + /* We know we're now in FL_STATUS mode, and 'status' is current */ + /* §4.8 of the 28FxxxJ3A datasheet says "Any time SR.4 and/or SR.5 is set + [...], the device will not accept any more Write to Buffer commands". + So we must check here and reset those bits if they're set. Otherwise + we're just pissing in the wind */ + if (status & CMD(0x30)) { + printk(KERN_WARNING "SR.4 or SR.5 bits set in buffer write (status %x). Clearing.\n", status); + cfi_write(map, CMD(0x50), cmd_adr); + cfi_write(map, CMD(0x70), cmd_adr); + } ENABLE_VPP(map); - cfi_write(map, CMD(0xe8), cmd_adr); chip->state = FL_WRITING_TO_BUFFER; z = 0; for (;;) { + cfi_write(map, CMD(0xe8), cmd_adr); + status = cfi_read(map, cmd_adr); if ((status & status_OK) == status_OK) break; @@ -760,8 +898,11 @@ cfi_write(map, CMD(0x70), cmd_adr); chip->state = FL_STATUS; DISABLE_VPP(map); + printk(KERN_ERR "Chip not ready for buffer write. Xstatus = %llx, status = %llx\n", (__u64)status, (__u64)cfi_read(map, cmd_adr)); + /* Odd. Clear status bits */ + cfi_write(map, CMD(0x50), cmd_adr); + cfi_write(map, CMD(0x70), cmd_adr); spin_unlock_bh(chip->mutex); - printk(KERN_ERR "Chip not ready for buffer write. Xstatus = %x, status = %x\n", status, cfi_read(map, cmd_adr)); return -EIO; } } @@ -777,6 +918,8 @@ map->write16 (map, *((__u16*)buf)++, adr+z); } else if (cfi_buswidth_is_4()) { map->write32 (map, *((__u32*)buf)++, adr+z); + } else if (cfi_buswidth_is_8()) { + map->write64 (map, *((__u64*)buf)++, adr+z); } else { DISABLE_VPP(map); return -EINVAL; @@ -926,11 +1069,102 @@ return 0; } +typedef int (*varsize_frob_t)(struct map_info *map, struct flchip *chip, + unsigned long adr, void *thunk); -static inline int do_erase_oneblock(struct map_info *map, struct flchip *chip, unsigned long adr) +static int cfi_intelext_varsize_frob(struct mtd_info *mtd, varsize_frob_t frob, + loff_t ofs, size_t len, void *thunk) { + struct map_info *map = mtd->priv; struct cfi_private *cfi = map->fldrv_priv; - __u32 status, status_OK; + unsigned long adr; + int chipnum, ret = 0; + int i, first; + struct mtd_erase_region_info *regions = mtd->eraseregions; + + if (ofs > mtd->size) + return -EINVAL; + + if ((len + ofs) > mtd->size) + return -EINVAL; + + /* Check that both start and end of the requested erase are + * aligned with the erasesize at the appropriate addresses. + */ + + i = 0; + + /* Skip all erase regions which are ended before the start of + the requested erase. Actually, to save on the calculations, + we skip to the first erase region which starts after the + start of the requested erase, and then go back one. + */ + + while (i < mtd->numeraseregions && ofs >= regions[i].offset) + i++; + i--; + + /* OK, now i is pointing at the erase region in which this + erase request starts. Check the start of the requested + erase range is aligned with the erase size which is in + effect here. + */ + + if (ofs & (regions[i].erasesize-1)) + return -EINVAL; + + /* Remember the erase region we start on */ + first = i; + + /* Next, check that the end of the requested erase is aligned + * with the erase region at that address. + */ + + while (inumeraseregions && (ofs + len) >= regions[i].offset) + i++; + + /* As before, drop back one to point at the region in which + the address actually falls + */ + i--; + + if ((ofs + len) & (regions[i].erasesize-1)) + return -EINVAL; + + chipnum = ofs >> cfi->chipshift; + adr = ofs - (chipnum << cfi->chipshift); + + i=first; + + while(len) { + ret = (*frob)(map, &cfi->chips[chipnum], adr, thunk); + + if (ret) + return ret; + + adr += regions[i].erasesize; + len -= regions[i].erasesize; + + if (adr % (1<< cfi->chipshift) == ((regions[i].offset + (regions[i].erasesize * regions[i].numblocks)) %( 1<< cfi->chipshift))) + i++; + + if (adr >> cfi->chipshift) { + adr = 0; + chipnum++; + + if (chipnum >= cfi->numchips) + break; + } + } + + return 0; +} + + +static int do_erase_oneblock(struct map_info *map, struct flchip *chip, unsigned long adr, void *thunk) +{ + struct cfi_private *cfi = map->fldrv_priv; + cfi_word status, status_OK; unsigned long timeo; int retries = 3; DECLARE_WAITQUEUE(wait, current); @@ -990,7 +1224,8 @@ cfi_write(map, CMD(0x20), adr); cfi_write(map, CMD(0xD0), adr); chip->state = FL_ERASING; - + chip->oldstate = 0; + spin_unlock_bh(chip->mutex); schedule_timeout(HZ); spin_lock_bh(chip->mutex); @@ -1007,10 +1242,15 @@ spin_unlock_bh(chip->mutex); schedule(); remove_wait_queue(&chip->wq, &wait); - timeo = jiffies + (HZ*20); /* FIXME */ spin_lock_bh(chip->mutex); continue; } + if (chip->oldstate) { + /* This erase was suspended and resumed. + Adjust the timeout */ + timeo = jiffies + (HZ*20); /* FIXME */ + chip->oldstate = 0; + } status = cfi_read(map, adr); if ((status & status_OK) == status_OK) @@ -1020,7 +1260,11 @@ if (time_after(jiffies, timeo)) { cfi_write(map, CMD(0x70), adr); chip->state = FL_STATUS; - printk(KERN_ERR "waiting for erase to complete timed out. Xstatus = %x, status = %x.\n", status, cfi_read(map, adr)); + printk(KERN_ERR "waiting for erase at %08lx to complete timed out. Xstatus = %llx, status = %llx.\n", + adr, (__u64)status, (__u64)cfi_read(map, adr)); + /* Clear status bits */ + cfi_write(map, CMD(0x50), adr); + cfi_write(map, CMD(0x70), adr); DISABLE_VPP(map); spin_unlock_bh(chip->mutex); return -EIO; @@ -1048,31 +1292,31 @@ for (i = 1; i> (cfi->device_type * 8); } - printk(KERN_WARNING "Status is not identical for all chips: 0x%x. Merging to give 0x%02x\n", status, chipstatus); + printk(KERN_WARNING "Status is not identical for all chips: 0x%llx. Merging to give 0x%02x\n", (__u64)status, chipstatus); } /* Reset the error bits */ cfi_write(map, CMD(0x50), adr); cfi_write(map, CMD(0x70), adr); if ((chipstatus & 0x30) == 0x30) { - printk(KERN_NOTICE "Chip reports improper command sequence: status 0x%x\n", status); + printk(KERN_NOTICE "Chip reports improper command sequence: status 0x%llx\n", (__u64)status); ret = -EIO; } else if (chipstatus & 0x02) { /* Protection bit set */ ret = -EROFS; } else if (chipstatus & 0x8) { /* Voltage */ - printk(KERN_WARNING "Chip reports voltage low on erase: status 0x%x\n", status); + printk(KERN_WARNING "Chip reports voltage low on erase: status 0x%llx\n", (__u64)status); ret = -EIO; } else if (chipstatus & 0x20) { if (retries--) { - printk(KERN_DEBUG "Chip erase failed at 0x%08lx: status 0x%x. Retrying...\n", adr, status); + printk(KERN_DEBUG "Chip erase failed at 0x%08lx: status 0x%llx. Retrying...\n", adr, (__u64)status); timeo = jiffies + HZ; chip->state = FL_STATUS; spin_unlock_bh(chip->mutex); goto retry; } - printk(KERN_DEBUG "Chip erase failed at 0x%08lx: status 0x%x\n", adr, status); + printk(KERN_DEBUG "Chip erase failed at 0x%08lx: status 0x%llx\n", adr, (__u64)status); ret = -EIO; } } @@ -1083,89 +1327,17 @@ } int cfi_intelext_erase_varsize(struct mtd_info *mtd, struct erase_info *instr) -{ struct map_info *map = mtd->priv; - struct cfi_private *cfi = map->fldrv_priv; - unsigned long adr, len; - int chipnum, ret = 0; - int i, first; - struct mtd_erase_region_info *regions = mtd->eraseregions; - - if (instr->addr > mtd->size) - return -EINVAL; - - if ((instr->len + instr->addr) > mtd->size) - return -EINVAL; - - /* Check that both start and end of the requested erase are - * aligned with the erasesize at the appropriate addresses. - */ - - i = 0; - - /* Skip all erase regions which are ended before the start of - the requested erase. Actually, to save on the calculations, - we skip to the first erase region which starts after the - start of the requested erase, and then go back one. - */ - - while (i < mtd->numeraseregions && instr->addr >= regions[i].offset) - i++; - i--; - - /* OK, now i is pointing at the erase region in which this - erase request starts. Check the start of the requested - erase range is aligned with the erase size which is in - effect here. - */ - - if (instr->addr & (regions[i].erasesize-1)) - return -EINVAL; - - /* Remember the erase region we start on */ - first = i; - - /* Next, check that the end of the requested erase is aligned - * with the erase region at that address. - */ - - while (inumeraseregions && (instr->addr + instr->len) >= regions[i].offset) - i++; - - /* As before, drop back one to point at the region in which - the address actually falls - */ - i--; - - if ((instr->addr + instr->len) & (regions[i].erasesize-1)) - return -EINVAL; +{ + unsigned long ofs, len; + int ret; - chipnum = instr->addr >> cfi->chipshift; - adr = instr->addr - (chipnum << cfi->chipshift); + ofs = instr->addr; len = instr->len; - i=first; - - while(len) { - ret = do_erase_oneblock(map, &cfi->chips[chipnum], adr); - - if (ret) - return ret; - - adr += regions[i].erasesize; - len -= regions[i].erasesize; + ret = cfi_intelext_varsize_frob(mtd, do_erase_oneblock, ofs, len, 0); + if (ret) + return ret; - if (adr % (1<< cfi->chipshift) == ((regions[i].offset + (regions[i].erasesize * regions[i].numblocks)) %( 1<< cfi->chipshift))) - i++; - - if (adr >> cfi->chipshift) { - adr = 0; - chipnum++; - - if (chipnum >= cfi->numchips) - break; - } - } - instr->state = MTD_ERASE_DONE; if (instr->callback) instr->callback(instr); @@ -1230,159 +1402,28 @@ } } -static inline int do_lock_oneblock(struct map_info *map, struct flchip *chip, unsigned long adr) +#ifdef DEBUG_LOCK_BITS +static int do_printlockstatus_oneblock(struct map_info *map, struct flchip *chip, unsigned long adr, void *thunk) { struct cfi_private *cfi = map->fldrv_priv; - __u32 status, status_OK; - unsigned long timeo = jiffies + HZ; - DECLARE_WAITQUEUE(wait, current); - - adr += chip->start; - - /* Let's determine this according to the interleave only once */ - status_OK = CMD(0x80); - - timeo = jiffies + HZ; -retry: - spin_lock_bh(chip->mutex); - - /* Check that the chip's ready to talk to us. */ - switch (chip->state) { - case FL_CFI_QUERY: - case FL_JEDEC_QUERY: - case FL_READY: - cfi_write(map, CMD(0x70), adr); - chip->state = FL_STATUS; - - case FL_STATUS: - status = cfi_read(map, adr); - if ((status & status_OK) == status_OK) - break; - - /* Urgh. Chip not yet ready to talk to us. */ - if (time_after(jiffies, timeo)) { - spin_unlock_bh(chip->mutex); - printk(KERN_ERR "waiting for chip to be ready timed out in lock\n"); - return -EIO; - } - - /* Latency issues. Drop the lock, wait a while and retry */ - spin_unlock_bh(chip->mutex); - cfi_udelay(1); - goto retry; - - default: - /* Stick ourselves on a wait queue to be woken when - someone changes the status */ - set_current_state(TASK_UNINTERRUPTIBLE); - add_wait_queue(&chip->wq, &wait); - spin_unlock_bh(chip->mutex); - schedule(); - remove_wait_queue(&chip->wq, &wait); - timeo = jiffies + HZ; - goto retry; - } - - ENABLE_VPP(map); - cfi_write(map, CMD(0x60), adr); - cfi_write(map, CMD(0x01), adr); - chip->state = FL_LOCKING; - - spin_unlock_bh(chip->mutex); - schedule_timeout(HZ); - spin_lock_bh(chip->mutex); - - /* FIXME. Use a timer to check this, and return immediately. */ - /* Once the state machine's known to be working I'll do that */ - - timeo = jiffies + (HZ*2); - for (;;) { + int ofs_factor = cfi->interleave * cfi->device_type; - status = cfi_read(map, adr); - if ((status & status_OK) == status_OK) - break; - - /* OK Still waiting */ - if (time_after(jiffies, timeo)) { - cfi_write(map, CMD(0x70), adr); - chip->state = FL_STATUS; - printk(KERN_ERR "waiting for lock to complete timed out. Xstatus = %x, status = %x.\n", status, cfi_read(map, adr)); - DISABLE_VPP(map); - spin_unlock_bh(chip->mutex); - return -EIO; - } - - /* Latency issues. Drop the lock, wait a while and retry */ - spin_unlock_bh(chip->mutex); - cfi_udelay(1); - spin_lock_bh(chip->mutex); - } + cfi_send_gen_cmd(0x90, 0x55, 0, map, cfi, cfi->device_type, NULL); + printk(KERN_DEBUG "block status register for 0x%08lx is %x\n", + adr, cfi_read_query(map, adr+(2*ofs_factor))); + cfi_send_gen_cmd(0xff, 0x55, 0, map, cfi, cfi->device_type, NULL); - /* Done and happy. */ - chip->state = FL_STATUS; - DISABLE_VPP(map); - wake_up(&chip->wq); - spin_unlock_bh(chip->mutex); return 0; } -static int cfi_intelext_lock(struct mtd_info *mtd, loff_t ofs, size_t len) -{ - struct map_info *map = mtd->priv; - struct cfi_private *cfi = map->fldrv_priv; - unsigned long adr; - int chipnum, ret = 0; -#ifdef DEBUG_LOCK_BITS - int ofs_factor = cfi->interleave * cfi->device_type; -#endif - - if (ofs & (mtd->erasesize - 1)) - return -EINVAL; - - if (len & (mtd->erasesize -1)) - return -EINVAL; - - if ((len + ofs) > mtd->size) - return -EINVAL; - - chipnum = ofs >> cfi->chipshift; - adr = ofs - (chipnum << cfi->chipshift); - - while(len) { - -#ifdef DEBUG_LOCK_BITS - cfi_send_gen_cmd(0x90, 0x55, 0, map, cfi, cfi->device_type, NULL); - printk("before lock: block status register is %x\n",cfi_read_query(map, adr+(2*ofs_factor))); - cfi_send_gen_cmd(0xff, 0x55, 0, map, cfi, cfi->device_type, NULL); #endif - ret = do_lock_oneblock(map, &cfi->chips[chipnum], adr); - -#ifdef DEBUG_LOCK_BITS - cfi_send_gen_cmd(0x90, 0x55, 0, map, cfi, cfi->device_type, NULL); - printk("after lock: block status register is %x\n",cfi_read_query(map, adr+(2*ofs_factor))); - cfi_send_gen_cmd(0xff, 0x55, 0, map, cfi, cfi->device_type, NULL); -#endif - - if (ret) - return ret; - - adr += mtd->erasesize; - len -= mtd->erasesize; +#define DO_XXLOCK_ONEBLOCK_LOCK ((void *) 1) +#define DO_XXLOCK_ONEBLOCK_UNLOCK ((void *) 2) - if (adr >> cfi->chipshift) { - adr = 0; - chipnum++; - - if (chipnum >= cfi->numchips) - break; - } - } - return 0; -} -static inline int do_unlock_oneblock(struct map_info *map, struct flchip *chip, unsigned long adr) +static int do_xxlock_oneblock(struct map_info *map, struct flchip *chip, unsigned long adr, void *thunk) { struct cfi_private *cfi = map->fldrv_priv; - __u32 status, status_OK; + cfi_word status, status_OK; unsigned long timeo = jiffies + HZ; DECLARE_WAITQUEUE(wait, current); @@ -1411,7 +1452,7 @@ /* Urgh. Chip not yet ready to talk to us. */ if (time_after(jiffies, timeo)) { spin_unlock_bh(chip->mutex); - printk(KERN_ERR "waiting for chip to be ready timed out in unlock\n"); + printk(KERN_ERR __FUNCTION__ ": waiting for chip to be ready timed out\n"); return -EIO; } @@ -1434,9 +1475,16 @@ ENABLE_VPP(map); cfi_write(map, CMD(0x60), adr); - cfi_write(map, CMD(0xD0), adr); - chip->state = FL_UNLOCKING; - + + if (thunk == DO_XXLOCK_ONEBLOCK_LOCK) { + cfi_write(map, CMD(0x01), adr); + chip->state = FL_LOCKING; + } else if (thunk == DO_XXLOCK_ONEBLOCK_UNLOCK) { + cfi_write(map, CMD(0xD0), adr); + chip->state = FL_UNLOCKING; + } else + BUG(); + spin_unlock_bh(chip->mutex); schedule_timeout(HZ); spin_lock_bh(chip->mutex); @@ -1444,7 +1492,7 @@ /* FIXME. Use a timer to check this, and return immediately. */ /* Once the state machine's known to be working I'll do that */ - timeo = jiffies + (HZ*2); + timeo = jiffies + (HZ*20); for (;;) { status = cfi_read(map, adr); @@ -1455,13 +1503,13 @@ if (time_after(jiffies, timeo)) { cfi_write(map, CMD(0x70), adr); chip->state = FL_STATUS; - printk(KERN_ERR "waiting for unlock to complete timed out. Xstatus = %x, status = %x.\n", status, cfi_read(map, adr)); + printk(KERN_ERR "waiting for unlock to complete timed out. Xstatus = %llx, status = %llx.\n", (__u64)status, (__u64)cfi_read(map, adr)); DISABLE_VPP(map); spin_unlock_bh(chip->mutex); return -EIO; } - /* Latency issues. Drop the unlock, wait a while and retry */ + /* Latency issues. Drop the lock, wait a while and retry */ spin_unlock_bh(chip->mutex); cfi_udelay(1); spin_lock_bh(chip->mutex); @@ -1474,40 +1522,52 @@ spin_unlock_bh(chip->mutex); return 0; } -static int cfi_intelext_unlock(struct mtd_info *mtd, loff_t ofs, size_t len) + +static int cfi_intelext_lock(struct mtd_info *mtd, loff_t ofs, size_t len) { - struct map_info *map = mtd->priv; - struct cfi_private *cfi = map->fldrv_priv; - unsigned long adr; - int chipnum, ret = 0; + int ret; + #ifdef DEBUG_LOCK_BITS - int ofs_factor = cfi->interleave * cfi->device_type; + printk(KERN_DEBUG __FUNCTION__ + ": lock status before, ofs=0x%08llx, len=0x%08X\n", + ofs, len); + cfi_intelext_varsize_frob(mtd, do_printlockstatus_oneblock, + ofs, len, 0); #endif - chipnum = ofs >> cfi->chipshift; - adr = ofs - (chipnum << cfi->chipshift); - + ret = cfi_intelext_varsize_frob(mtd, do_xxlock_oneblock, + ofs, len, DO_XXLOCK_ONEBLOCK_LOCK); + #ifdef DEBUG_LOCK_BITS - { - unsigned long temp_adr = adr; - unsigned long temp_len = len; - - cfi_send_gen_cmd(0x90, 0x55, 0, map, cfi, cfi->device_type, NULL); - while (temp_len) { - printk("before unlock %x: block status register is %x\n",temp_adr,cfi_read_query(map, temp_adr+(2*ofs_factor))); - temp_adr += mtd->erasesize; - temp_len -= mtd->erasesize; - } - cfi_send_gen_cmd(0xff, 0x55, 0, map, cfi, cfi->device_type, NULL); - } + printk(KERN_DEBUG __FUNCTION__ + ": lock status after, ret=%d\n", ret); + cfi_intelext_varsize_frob(mtd, do_printlockstatus_oneblock, + ofs, len, 0); #endif - ret = do_unlock_oneblock(map, &cfi->chips[chipnum], adr); + return ret; +} + +static int cfi_intelext_unlock(struct mtd_info *mtd, loff_t ofs, size_t len) +{ + int ret; + +#ifdef DEBUG_LOCK_BITS + printk(KERN_DEBUG __FUNCTION__ + ": lock status before, ofs=0x%08llx, len=0x%08X\n", + ofs, len); + cfi_intelext_varsize_frob(mtd, do_printlockstatus_oneblock, + ofs, len, 0); +#endif + ret = cfi_intelext_varsize_frob(mtd, do_xxlock_oneblock, + ofs, len, DO_XXLOCK_ONEBLOCK_UNLOCK); + #ifdef DEBUG_LOCK_BITS - cfi_send_gen_cmd(0x90, 0x55, 0, map, cfi, cfi->device_type, NULL); - printk("after unlock: block status register is %x\n",cfi_read_query(map, adr+(2*ofs_factor))); - cfi_send_gen_cmd(0xff, 0x55, 0, map, cfi, cfi->device_type, NULL); + printk(KERN_DEBUG __FUNCTION__ + ": lock status after, ret=%d\n", ret); + cfi_intelext_varsize_frob(mtd, do_printlockstatus_oneblock, + ofs, len, 0); #endif return ret; diff -urN orig/drivers/mtd/chips/cfi_cmdset_0002.c linux/drivers/mtd/chips/cfi_cmdset_0002.c --- orig/drivers/mtd/chips/cfi_cmdset_0002.c Fri Nov 16 10:10:03 2001 +++ linux/drivers/mtd/chips/cfi_cmdset_0002.c Fri May 31 11:48:55 2002 @@ -8,7 +8,7 @@ * * This code is GPL * - * $Id: cfi_cmdset_0002.c,v 1.52 2001/10/24 09:37:30 dwmw2 Exp $ + * $Id: cfi_cmdset_0002.c,v 1.55 2002/05/18 07:42:03 dwmw2 Exp $ * */ @@ -30,6 +30,7 @@ static int cfi_amdstd_read (struct mtd_info *, loff_t, size_t, size_t *, u_char *); static int cfi_amdstd_write(struct mtd_info *, loff_t, size_t, size_t *, const u_char *); +static int cfi_amdstd_erase_chip(struct mtd_info *, struct erase_info *); static int cfi_amdstd_erase_onesize(struct mtd_info *, struct erase_info *); static int cfi_amdstd_erase_varsize(struct mtd_info *, struct erase_info *); static void cfi_amdstd_sync (struct mtd_info *); @@ -58,7 +59,7 @@ __u8 major, minor; __u32 base = cfi->chips[0].start; - if (cfi->cfi_mode==1){ + if (cfi->cfi_mode==CFI_MODE_CFI){ __u16 adr = primary?cfi->cfiq->P_ADR:cfi->cfiq->A_ADR; cfi_send_gen_cmd(0x98, 0x55, base, map, cfi, cfi->device_type, NULL); @@ -148,7 +149,8 @@ unsigned long devsize = (1<cfiq->DevSize) * cfi->interleave; mtd = kmalloc(sizeof(*mtd), GFP_KERNEL); - printk(KERN_NOTICE "number of %s chips: %d\n", (cfi->cfi_mode)?"CFI":"JEDEC",cfi->numchips); + printk(KERN_NOTICE "number of %s chips: %d\n", + (cfi->cfi_mode == CFI_MODE_CFI)?"CFI":"JEDEC",cfi->numchips); if (!mtd) { printk(KERN_WARNING "Failed to allocate memory for MTD device\n"); @@ -220,6 +222,9 @@ mtd->erase = cfi_amdstd_erase_varsize; else #endif + if (((cfi->cfiq->EraseRegionInfo[0] & 0xffff) + 1) == 1) + mtd->erase = cfi_amdstd_erase_chip; + else mtd->erase = cfi_amdstd_erase_onesize; mtd->read = cfi_amdstd_read; mtd->write = cfi_amdstd_write; @@ -232,6 +237,15 @@ return NULL; break; } + if (cfi->fast_prog) { + /* In cfi_amdstd_write() we frob the protection stuff + without paying any attention to the state machine. + This upsets in-progress erases. So we turn this flag + off for now till the code gets fixed. */ + printk(KERN_NOTICE "cfi_cmdset_0002: Disabling fast programming due to code brokenness.\n"); + cfi->fast_prog = 0; + } + mtd->sync = cfi_amdstd_sync; mtd->suspend = cfi_amdstd_suspend; mtd->resume = cfi_amdstd_resume; @@ -458,10 +472,12 @@ } } - /* Go into unlock bypass mode */ - cfi_send_gen_cmd(0xAA, cfi->addr_unlock1, chipstart, map, cfi, CFI_DEVICETYPE_X8, NULL); - cfi_send_gen_cmd(0x55, cfi->addr_unlock2, chipstart, map, cfi, CFI_DEVICETYPE_X8, NULL); - cfi_send_gen_cmd(0x20, cfi->addr_unlock1, chipstart, map, cfi, CFI_DEVICETYPE_X8, NULL); + if (cfi->fast_prog) { + /* Go into unlock bypass mode */ + cfi_send_gen_cmd(0xAA, cfi->addr_unlock1, chipstart, map, cfi, CFI_DEVICETYPE_X8, NULL); + cfi_send_gen_cmd(0x55, cfi->addr_unlock2, chipstart, map, cfi, CFI_DEVICETYPE_X8, NULL); + cfi_send_gen_cmd(0x20, cfi->addr_unlock1, chipstart, map, cfi, CFI_DEVICETYPE_X8, NULL); + } /* We are now aligned, write as much as possible */ while(len >= CFIDEV_BUSWIDTH) { @@ -547,12 +563,139 @@ return 0; } +static inline int do_erase_chip(struct map_info *map, struct flchip *chip) +{ + unsigned int oldstatus, status; + unsigned int dq6, dq5; + unsigned long timeo = jiffies + HZ; + unsigned int adr; + struct cfi_private *cfi = map->fldrv_priv; + DECLARE_WAITQUEUE(wait, current); + + retry: + cfi_spin_lock(chip->mutex); + + if (chip->state != FL_READY){ + set_current_state(TASK_UNINTERRUPTIBLE); + add_wait_queue(&chip->wq, &wait); + + cfi_spin_unlock(chip->mutex); + + schedule(); + remove_wait_queue(&chip->wq, &wait); +#if 0 + if(signal_pending(current)) + return -EINTR; +#endif + timeo = jiffies + HZ; + + goto retry; + } + + chip->state = FL_ERASING; + + /* Handle devices with one erase region, that only implement + * the chip erase command. + */ + ENABLE_VPP(map); + cfi_send_gen_cmd(0xAA, cfi->addr_unlock1, chip->start, map, cfi, CFI_DEVICETYPE_X8, NULL); + cfi_send_gen_cmd(0x55, cfi->addr_unlock2, chip->start, map, cfi, CFI_DEVICETYPE_X8, NULL); + cfi_send_gen_cmd(0x80, cfi->addr_unlock1, chip->start, map, cfi, CFI_DEVICETYPE_X8, NULL); + cfi_send_gen_cmd(0xAA, cfi->addr_unlock1, chip->start, map, cfi, CFI_DEVICETYPE_X8, NULL); + cfi_send_gen_cmd(0x55, cfi->addr_unlock2, chip->start, map, cfi, CFI_DEVICETYPE_X8, NULL); + cfi_send_gen_cmd(0x10, cfi->addr_unlock1, chip->start, map, cfi, CFI_DEVICETYPE_X8, NULL); + timeo = jiffies + (HZ*20); + adr = cfi->addr_unlock1; + + /* Wait for the end of programing/erasure by using the toggle method. + * As long as there is a programming procedure going on, bit 6 of the last + * written byte is toggling it's state with each consectuve read. + * The toggling stops as soon as the procedure is completed. + * + * If the process has gone on for too long on the chip bit 5 gets. + * After bit5 is set you can kill the operation by sending a reset + * command to the chip. + */ + dq6 = CMD(1<<6); + dq5 = CMD(1<<5); + + oldstatus = cfi_read(map, adr); + status = cfi_read(map, adr); + while( ((status & dq6) != (oldstatus & dq6)) && + ((status & dq5) != dq5) && + !time_after(jiffies, timeo)) { + int wait_reps; + + /* an initial short sleep */ + cfi_spin_unlock(chip->mutex); + schedule_timeout(HZ/100); + cfi_spin_lock(chip->mutex); + + if (chip->state != FL_ERASING) { + /* Someone's suspended the erase. Sleep */ + set_current_state(TASK_UNINTERRUPTIBLE); + add_wait_queue(&chip->wq, &wait); + + cfi_spin_unlock(chip->mutex); + printk("erase suspended. Sleeping\n"); + + schedule(); + remove_wait_queue(&chip->wq, &wait); +#if 0 + if (signal_pending(current)) + return -EINTR; +#endif + timeo = jiffies + (HZ*2); /* FIXME */ + cfi_spin_lock(chip->mutex); + continue; + } + + /* Busy wait for 1/10 of a milisecond */ + for(wait_reps = 0; + (wait_reps < 100) && + ((status & dq6) != (oldstatus & dq6)) && + ((status & dq5) != dq5); + wait_reps++) { + + /* Latency issues. Drop the lock, wait a while and retry */ + cfi_spin_unlock(chip->mutex); + + cfi_udelay(1); + + cfi_spin_lock(chip->mutex); + oldstatus = cfi_read(map, adr); + status = cfi_read(map, adr); + } + oldstatus = cfi_read(map, adr); + status = cfi_read(map, adr); + } + if ((status & dq6) != (oldstatus & dq6)) { + /* The erasing didn't stop?? */ + if ((status & dq5) == dq5) { + /* dq5 is active so we can do a reset and stop the erase */ + cfi_write(map, CMD(0xF0), chip->start); + } + chip->state = FL_READY; + wake_up(&chip->wq); + cfi_spin_unlock(chip->mutex); + printk("waiting for erase to complete timed out."); + DISABLE_VPP(map); + return -EIO; + } + DISABLE_VPP(map); + chip->state = FL_READY; + wake_up(&chip->wq); + cfi_spin_unlock(chip->mutex); + return 0; + +} + static inline int do_erase_oneblock(struct map_info *map, struct flchip *chip, unsigned long adr) { - unsigned int status; + unsigned int oldstatus, status; + unsigned int dq6, dq5; unsigned long timeo = jiffies + HZ; struct cfi_private *cfi = map->fldrv_priv; - unsigned int rdy_mask; DECLARE_WAITQUEUE(wait, current); retry: @@ -588,18 +731,30 @@ timeo = jiffies + (HZ*20); - cfi_spin_unlock(chip->mutex); - schedule_timeout(HZ); - cfi_spin_lock(chip->mutex); - - rdy_mask = CMD(0x80); - - /* FIXME. Use a timer to check this, and return immediately. */ - /* Once the state machine's known to be working I'll do that */ + /* Wait for the end of programing/erasure by using the toggle method. + * As long as there is a programming procedure going on, bit 6 of the last + * written byte is toggling it's state with each consectuve read. + * The toggling stops as soon as the procedure is completed. + * + * If the process has gone on for too long on the chip bit 5 gets. + * After bit5 is set you can kill the operation by sending a reset + * command to the chip. + */ + dq6 = CMD(1<<6); + dq5 = CMD(1<<5); - while ( ( (status = cfi_read(map,adr)) & rdy_mask ) != rdy_mask ) { - static int z=0; + oldstatus = cfi_read(map, adr); + status = cfi_read(map, adr); + while( ((status & dq6) != (oldstatus & dq6)) && + ((status & dq5) != dq5) && + !time_after(jiffies, timeo)) { + int wait_reps; + /* an initial short sleep */ + cfi_spin_unlock(chip->mutex); + schedule_timeout(HZ/100); + cfi_spin_lock(chip->mutex); + if (chip->state != FL_ERASING) { /* Someone's suspended the erase. Sleep */ set_current_state(TASK_UNINTERRUPTIBLE); @@ -619,29 +774,38 @@ continue; } - /* OK Still waiting */ - if (time_after(jiffies, timeo)) { - chip->state = FL_READY; + /* Busy wait for 1/10 of a milisecond */ + for(wait_reps = 0; + (wait_reps < 100) && + ((status & dq6) != (oldstatus & dq6)) && + ((status & dq5) != dq5); + wait_reps++) { + + /* Latency issues. Drop the lock, wait a while and retry */ cfi_spin_unlock(chip->mutex); - printk(KERN_WARNING "waiting for erase to complete timed out."); - DISABLE_VPP(map); - return -EIO; - } + + cfi_udelay(1); - /* Latency issues. Drop the lock, wait a while and retry */ + cfi_spin_lock(chip->mutex); + oldstatus = cfi_read(map, adr); + status = cfi_read(map, adr); + } + oldstatus = cfi_read(map, adr); + status = cfi_read(map, adr); + } + if ((status & dq6) != (oldstatus & dq6)) { + /* The erasing didn't stop?? */ + if ((status & dq5) == dq5) { + /* dq5 is active so we can do a reset and stop the erase */ + cfi_write(map, CMD(0xF0), chip->start); + } + chip->state = FL_READY; + wake_up(&chip->wq); cfi_spin_unlock(chip->mutex); - - z++; - if ( 0 && !(z % 100 )) - printk(KERN_WARNING "chip not ready yet after erase. looping\n"); - - cfi_udelay(1); - - cfi_spin_lock(chip->mutex); - continue; + printk("waiting for erase to complete timed out."); + DISABLE_VPP(map); + return -EIO; } - - /* Done and happy. */ DISABLE_VPP(map); chip->state = FL_READY; wake_up(&chip->wq); @@ -779,6 +943,29 @@ } } + instr->state = MTD_ERASE_DONE; + if (instr->callback) + instr->callback(instr); + + return 0; +} + +static int cfi_amdstd_erase_chip(struct mtd_info *mtd, struct erase_info *instr) +{ + struct map_info *map = mtd->priv; + struct cfi_private *cfi = map->fldrv_priv; + int ret = 0; + + if (instr->addr != 0) + return -EINVAL; + + if (instr->len != mtd->size) + return -EINVAL; + + ret = do_erase_chip(map, &cfi->chips[0]); + if (ret) + return ret; + instr->state = MTD_ERASE_DONE; if (instr->callback) instr->callback(instr); diff -urN orig/drivers/mtd/chips/cfi_probe.c linux/drivers/mtd/chips/cfi_probe.c --- orig/drivers/mtd/chips/cfi_probe.c Sun Oct 14 20:53:08 2001 +++ linux/drivers/mtd/chips/cfi_probe.c Fri Feb 21 15:17:33 2003 @@ -1,7 +1,7 @@ /* Common Flash Interface probe code. (C) 2000 Red Hat. GPL'd. - $Id: cfi_probe.c,v 1.66 2001/10/02 15:05:12 dwmw2 Exp $ + $Id: cfi_probe.c,v 1.69 2002/05/11 22:13:03 dwmw2 Exp $ */ #include @@ -24,16 +24,13 @@ static void print_cfi_ident(struct cfi_ident *); #endif -int cfi_jedec_setup(struct cfi_private *p_cfi, int index); -int cfi_jedec_lookup(int index, int mfr_id, int dev_id); - static int cfi_probe_chip(struct map_info *map, __u32 base, struct flchip *chips, struct cfi_private *cfi); static int cfi_chip_setup(struct map_info *map, struct cfi_private *cfi); struct mtd_info *cfi_probe(struct map_info *map); -/* check for QRY, or search for jedec id. +/* check for QRY. in: interleave,type,mode ret: table index, <0 for error */ @@ -55,7 +52,23 @@ { int i; + if ((base + 0) >= map->size) { + printk(KERN_NOTICE + "Probe at base[0x00](0x%08lx) past the end of the map(0x%08lx)\n", + (unsigned long)base, map->size -1); + return 0; + } + if ((base + 0xff) >= map->size) { + printk(KERN_NOTICE + "Probe at base[0x55](0x%08lx) past the end of the map(0x%08lx)\n", + (unsigned long)base + 0x55, map->size -1); + return 0; + } cfi_send_gen_cmd(0xF0, 0, base, map, cfi, cfi->device_type, NULL); + + /* some devices don't respond to 0xF0, so send 0xFF to be sure */ + cfi_send_gen_cmd(0xFF, 0, base, map, cfi, cfi->device_type, NULL); + cfi_send_gen_cmd(0x98, 0x55, base, map, cfi, cfi->device_type, NULL); if (!qry_present(map,base,cfi)) @@ -75,6 +88,8 @@ /* Eep. This chip also had the QRY marker. * Is it an alias for the new one? */ cfi_send_gen_cmd(0xF0, 0, chips[i].start, map, cfi, cfi->device_type, NULL); + /* some devices don't respond to 0xF0, so send 0xFF to be sure */ + cfi_send_gen_cmd(0xFF, 0, chips[i].start, map, cfi, cfi->device_type, NULL); /* If the QRY marker goes away, it's an alias */ if (!qry_present(map, chips[i].start, cfi)) { @@ -87,7 +102,8 @@ * too and if it's the same, assume it's an alias. */ /* FIXME: Use other modes to do a proper check */ cfi_send_gen_cmd(0xF0, 0, base, map, cfi, cfi->device_type, NULL); - + /* some devices don't respond to 0xF0, so send 0xFF to be sure */ + cfi_send_gen_cmd(0xFF, 0, base, map, cfi, cfi->device_type, NULL); if (qry_present(map, base, cfi)) { printk(KERN_DEBUG "%s: Found an alias at 0x%x for the chip at 0x%lx\n", map->name, base, chips[i].start); @@ -110,6 +126,10 @@ /* Put it back into Read Mode */ cfi_send_gen_cmd(0xF0, 0, base, map, cfi, cfi->device_type, NULL); + /* some devices don't respond to 0xF0, so send 0xFF to be sure */ + cfi_send_gen_cmd(0xFF, 0, base, map, cfi, cfi->device_type, NULL); + + printk(KERN_INFO "%s: Found %d x%d devices at 0x%x in %d-bit mode\n", map->name, cfi->interleave, cfi->device_type*8, base, map->buswidth*8); @@ -139,7 +159,7 @@ memset(cfi->cfiq,0,sizeof(struct cfi_ident)); - cfi->cfi_mode = 1; + cfi->cfi_mode = CFI_MODE_CFI; cfi->fast_prog=1; /* CFI supports fast programming */ /* Read the CFI info structure */ @@ -156,6 +176,20 @@ cfi->cfiq->InterfaceDesc = le16_to_cpu(cfi->cfiq->InterfaceDesc); cfi->cfiq->MaxBufWriteSize = le16_to_cpu(cfi->cfiq->MaxBufWriteSize); + /* + * ST screwed up the CFI interface for buffer writes on their parts, + * so this needs to be fixed up by hand here. + * + * A possible enhancment is that instead of just reverting back + * to word write (as this does), we could use the ST specific double + * word write instead. + */ + + if (cfi_read_query(map,base) == 0x20){ + cfi->cfiq->BufWriteTimeoutTyp = 0; + cfi->cfiq->BufWriteTimeoutMax = 0; + } + #ifdef DEBUG_CFI /* Dump the information therein */ print_cfi_ident(cfi->cfiq); @@ -173,6 +207,9 @@ /* Put it back into Read Mode */ cfi_send_gen_cmd(0xF0, 0, base, map, cfi, cfi->device_type, NULL); + /* some devices don't respond to 0xF0, so send 0xFF to be sure */ + cfi_send_gen_cmd(0xFF, 0, base, map, cfi, cfi->device_type, NULL); + return 1; } @@ -250,11 +287,11 @@ else printk("Full buffer write not supported\n"); - printk("Typical block erase timeout: %d ĩs\n", 1<BlockEraseTimeoutTyp); - printk("Maximum block erase timeout: %d ĩs\n", (1<BlockEraseTimeoutMax) * (1<BlockEraseTimeoutTyp)); + printk("Typical block erase timeout: %d ms\n", 1<BlockEraseTimeoutTyp); + printk("Maximum block erase timeout: %d ms\n", (1<BlockEraseTimeoutMax) * (1<BlockEraseTimeoutTyp)); if (cfip->ChipEraseTimeoutTyp || cfip->ChipEraseTimeoutMax) { - printk("Typical chip erase timeout: %d ĩs\n", 1<ChipEraseTimeoutTyp); - printk("Maximum chip erase timeout: %d ĩs\n", (1<ChipEraseTimeoutMax) * (1<ChipEraseTimeoutTyp)); + printk("Typical chip erase timeout: %d ms\n", 1<ChipEraseTimeoutTyp); + printk("Maximum chip erase timeout: %d ms\n", (1<ChipEraseTimeoutMax) * (1<ChipEraseTimeoutTyp)); } else printk("Chip erase not supported\n"); diff -urN orig/drivers/mtd/chips/chipreg.c linux/drivers/mtd/chips/chipreg.c --- orig/drivers/mtd/chips/chipreg.c Sun Oct 14 20:53:08 2001 +++ linux/drivers/mtd/chips/chipreg.c Fri Mar 1 10:38:39 2002 @@ -1,5 +1,5 @@ /* - * $Id: chipreg.c,v 1.12 2001/10/02 15:29:53 dwmw2 Exp $ + * $Id: chipreg.c,v 1.13 2002/02/21 08:26:58 dwmw2 Exp $ * * Registration for chip drivers * @@ -29,7 +29,7 @@ spin_unlock(&chip_drvs_lock); } -static struct mtd_chip_driver *get_mtd_chip_driver (char *name) +static struct mtd_chip_driver *get_mtd_chip_driver (const char *name) { struct list_head *pos; struct mtd_chip_driver *ret = NULL, *this; @@ -57,7 +57,7 @@ /* Hide all the horrid details, like some silly person taking get_module_symbol() away from us, from the caller. */ -struct mtd_info *do_map_probe(char *name, struct map_info *map) +struct mtd_info *do_map_probe(const char *name, struct map_info *map) { struct mtd_chip_driver *drv; struct mtd_info *ret; diff -urN orig/drivers/mtd/chips/gen_probe.c linux/drivers/mtd/chips/gen_probe.c --- orig/drivers/mtd/chips/gen_probe.c Sun Oct 14 20:53:09 2001 +++ linux/drivers/mtd/chips/gen_probe.c Fri Mar 1 10:38:39 2002 @@ -2,7 +2,7 @@ * Routines common to all CFI-type probes. * (C) 2001, 2001 Red Hat, Inc. * GPL'd - * $Id: gen_probe.c,v 1.5 2001/10/02 15:05:12 dwmw2 Exp $ + * $Id: gen_probe.c,v 1.7 2002/01/30 09:08:31 rkaiser Exp $ */ #include @@ -38,7 +38,7 @@ if (mtd) return mtd; - printk(KERN_WARNING"cfi_probe: No supported Vendor Command Set found\n"); + printk(KERN_WARNING"gen_probe: No supported Vendor Command Set found\n"); kfree(cfi->cfiq); kfree(cfi); @@ -106,6 +106,12 @@ * Now probe for other chips, checking sensibly for aliases while * we're at it. The new_chip probe above should have let the first * chip in read mode. + * + * NOTE: Here, we're checking if there is room for another chip + * the same size within the mapping. Therefore, + * base + chipsize <= map->size is the correct thing to do, + * because, base + chipsize would be the _first_ byte of the + * next chip, not the one we're currently pondering. */ for (base = (1<size; diff -urN orig/drivers/mtd/chips/jedec_probe.c linux/drivers/mtd/chips/jedec_probe.c --- orig/drivers/mtd/chips/jedec_probe.c Sun Oct 14 20:53:09 2001 +++ linux/drivers/mtd/chips/jedec_probe.c Thu Jun 20 20:04:19 2002 @@ -1,7 +1,9 @@ /* Common Flash Interface probe code. (C) 2000 Red Hat. GPL'd. - $Id: jedec_probe.c,v 1.3 2001/10/02 15:05:12 dwmw2 Exp $ + $Id: jedec_probe.c,v 1.16 2002/06/07 11:20:52 spse Exp $ + See JEDEC (http://www.jedec.org/) standard JESD21C (section 3.5) + for the standard this probe goes back to. */ #include @@ -18,15 +20,17 @@ #include #include - /* Manufacturers */ #define MANUFACTURER_AMD 0x0001 -#define MANUFACTURER_FUJITSU 0x0004 #define MANUFACTURER_ATMEL 0x001f +#define MANUFACTURER_FUJITSU 0x0004 +#define MANUFACTURER_INTEL 0x0089 +#define MANUFACTURER_MACRONIX 0x00C2 #define MANUFACTURER_ST 0x0020 #define MANUFACTURER_SST 0x00BF #define MANUFACTURER_TOSHIBA 0x0098 + /* AMD */ #define AM29F800BB 0x2258 #define AM29F800BT 0x22D6 @@ -34,27 +38,77 @@ #define AM29LV800BT 0x22DA #define AM29LV160DT 0x22C4 #define AM29LV160DB 0x2249 +#define AM29F017D 0x003D +#define AM29F016 0x00AD +#define AM29F080 0x00D5 +#define AM29F040 0x00A4 /* Atmel */ -#define AT49BV16X4 0x00c0 -#define AT49BV16X4T 0x00c2 +#define AT49BV512 0x0003 +#define AT49BV16X 0x00C0 +#define AT49BV16XT 0x00C2 +#define AT49BV32X 0x00C8 +#define AT49BV32XT 0x00C9 /* Fujitsu */ +#define MBM29LV650UE 0x22D7 +#define MBM29LV320TE 0x22F6 +#define MBM29LV320BE 0x22F9 #define MBM29LV160TE 0x22C4 #define MBM29LV160BE 0x2249 +#define MBM29LV800BA 0x225B +#define MBM29LV800TA 0x22DA + +/* Intel */ +#define I28F004B3T 0x00d4 +#define I28F004B3B 0x00d5 +#define I28F400B3T 0x8894 +#define I28F400B3B 0x8895 +#define I28F008S5 0x00a6 +#define I28F016S5 0x00a0 +#define I28F008SA 0x00a2 +#define I28F008B3T 0x00d2 +#define I28F008B3B 0x00d3 +#define I28F800B3T 0x8892 +#define I28F800B3B 0x8893 +#define I28F016S3 0x00aa +#define I28F016B3T 0x00d0 +#define I28F016B3B 0x00d1 +#define I28F160B3T 0x8890 +#define I28F160B3B 0x8891 +#define I28F320B3T 0x8896 +#define I28F320B3B 0x8897 +#define I28F640B3T 0x8898 +#define I28F640B3B 0x8899 +#define I82802AB 0x00ad +#define I82802AC 0x00ac + +/* Macronix */ +#define MX29LV160T 0x22C4 +#define MX29LV160B 0x2249 +#define MX29F016 0x00AD +#define MX29F004T 0x0045 +#define MX29F004B 0x0046 /* ST - www.st.com */ #define M29W800T 0x00D7 #define M29W160DT 0x22C4 #define M29W160DB 0x2249 +#define M29W040B 0x00E3 /* SST */ #define SST39LF800 0x2781 #define SST39LF160 0x2782 +#define SST39SF010A 0x00B5 +#define SST39SF020A 0x00B6 /* Toshiba */ #define TC58FVT160 0x00C2 #define TC58FVB160 0x0043 +#define TC58FVT321 0x009A +#define TC58FVB321 0x009C +#define TC58FVT641 0x0093 +#define TC58FVB641 0x0095 struct amd_flash_info { @@ -64,14 +118,20 @@ const int DevSize; const int InterfaceDesc; const int NumEraseRegions; + const int CmdSet; const ulong regions[4]; }; #define ERASEINFO(size,blocks) (size<<8)|(blocks-1) -#define SIZE_1MiB 20 -#define SIZE_2MiB 21 -#define SIZE_4MiB 22 +#define SIZE_64KiB 16 +#define SIZE_128KiB 17 +#define SIZE_256KiB 18 +#define SIZE_512KiB 19 +#define SIZE_1MiB 20 +#define SIZE_2MiB 21 +#define SIZE_4MiB 22 +#define SIZE_8MiB 23 static const struct amd_flash_info jedec_table[] = { { @@ -79,6 +139,7 @@ dev_id: AM29LV160DT, name: "AMD AM29LV160DT", DevSize: SIZE_2MiB, + CmdSet: P_ID_AMD_STD, NumEraseRegions: 4, regions: {ERASEINFO(0x10000,31), ERASEINFO(0x08000,1), @@ -90,6 +151,7 @@ dev_id: AM29LV160DB, name: "AMD AM29LV160DB", DevSize: SIZE_2MiB, + CmdSet: P_ID_AMD_STD, NumEraseRegions: 4, regions: {ERASEINFO(0x04000,1), ERASEINFO(0x02000,2), @@ -101,6 +163,7 @@ dev_id: TC58FVT160, name: "Toshiba TC58FVT160", DevSize: SIZE_2MiB, + CmdSet: P_ID_AMD_STD, NumEraseRegions: 4, regions: {ERASEINFO(0x10000,31), ERASEINFO(0x08000,1), @@ -108,10 +171,92 @@ ERASEINFO(0x04000,1) } }, { + mfr_id: MANUFACTURER_TOSHIBA, + dev_id: TC58FVB160, + name: "Toshiba TC58FVB160", + DevSize: SIZE_2MiB, + CmdSet: P_ID_AMD_STD, + NumEraseRegions: 4, + regions: {ERASEINFO(0x04000,1), + ERASEINFO(0x02000,2), + ERASEINFO(0x08000,1), + ERASEINFO(0x10000,31) + } + }, { + mfr_id: MANUFACTURER_TOSHIBA, + dev_id: TC58FVB321, + name: "Toshiba TC58FVB321", + DevSize: SIZE_4MiB, + CmdSet: P_ID_AMD_STD, + NumEraseRegions: 4, + regions: {ERASEINFO(0x02000,8), + ERASEINFO(0x10000,63) + } + }, { + mfr_id: MANUFACTURER_TOSHIBA, + dev_id: TC58FVT321, + name: "Toshiba TC58FVT321", + DevSize: SIZE_4MiB, + CmdSet: P_ID_AMD_STD, + NumEraseRegions: 4, + regions: {ERASEINFO(0x10000,63), + ERASEINFO(0x02000,8) + } + }, { + mfr_id: MANUFACTURER_TOSHIBA, + dev_id: TC58FVB641, + name: "Toshiba TC58FVB641", + DevSize: SIZE_8MiB, + CmdSet: P_ID_AMD_STD, + NumEraseRegions: 4, + regions: {ERASEINFO(0x02000,8), + ERASEINFO(0x10000,127) + } + }, { + mfr_id: MANUFACTURER_TOSHIBA, + dev_id: TC58FVT641, + name: "Toshiba TC58FVT641", + DevSize: SIZE_8MiB, + CmdSet: P_ID_AMD_STD, + NumEraseRegions: 4, + regions: {ERASEINFO(0x10000,127), + ERASEINFO(0x02000,8) + } + }, { + mfr_id: MANUFACTURER_FUJITSU, + dev_id: MBM29LV650UE, + name: "Fujitsu MBM29LV650UE", + DevSize: SIZE_8MiB, + CmdSet: P_ID_AMD_STD, + NumEraseRegions: 1, + regions: {ERASEINFO(0x10000,128) + } + }, { + mfr_id: MANUFACTURER_FUJITSU, + dev_id: MBM29LV320TE, + name: "Fujitsu MBM29LV320TE", + DevSize: SIZE_4MiB, + CmdSet: P_ID_AMD_STD, + NumEraseRegions: 2, + regions: {ERASEINFO(0x10000,63), + ERASEINFO(0x02000,8) + } + }, { + mfr_id: MANUFACTURER_FUJITSU, + dev_id: MBM29LV320BE, + name: "Fujitsu MBM29LV320BE", + DevSize: SIZE_4MiB, + CmdSet: P_ID_AMD_STD, + NumEraseRegions: 2, + regions: {ERASEINFO(0x02000,8), + ERASEINFO(0x10000,63) + } + }, { mfr_id: MANUFACTURER_FUJITSU, dev_id: MBM29LV160TE, name: "Fujitsu MBM29LV160TE", DevSize: SIZE_2MiB, + CmdSet: P_ID_AMD_STD, NumEraseRegions: 4, regions: {ERASEINFO(0x10000,31), ERASEINFO(0x08000,1), @@ -119,10 +264,11 @@ ERASEINFO(0x04000,1) } }, { - mfr_id: MANUFACTURER_TOSHIBA, - dev_id: TC58FVB160, - name: "Toshiba TC58FVB160", + mfr_id: MANUFACTURER_FUJITSU, + dev_id: MBM29LV160BE, + name: "Fujitsu MBM29LV160BE", DevSize: SIZE_2MiB, + CmdSet: P_ID_AMD_STD, NumEraseRegions: 4, regions: {ERASEINFO(0x04000,1), ERASEINFO(0x02000,2), @@ -131,20 +277,34 @@ } }, { mfr_id: MANUFACTURER_FUJITSU, - dev_id: MBM29LV160BE, - name: "Fujitsu MBM29LV160BE", - DevSize: SIZE_2MiB, + dev_id: MBM29LV800BA, + name: "Fujitsu MBM29LV800BA", + DevSize: SIZE_1MiB, + CmdSet: P_ID_AMD_STD, NumEraseRegions: 4, regions: {ERASEINFO(0x04000,1), ERASEINFO(0x02000,2), ERASEINFO(0x08000,1), - ERASEINFO(0x10000,31) + ERASEINFO(0x10000,15) + } + }, { + mfr_id: MANUFACTURER_FUJITSU, + dev_id: MBM29LV800TA, + name: "Fujitsu MBM29LV800TA", + DevSize: SIZE_1MiB, + CmdSet: P_ID_AMD_STD, + NumEraseRegions: 4, + regions: {ERASEINFO(0x10000,15), + ERASEINFO(0x08000,1), + ERASEINFO(0x02000,2), + ERASEINFO(0x04000,1) } }, { mfr_id: MANUFACTURER_AMD, dev_id: AM29LV800BB, name: "AMD AM29LV800BB", DevSize: SIZE_1MiB, + CmdSet: P_ID_AMD_STD, NumEraseRegions: 4, regions: {ERASEINFO(0x04000,1), ERASEINFO(0x02000,2), @@ -156,6 +316,7 @@ dev_id: AM29F800BB, name: "AMD AM29F800BB", DevSize: SIZE_1MiB, + CmdSet: P_ID_AMD_STD, NumEraseRegions: 4, regions: {ERASEINFO(0x04000,1), ERASEINFO(0x02000,2), @@ -167,6 +328,7 @@ dev_id: AM29LV800BT, name: "AMD AM29LV800BT", DevSize: SIZE_1MiB, + CmdSet: P_ID_AMD_STD, NumEraseRegions: 4, regions: {ERASEINFO(0x10000,15), ERASEINFO(0x08000,1), @@ -178,6 +340,7 @@ dev_id: AM29F800BT, name: "AMD AM29F800BT", DevSize: SIZE_1MiB, + CmdSet: P_ID_AMD_STD, NumEraseRegions: 4, regions: {ERASEINFO(0x10000,15), ERASEINFO(0x08000,1), @@ -189,6 +352,7 @@ dev_id: AM29LV800BB, name: "AMD AM29LV800BB", DevSize: SIZE_1MiB, + CmdSet: P_ID_AMD_STD, NumEraseRegions: 4, regions: {ERASEINFO(0x10000,15), ERASEINFO(0x08000,1), @@ -196,10 +360,243 @@ ERASEINFO(0x04000,1) } }, { + mfr_id: MANUFACTURER_INTEL, + dev_id: I28F004B3B, + name: "Intel 28F004B3B", + DevSize: SIZE_512KiB, + CmdSet: P_ID_INTEL_STD, + NumEraseRegions: 2, + regions: { + ERASEINFO(0x02000, 8), + ERASEINFO(0x10000, 7), + } + }, { + mfr_id: MANUFACTURER_INTEL, + dev_id: I28F004B3T, + name: "Intel 28F004B3T", + DevSize: SIZE_512KiB, + CmdSet: P_ID_INTEL_STD, + NumEraseRegions: 2, + regions: { + ERASEINFO(0x10000, 7), + ERASEINFO(0x02000, 8), + } + }, { + mfr_id: MANUFACTURER_INTEL, + dev_id: I28F400B3B, + name: "Intel 28F400B3B", + DevSize: SIZE_512KiB, + CmdSet: P_ID_INTEL_STD, + NumEraseRegions: 2, + regions: { + ERASEINFO(0x02000, 8), + ERASEINFO(0x10000, 7), + } + }, { + mfr_id: MANUFACTURER_INTEL, + dev_id: I28F400B3T, + name: "Intel 28F400B3T", + DevSize: SIZE_512KiB, + CmdSet: P_ID_INTEL_STD, + NumEraseRegions: 2, + regions: { + ERASEINFO(0x10000, 7), + ERASEINFO(0x02000, 8), + } + }, { + mfr_id: MANUFACTURER_INTEL, + dev_id: I28F008B3B, + name: "Intel 28F008B3B", + DevSize: SIZE_1MiB, + CmdSet: P_ID_INTEL_STD, + NumEraseRegions: 2, + regions: { + ERASEINFO(0x02000, 8), + ERASEINFO(0x10000, 15), + } + }, { + mfr_id: MANUFACTURER_INTEL, + dev_id: I28F008B3T, + name: "Intel 28F008B3T", + DevSize: SIZE_1MiB, + CmdSet: P_ID_INTEL_STD, + NumEraseRegions: 2, + regions: { + ERASEINFO(0x10000, 15), + ERASEINFO(0x02000, 8), + } + }, { + mfr_id: MANUFACTURER_INTEL, + dev_id: I28F008S5, + name: "Intel 28F008S5", + DevSize: SIZE_1MiB, + CmdSet: P_ID_INTEL_EXT, + NumEraseRegions: 1, + regions: {ERASEINFO(0x10000,16), + } + }, { + mfr_id: MANUFACTURER_INTEL, + dev_id: I28F016S5, + name: "Intel 28F016S5", + DevSize: SIZE_2MiB, + CmdSet: P_ID_INTEL_EXT, + NumEraseRegions: 1, + regions: {ERASEINFO(0x10000,32), + } + }, { + mfr_id: MANUFACTURER_INTEL, + dev_id: I28F008SA, + name: "Intel 28F008SA", + DevSize: SIZE_1MiB, + CmdSet: P_ID_INTEL_STD, + NumEraseRegions: 1, + regions: { + ERASEINFO(0x10000, 16), + } + }, { + mfr_id: MANUFACTURER_INTEL, + dev_id: I28F800B3B, + name: "Intel 28F800B3B", + DevSize: SIZE_1MiB, + CmdSet: P_ID_INTEL_STD, + NumEraseRegions: 2, + regions: { + ERASEINFO(0x02000, 8), + ERASEINFO(0x10000, 15), + } + }, { + mfr_id: MANUFACTURER_INTEL, + dev_id: I28F800B3T, + name: "Intel 28F800B3T", + DevSize: SIZE_1MiB, + CmdSet: P_ID_INTEL_STD, + NumEraseRegions: 2, + regions: { + ERASEINFO(0x10000, 15), + ERASEINFO(0x02000, 8), + } + }, { + mfr_id: MANUFACTURER_INTEL, + dev_id: I28F016B3B, + name: "Intel 28F016B3B", + DevSize: SIZE_2MiB, + CmdSet: P_ID_INTEL_STD, + NumEraseRegions: 2, + regions: { + ERASEINFO(0x02000, 8), + ERASEINFO(0x10000, 31), + } + }, { + mfr_id: MANUFACTURER_INTEL, + dev_id: I28F016S3, + name: "Intel I28F016S3", + DevSize: SIZE_2MiB, + CmdSet: P_ID_INTEL_STD, + NumEraseRegions: 1, + regions: { + ERASEINFO(0x10000, 32), + } + }, { + mfr_id: MANUFACTURER_INTEL, + dev_id: I28F016B3T, + name: "Intel 28F016B3T", + DevSize: SIZE_2MiB, + CmdSet: P_ID_INTEL_STD, + NumEraseRegions: 2, + regions: { + ERASEINFO(0x10000, 31), + ERASEINFO(0x02000, 8), + } + }, { + mfr_id: MANUFACTURER_INTEL, + dev_id: I28F160B3B, + name: "Intel 28F160B3B", + DevSize: SIZE_2MiB, + CmdSet: P_ID_INTEL_STD, + NumEraseRegions: 2, + regions: { + ERASEINFO(0x02000, 8), + ERASEINFO(0x10000, 31), + } + }, { + mfr_id: MANUFACTURER_INTEL, + dev_id: I28F160B3T, + name: "Intel 28F160B3T", + DevSize: SIZE_2MiB, + CmdSet: P_ID_INTEL_STD, + NumEraseRegions: 2, + regions: { + ERASEINFO(0x10000, 31), + ERASEINFO(0x02000, 8), + } + }, { + mfr_id: MANUFACTURER_INTEL, + dev_id: I28F320B3B, + name: "Intel 28F320B3B", + DevSize: SIZE_4MiB, + CmdSet: P_ID_INTEL_STD, + NumEraseRegions: 2, + regions: { + ERASEINFO(0x02000, 8), + ERASEINFO(0x10000, 63), + } + }, { + mfr_id: MANUFACTURER_INTEL, + dev_id: I28F320B3T, + name: "Intel 28F320B3T", + DevSize: SIZE_4MiB, + CmdSet: P_ID_INTEL_STD, + NumEraseRegions: 2, + regions: { + ERASEINFO(0x10000, 63), + ERASEINFO(0x02000, 8), + } + }, { + mfr_id: MANUFACTURER_INTEL, + dev_id: I28F640B3B, + name: "Intel 28F640B3B", + DevSize: SIZE_8MiB, + CmdSet: P_ID_INTEL_STD, + NumEraseRegions: 2, + regions: { + ERASEINFO(0x02000, 8), + ERASEINFO(0x10000, 127), + } + }, { + mfr_id: MANUFACTURER_INTEL, + dev_id: I28F640B3T, + name: "Intel 28F640B3T", + DevSize: SIZE_8MiB, + CmdSet: P_ID_INTEL_STD, + NumEraseRegions: 2, + regions: { + ERASEINFO(0x10000, 127), + ERASEINFO(0x02000, 8), + } + }, { + mfr_id: MANUFACTURER_INTEL, + dev_id: I82802AB, + name: "Intel 82802AB", + DevSize: SIZE_512KiB, + CmdSet: P_ID_INTEL_EXT, + NumEraseRegions: 1, + regions: {ERASEINFO(0x10000,8), + } + }, { + mfr_id: MANUFACTURER_INTEL, + dev_id: I82802AC, + name: "Intel 82802AC", + DevSize: SIZE_1MiB, + CmdSet: P_ID_INTEL_EXT, + NumEraseRegions: 1, + regions: {ERASEINFO(0x10000,16), + } + }, { mfr_id: MANUFACTURER_ST, dev_id: M29W800T, name: "ST M29W800T", DevSize: SIZE_1MiB, + CmdSet: P_ID_AMD_STD, NumEraseRegions: 4, regions: {ERASEINFO(0x10000,15), ERASEINFO(0x08000,1), @@ -211,6 +608,7 @@ dev_id: M29W160DT, name: "ST M29W160DT", DevSize: SIZE_2MiB, + CmdSet: P_ID_AMD_STD, NumEraseRegions: 4, regions: {ERASEINFO(0x10000,31), ERASEINFO(0x08000,1), @@ -222,6 +620,7 @@ dev_id: M29W160DB, name: "ST M29W160DB", DevSize: SIZE_2MiB, + CmdSet: P_ID_AMD_STD, NumEraseRegions: 4, regions: {ERASEINFO(0x04000,1), ERASEINFO(0x02000,2), @@ -230,24 +629,173 @@ } }, { mfr_id: MANUFACTURER_ATMEL, - dev_id: AT49BV16X4, - name: "Atmel AT49BV16X4", + dev_id: AT49BV512, + name: "Atmel AT49BV512", + DevSize: SIZE_64KiB, + CmdSet: P_ID_AMD_STD, + NumEraseRegions: 1, + regions: {ERASEINFO(0x10000,1) + } + }, { + mfr_id: MANUFACTURER_ATMEL, + dev_id: AT49BV16X, + name: "Atmel AT49BV16X", DevSize: SIZE_2MiB, - NumEraseRegions: 3, + CmdSet: P_ID_AMD_STD, + NumEraseRegions: 2, regions: {ERASEINFO(0x02000,8), - ERASEINFO(0x08000,2), - ERASEINFO(0x10000,30) + ERASEINFO(0x10000,31) } }, { - mfr_id: MANUFACTURER_ATMEL, - dev_id: AT49BV16X4T, - name: "Atmel AT49BV16X4T", - DevSize: SIZE_2MiB, - NumEraseRegions: 3, - regions: {ERASEINFO(0x10000,30), - ERASEINFO(0x08000,2), + mfr_id: MANUFACTURER_ATMEL, + dev_id: AT49BV16XT, + name: "Atmel AT49BV16XT", + DevSize: SIZE_2MiB, + CmdSet: P_ID_AMD_STD, + NumEraseRegions: 2, + regions: {ERASEINFO(0x10000,31), ERASEINFO(0x02000,8) - } + } + }, { + mfr_id: MANUFACTURER_ATMEL, + dev_id: AT49BV32X, + name: "Atmel AT49BV32X", + DevSize: SIZE_4MiB, + CmdSet: P_ID_AMD_STD, + NumEraseRegions: 2, + regions: {ERASEINFO(0x02000,8), + ERASEINFO(0x10000,63) + } + }, { + mfr_id: MANUFACTURER_ATMEL, + dev_id: AT49BV32XT, + name: "Atmel AT49BV32XT", + DevSize: SIZE_4MiB, + CmdSet: P_ID_AMD_STD, + NumEraseRegions: 2, + regions: {ERASEINFO(0x10000,63), + ERASEINFO(0x02000,8) + } + }, { + mfr_id: MANUFACTURER_AMD, + dev_id: AM29F017D, + name: "AMD AM29F017D", + DevSize: SIZE_2MiB, + CmdSet: P_ID_AMD_STD, + NumEraseRegions: 1, + regions: {ERASEINFO(0x10000,32), + } + }, { + mfr_id: MANUFACTURER_AMD, + dev_id: AM29F016, + name: "AMD AM29F016", + DevSize: SIZE_2MiB, + CmdSet: P_ID_AMD_STD, + NumEraseRegions: 1, + regions: {ERASEINFO(0x10000,32), + } + }, { + mfr_id: MANUFACTURER_AMD, + dev_id: AM29F080, + name: "AMD AM29F080", + DevSize: SIZE_1MiB, + CmdSet: P_ID_AMD_STD, + NumEraseRegions: 1, + regions: {ERASEINFO(0x10000,16), + } + }, { + mfr_id: MANUFACTURER_AMD, + dev_id: AM29F040, + name: "AMD AM29F040", + DevSize: SIZE_512KiB, + CmdSet: P_ID_AMD_STD, + NumEraseRegions: 1, + regions: {ERASEINFO(0x10000,8), + } + }, { + mfr_id: MANUFACTURER_ST, + dev_id: M29W040B, + name: "ST M29W040B", + DevSize: SIZE_512KiB, + CmdSet: P_ID_AMD_STD, + NumEraseRegions: 1, + regions: {ERASEINFO(0x10000,8), + } + }, { + mfr_id: MANUFACTURER_MACRONIX, + dev_id: MX29LV160T, + name: "MXIC MX29LV160T", + DevSize: SIZE_2MiB, + CmdSet: P_ID_AMD_STD, + NumEraseRegions: 4, + regions: {ERASEINFO(0x10000,31), + ERASEINFO(0x08000,1), + ERASEINFO(0x02000,2), + ERASEINFO(0x04000,1) + } + }, { + mfr_id: MANUFACTURER_MACRONIX, + dev_id: MX29LV160B, + name: "MXIC MX29LV160B", + DevSize: SIZE_2MiB, + CmdSet: P_ID_AMD_STD, + NumEraseRegions: 4, + regions: {ERASEINFO(0x04000,1), + ERASEINFO(0x02000,2), + ERASEINFO(0x08000,1), + ERASEINFO(0x10000,31) + } + }, { + mfr_id: MANUFACTURER_MACRONIX, + dev_id: MX29F016, + name: "Macronix MX29F016", + DevSize: SIZE_2MiB, + CmdSet: P_ID_AMD_STD, + NumEraseRegions: 1, + regions: {ERASEINFO(0x10000,32), + } + }, { + mfr_id: MANUFACTURER_MACRONIX, + dev_id: MX29F004T, + name: "Macronix MX29F004T", + DevSize: SIZE_512KiB, + CmdSet: P_ID_AMD_STD, + NumEraseRegions: 4, + regions: {ERASEINFO(0x10000,7), + ERASEINFO(0x08000,1), + ERASEINFO(0x02000,2), + ERASEINFO(0x04000,1), + } + }, { + mfr_id: MANUFACTURER_MACRONIX, + dev_id: MX29F004B, + name: "Macronix MX29F004B", + DevSize: SIZE_512KiB, + CmdSet: P_ID_AMD_STD, + NumEraseRegions: 4, + regions: {ERASEINFO(0x04000,1), + ERASEINFO(0x02000,2), + ERASEINFO(0x08000,1), + ERASEINFO(0x10000,7), + } + }, { + mfr_id: MANUFACTURER_SST, + dev_id: SST39SF010A, + name: "SST 39SF010A", + DevSize: SIZE_128KiB, + CmdSet: P_ID_AMD_STD, + NumEraseRegions: 1, + regions: {ERASEINFO(0x01000,32), + } + }, { + mfr_id: MANUFACTURER_SST, + dev_id: SST39SF020A, + name: "SST 39SF020A", + DevSize: SIZE_256KiB, + CmdSet: P_ID_AMD_STD, + NumEraseRegions: 1, + regions: {ERASEINFO(0x01000,64), + } } }; @@ -258,9 +806,42 @@ struct flchip *chips, struct cfi_private *cfi); struct mtd_info *jedec_probe(struct map_info *map); -#define jedec_read_mfr(map, base, osf) cfi_read(map, base) -#define jedec_read_id(map, base, osf) cfi_read(map, (base)+(osf)) +static inline u32 jedec_read_mfr(struct map_info *map, __u32 base, + struct cfi_private *cfi) +{ + u32 result, mask; + mask = (1 << (cfi->device_type * 8)) -1; + result = cfi_read(map, base); + result &= mask; + return result; +} + +static inline u32 jedec_read_id(struct map_info *map, __u32 base, + struct cfi_private *cfi) +{ + int osf; + u32 result, mask; + osf = cfi->interleave *cfi->device_type; + mask = (1 << (cfi->device_type * 8)) -1; + result = cfi_read(map, base + osf); + result &= mask; + return result; +} + +static inline void jedec_reset(u32 base, struct map_info *map, + struct cfi_private *cfi) +{ + /* Reset */ + cfi_send_gen_cmd(0xF0, 0, base, map, cfi, cfi->device_type, NULL); + /* Some misdesigned intel chips do not respond for 0xF0 for a reset, + * so ensure we're in read mode. Send both the Intel and the AMD command + * for this. Intel uses 0xff for this, AMD uses 0xff for NOP, so + * this should be safe. + */ + cfi_send_gen_cmd(0xFF, 0, base, map, cfi, cfi->device_type, NULL); + +} static int cfi_jedec_setup(struct cfi_private *p_cfi, int index) { int i,num_erase_regions; @@ -276,14 +857,16 @@ } memset(p_cfi->cfiq,0,sizeof(struct cfi_ident)); - - p_cfi->cfiq->P_ID = P_ID_AMD_STD; + + p_cfi->cfiq->P_ID = jedec_table[index].CmdSet; p_cfi->cfiq->NumEraseRegions = jedec_table[index].NumEraseRegions; p_cfi->cfiq->DevSize = jedec_table[index].DevSize; + p_cfi->cfi_mode = CFI_MODE_JEDEC; for (i=0; icfiq->EraseRegionInfo[i] = jedec_table[index].regions[i]; - } + } + p_cfi->cmdset_priv = 0; return 1; /* ok */ } @@ -291,8 +874,7 @@ struct flchip *chips, struct cfi_private *cfi) { int i; - int osf = cfi->interleave * cfi->device_type; - int retried = 0; + int unlockpass = 0; if (!cfi->numchips) { switch (cfi->device_type) { @@ -320,48 +902,94 @@ } retry: + /* Make certain we aren't probing past the end of map */ + if (base >= map->size) { + printk(KERN_NOTICE + "Probe at base(0x%08x) past the end of the map(0x%08lx)\n", + base, map->size -1); + return 0; + + } + if ((base + cfi->addr_unlock1) >= map->size) { + printk(KERN_NOTICE + "Probe at addr_unlock1(0x%08x + 0x%08x) past the end of the map(0x%08lx)\n", + base, cfi->addr_unlock1, map->size -1); + + return 0; + } + if ((base + cfi->addr_unlock2) >= map->size) { + printk(KERN_NOTICE + "Probe at addr_unlock2(0x%08x + 0x%08x) past the end of the map(0x%08lx)\n", + base, cfi->addr_unlock2, map->size -1); + return 0; + + } + /* Reset */ - cfi_send_gen_cmd(0xF0, 0, base, map, cfi, cfi->device_type, NULL); + jedec_reset(base, map, cfi); /* Autoselect Mode */ - cfi_send_gen_cmd(0xaa, cfi->addr_unlock1, base, map, cfi, CFI_DEVICETYPE_X8, NULL); - cfi_send_gen_cmd(0x55, cfi->addr_unlock2, base, map, cfi, CFI_DEVICETYPE_X8, NULL); + if(cfi->addr_unlock1) { + cfi_send_gen_cmd(0xaa, cfi->addr_unlock1, base, map, cfi, CFI_DEVICETYPE_X8, NULL); + cfi_send_gen_cmd(0x55, cfi->addr_unlock2, base, map, cfi, CFI_DEVICETYPE_X8, NULL); + } cfi_send_gen_cmd(0x90, cfi->addr_unlock1, base, map, cfi, CFI_DEVICETYPE_X8, NULL); if (!cfi->numchips) { /* This is the first time we're called. Set up the CFI stuff accordingly and return */ - cfi->mfr = jedec_read_mfr(map, base, osf); - cfi->id = jedec_read_id(map, base, osf); - + cfi->mfr = jedec_read_mfr(map, base, cfi); + cfi->id = jedec_read_id(map, base, cfi); + printk(KERN_INFO "Search for id:(%02x %02x) interleave(%d) type(%d)\n", + cfi->mfr, cfi->id, cfi->interleave, cfi->device_type); for (i=0; imfr == jedec_table[i].mfr_id && - cfi->id == jedec_table[i].dev_id) - return cfi_jedec_setup(cfi, i); + cfi->id == jedec_table[i].dev_id) { + if (!cfi_jedec_setup(cfi, i)) + return 0; + goto ok_out; + } } - if (!retried++) { - /* Deal with whichever strange chips these were */ - cfi->addr_unlock1 |= cfi->addr_unlock1 << 8; - cfi->addr_unlock2 |= cfi->addr_unlock2 << 8; + switch(unlockpass++) { + case 0: + cfi->addr_unlock1 |= cfi->addr_unlock1 << 4; + cfi->addr_unlock2 |= cfi->addr_unlock2 << 4; + goto retry; + case 1: + cfi->addr_unlock1 = cfi->addr_unlock2 = 0; goto retry; } return 0; + } else { + __u16 mfr; + __u16 id; + + /* Make sure it is a chip of the same manufacturer and id */ + mfr = jedec_read_mfr(map, base, cfi); + id = jedec_read_id(map, base, cfi); + + if ((mfr != cfi->mfr) || (id != cfi->id)) { + printk(KERN_DEBUG "%s: Found different chip or no chip at all (mfr 0x%x, id 0x%x) at 0x%x\n", + map->name, mfr, id, base); + jedec_reset(base, map, cfi); + return 0; + } } /* Check each previous chip to see if it's an alias */ for (i=0; inumchips; i++) { /* This chip should be in read mode if it's one we've already touched. */ - if (jedec_read_mfr(map, base, osf) == cfi->mfr && - jedec_read_id(map, base, osf) == cfi->id) { + if (jedec_read_mfr(map, chips[i].start, cfi) == cfi->mfr && + jedec_read_id(map, chips[i].start, cfi) == cfi->id) { /* Eep. This chip also looks like it's in autoselect mode. Is it an alias for the new one? */ - - cfi_send_gen_cmd(0xF0, 0, chips[i].start, map, cfi, cfi->device_type, NULL); + jedec_reset(chips[i].start, map, cfi); + /* If the device IDs go away, it's an alias */ - if (jedec_read_mfr(map, base, osf) != cfi->mfr || - jedec_read_id(map, base, osf) != cfi->id) { + if (jedec_read_mfr(map, base, cfi) != cfi->mfr || + jedec_read_id(map, base, cfi) != cfi->id) { printk(KERN_DEBUG "%s: Found an alias at 0x%x for the chip at 0x%lx\n", map->name, base, chips[i].start); return 0; @@ -371,9 +999,9 @@ * unfortunate. Stick the new chip in read mode * too and if it's the same, assume it's an alias. */ /* FIXME: Use other modes to do a proper check */ - cfi_send_gen_cmd(0xF0, 0, base, map, cfi, cfi->device_type, NULL); - if (jedec_read_mfr(map, base, osf) == cfi->mfr && - jedec_read_id(map, base, osf) == cfi->id) { + jedec_reset(base, map, cfi); + if (jedec_read_mfr(map, base, cfi) == cfi->mfr && + jedec_read_id(map, base, cfi) == cfi->id) { printk(KERN_DEBUG "%s: Found an alias at 0x%x for the chip at 0x%lx\n", map->name, base, chips[i].start); return 0; @@ -392,8 +1020,9 @@ chips[cfi->numchips].state = FL_READY; cfi->numchips++; +ok_out: /* Put it back into Read Mode */ - cfi_send_gen_cmd(0xF0, 0, base, map, cfi, cfi->device_type, NULL); + jedec_reset(base, map, cfi); printk(KERN_INFO "%s: Found %d x%d devices at 0x%x in %d-bit mode\n", map->name, cfi->interleave, cfi->device_type*8, base, diff -urN orig/drivers/mtd/chips/sharp.c linux/drivers/mtd/chips/sharp.c --- orig/drivers/mtd/chips/sharp.c Mon Aug 5 13:31:01 2002 +++ linux/drivers/mtd/chips/sharp.c Mon Aug 5 13:44:03 2002 @@ -4,7 +4,7 @@ * Copyright 2000,2001 David A. Schleef * 2000,2001 Lineo, Inc. * - * $Id: sharp.c,v 1.7 2002/02/13 15:49:07 dwmw2 Exp $ + * $Id: sharp.c,v 1.8 2002/05/17 08:59:19 dwmw2 Exp $ * * Devices supported: * LH28F016SCT Symmetrical block flash memory, 2Mx8 @@ -421,6 +421,7 @@ } } + instr->state = MTD_ERASE_DONE; if(instr->callback) instr->callback(instr); diff -urN orig/drivers/mtd/cmdline.c linux/drivers/mtd/cmdline.c --- orig/drivers/mtd/cmdline.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/mtd/cmdline.c Thu Jun 20 20:32:00 2002 @@ -0,0 +1,340 @@ +/* + * $Id: cmdline.c,v 1.3 2002/06/20 18:07:49 jamey Exp $ + * + * Read flash partition table from command line + * + * Copyright 2002 SYSGO Real-Time Solutions GmbH + * + * The format for the command line is as follows: + * + * mtdparts=[; := :[,] + * := [@offset][][ro] + * := unique id used in mapping driver/device + * := standard linux memsize OR "-" to denote all remaining space + * := '(' NAME ')' + * + * Examples: + * + * 1 NOR Flash, with 1 single writable partition: + * edb7312-nor:- + * + * 1 NOR Flash with 2 partitions, 1 NAND with one + * edb7312-nor:256k(ARMboot)ro,-(root);edb7312-nand:-(home) + */ + +#include +#include + +#include +#include +#include +#include + +/* error message prefix */ +#define ERRP "mtd: " + +/* debug macro */ +#if 0 +#define dbg(x) do { printk("DEBUG-CMDLINE-PART: "); printk x; } while(0) +#else +#define dbg(x) +#endif + + +/* special size referring to all the remaining space in a partition */ +#define SIZE_REMAINING 0xffffffff + +struct cmdline_mtd_partition { + struct cmdline_mtd_partition *next; + char *mtd_id; + int num_parts; + struct mtd_partition *parts; +}; + +/* mtdpart_setup() parses into here */ +static struct cmdline_mtd_partition *partitions; + +/* the command line passed to mtdpart_setupd() */ +static char *cmdline; +static int cmdline_parsed = 0; + +/* + * Parse one partition definition for an MTD. Since there can be many + * comma separated partition definitions, this function calls itself + * recursively until no more partition definitions are found. Nice side + * effect: the memory to keep the mtd_partition structs and the names + * is allocated upon the last definition being found. At that point the + * syntax has been verified ok. + */ +static struct mtd_partition * newpart(char *s, + char **retptr, + int *num_parts, + int this_part, + unsigned char **extra_mem_ptr, + int extra_mem_size) +{ + struct mtd_partition *parts; + unsigned long size; + unsigned long offset = 0; + char *name; + int name_len; + unsigned char *extra_mem; + char delim; + unsigned int mask_flags; + + /* fetch the partition size */ + if (*s == '-') + { /* assign all remaining space to this partition */ + size = SIZE_REMAINING; + s++; + } + else + { + size = memparse(s, &s); + if (size < PAGE_SIZE) + { + printk(KERN_ERR ERRP "partition size too small (%lx)\n", size); + return 0; + } + } + + /* fetch partition name and flags */ + mask_flags = 0; /* this is going to be a regular partition */ + delim = 0; + /* check for offset */ + if (*s == '@') + { + s++; + offset = memparse(s, &s); + } + /* now look for name */ + if (*s == '(') + { + delim = ')'; + } + if (delim) + { + char *p; + + name = ++s; + if ((p = strchr(name, delim)) == 0) + { + printk(KERN_ERR ERRP "no closing %c found in partition name\n", delim); + return 0; + } + name_len = p - name; + s = p + 1; + } + else + { + name = NULL; + name_len = 13; /* Partition_000 */ + } + + /* record name length for memory allocation later */ + extra_mem_size += name_len + 1; + + /* test for options */ + if (strncmp(s, "ro", 2) == 0) + { + mask_flags |= MTD_WRITEABLE; + s += 2; + } + + /* test if more partitions are following */ + if (*s == ',') + { + if (size == SIZE_REMAINING) + { + printk(KERN_ERR ERRP "no partitions allowed after a fill-up partition\n"); + return 0; + } + /* more partitions follow, parse them */ + if ((parts = newpart(s + 1, &s, num_parts, + this_part + 1, &extra_mem, extra_mem_size)) == 0) + return 0; + } + else + { /* this is the last partition: allocate space for all */ + int alloc_size; + + *num_parts = this_part + 1; + alloc_size = *num_parts * sizeof(struct mtd_partition) + + extra_mem_size; + parts = kmalloc(alloc_size, GFP_KERNEL); + if (!parts) + { + printk(KERN_ERR ERRP "out of memory\n"); + return 0; + } + memset(parts, 0, alloc_size); + extra_mem = (unsigned char *)(parts + *num_parts); + } + /* enter this partition (offset will be calculated later if it is zero at this point) */ + parts[this_part].size = size; + parts[this_part].offset = offset; + parts[this_part].mask_flags = mask_flags; + if (name) + { + strncpy(extra_mem, name, name_len); + extra_mem[name_len] = 0; + } + else + { + sprintf(extra_mem, "Partition_%03d", this_part); + } + parts[this_part].name = extra_mem; + extra_mem += name_len + 1; + + dbg(("partition %d: name <%s>, offset %x, size %x, mask flags %x\n", + this_part, + parts[this_part].name, + parts[this_part].offset, + parts[this_part].size, + parts[this_part].mask_flags)); + + /* return (updated) pointer to extra_mem memory */ + if (extra_mem_ptr) + *extra_mem_ptr = extra_mem; + + /* return (updated) pointer command line string */ + *retptr = s; + + /* return partition table */ + return parts; +} + +/* + * Parse the command line. + */ +static int mtdpart_setup_real(char *s) +{ + cmdline_parsed = 1; + + for( ; s != NULL; ) + { + struct cmdline_mtd_partition *this_mtd; + struct mtd_partition *parts; + int mtd_id_len; + int num_parts; + char *p, *mtd_id; + + mtd_id = s; + /* fetch */ + if (!(p = strchr(s, ':'))) + { + printk(KERN_ERR ERRP "no mtd-id\n"); + return 0; + } + mtd_id_len = p - mtd_id; + + dbg(("parsing <%s>\n", p+1)); + + /* + * parse one mtd. have it reserve memory for the + * struct cmdline_mtd_partition and the mtd-id string. + */ + parts = newpart(p + 1, /* cmdline */ + &s, /* out: updated cmdline ptr */ + &num_parts, /* out: number of parts */ + 0, /* first partition */ + (unsigned char**)&this_mtd, /* out: extra mem */ + mtd_id_len + 1 + sizeof(*this_mtd)); + + /* enter results */ + this_mtd->parts = parts; + this_mtd->num_parts = num_parts; + this_mtd->mtd_id = (char*)(this_mtd + 1); + strncpy(this_mtd->mtd_id, mtd_id, mtd_id_len); + this_mtd->mtd_id[mtd_id_len] = 0; + + /* link into chain */ + this_mtd->next = partitions; + partitions = this_mtd; + + dbg(("mtdid=<%s> num_parts=<%d>\n", + this_mtd->mtd_id, this_mtd->num_parts)); + + + /* EOS - we're done */ + if (*s == 0) + break; + + /* does another spec follow? */ + if (*s != ';') + { + printk(KERN_ERR ERRP "bad character after partition (%c)\n", *s); + return 0; + } + s++; + } + return 1; +} + +/* + * Main function to be called from the MTD mapping driver/device to + * obtain the partitioning information. At this point the command line + * arguments will actually be parsed and turned to struct mtd_partition + * information. + */ +int parse_cmdline_partitions(struct mtd_info *master, + struct mtd_partition **pparts, + const char *mtd_id) +{ + unsigned long offset; + int i; + struct cmdline_mtd_partition *part; + + /* parse command line */ + if (!cmdline_parsed) + mtdpart_setup_real(cmdline); + + for(part = partitions; part; part = part->next) + { + if (!strcmp(part->mtd_id, mtd_id)) + { + for(i = 0, offset = 0; i < part->num_parts; i++) + { + if (!part->parts[i].offset) + part->parts[i].offset = offset; + else + offset = part->parts[i].offset; + if (part->parts[i].size == SIZE_REMAINING) + part->parts[i].size = master->size - offset; + if (offset + part->parts[i].size > master->size) + { + printk(KERN_WARNING ERRP + "%s: partitioning exceeds flash size, truncating\n", + mtd_id); + part->parts[i].size = master->size - offset; + part->num_parts = i; + } + offset += part->parts[i].size; + } + *pparts = part->parts; + return part->num_parts; + } + } + return -EINVAL; +} + + +/* + * This is the handler for our kernel parameter, called from + * main.c::checksetup(). Note that we can not yet kmalloc() anything, + * so we only save the commandline for later processing. + */ +static int __init mtdpart_setup(char *s) +{ + cmdline = s; + return 1; +} + +__setup("mtdparts=", mtdpart_setup); + +EXPORT_SYMBOL(parse_cmdline_partitions); + +MODULE_LICENSE("GPL"); +MODULE_AUTHOR("Marius Groeger "); +MODULE_DESCRIPTION("Command line configuration of MTD partitions"); diff -urN orig/drivers/mtd/devices/blkmtd.c linux/drivers/mtd/devices/blkmtd.c --- orig/drivers/mtd/devices/blkmtd.c Wed Feb 27 14:25:03 2002 +++ linux/drivers/mtd/devices/blkmtd.c Wed Mar 13 15:54:52 2002 @@ -1,5 +1,5 @@ /* - * $Id: blkmtd.c,v 1.7 2001/11/10 17:06:30 spse Exp $ + * $Id: blkmtd.c,v 1.10 2002/03/03 15:42:06 spse Exp $ * * blkmtd.c - use a block device as a fake MTD * @@ -45,9 +45,8 @@ #include #include - #include -#include +#include #include #include #include @@ -65,9 +64,7 @@ /* Default erase size in K, always make it a multiple of PAGE_SIZE */ #define CONFIG_MTD_BLKDEV_ERASESIZE 128 -#define VERSION "1.7" -extern int *blk_size[]; -extern int *blksize_size[]; +#define VERSION "1.9" /* Info for the block device */ typedef struct mtd_raw_dev_data_s { @@ -112,27 +109,17 @@ static volatile int write_task_finish; /* ipc with the write thread */ -#if LINUX_VERSION_CODE > KERNEL_VERSION(2,3,0) static DECLARE_MUTEX_LOCKED(thread_sem); static DECLARE_WAIT_QUEUE_HEAD(thr_wq); static DECLARE_WAIT_QUEUE_HEAD(mtbd_sync_wq); -#else -static struct semaphore thread_sem = MUTEX_LOCKED; -DECLARE_WAIT_QUEUE_HEAD(thr_wq); -DECLARE_WAIT_QUEUE_HEAD(mtbd_sync_wq); -#endif - /* Module parameters passed by insmod/modprobe */ char *device; /* the block device to use */ int erasesz; /* optional default erase size */ int ro; /* optional read only flag */ -int bs; /* optionally force the block size (avoid using) */ -int count; /* optionally force the block count (avoid using) */ int wqs; /* optionally set the write queue size */ -#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,2,0) MODULE_LICENSE("GPL"); MODULE_AUTHOR("Simon Evans "); MODULE_DESCRIPTION("Emulate an MTD using a block device"); @@ -142,12 +129,7 @@ MODULE_PARM_DESC(erasesz, "optional erase size to use in KB. eg 4=4K."); MODULE_PARM(ro, "i"); MODULE_PARM_DESC(ro, "1=Read only, writes and erases cause errors"); -MODULE_PARM(bs, "i"); -MODULE_PARM_DESC(bs, "force the block size in bytes"); -MODULE_PARM(count, "i"); -MODULE_PARM_DESC(count, "force the block count"); MODULE_PARM(wqs, "i"); -#endif /* Page cache stuff */ @@ -170,7 +152,7 @@ unsigned long *blocks; if(!rawdevice) { - printk("blkmtd: readpage: PANIC file->private_data == NULL\n"); + printk(KERN_ERR "blkmtd: readpage: PANIC rawdevice == NULL\n"); return -EIO; } dev = to_kdev_t(rawdevice->binding->bd_dev); @@ -220,16 +202,16 @@ DEBUG(3, "blkmtd: readpage: getting kiovec\n"); err = alloc_kiovec(1, &iobuf); if (err) { - printk("blkmtd: cant allocate kiobuf\n"); + printk(KERN_CRIT "blkmtd: cant allocate kiobuf\n"); SetPageError(page); return err; } /* Pre 2.4.4 doesnt have space for the block list in the kiobuf */ #if LINUX_VERSION_CODE < KERNEL_VERSION(2,4,4) - blocks = kmalloc(KIO_MAX_SECTORS * sizeof(unsigned long)); + blocks = kmalloc(KIO_MAX_SECTORS * sizeof(unsigned long), GFP_KERNEL); if(blocks == NULL) { - printk("blkmtd: cant allocate iobuf blocks\n"); + printk(KERN_CRIT "blkmtd: cant allocate iobuf blocks\n"); free_kiovec(1, &iobuf); SetPageError(page); return -ENOMEM; @@ -257,7 +239,8 @@ if(rawdevice->partial_last_page && page->index == rawdevice->partial_last_page) { int offset = rawdevice->last_page_sectors << rawdevice->sector_bits; int count = PAGE_SIZE-offset; - DEBUG(3, "blkmtd: clear last partial page: offset = %d count = %d\n", offset, count); + DEBUG(3, "blkmtd: clear last partial page: offset = %d count = %d\n", + offset, count); memset(page_address(page)+offset, 0, count); sectors = rawdevice->last_page_sectors; } @@ -274,7 +257,7 @@ #endif if(err != PAGE_SIZE) { - printk("blkmtd: readpage: error reading page %ld\n", page->index); + printk(KERN_ERR "blkmtd: readpage: error reading page %ld\n", page->index); memset(page_address(page), 0, PAGE_SIZE); SetPageError(page); err = -EIO; @@ -311,20 +294,20 @@ tsk->tty = NULL; spin_lock_irq(&tsk->sigmask_lock); sigfillset(&tsk->blocked); - recalc_sigpending(tsk); + recalc_sigpending(); spin_unlock_irq(&tsk->sigmask_lock); exit_sighand(tsk); if(alloc_kiovec(1, &iobuf)) { - printk("blkmtd: write_queue_task cant allocate kiobuf\n"); + printk(KERN_CRIT "blkmtd: write_queue_task cant allocate kiobuf\n"); return 0; } /* Pre 2.4.4 doesnt have space for the block list in the kiobuf */ #if LINUX_VERSION_CODE < KERNEL_VERSION(2,4,4) - blocks = kmalloc(KIO_MAX_SECTORS * sizeof(unsigned long)); + blocks = kmalloc(KIO_MAX_SECTORS * sizeof(unsigned long), GFP_KERNEL); if(blocks == NULL) { - printk("blkmtd: write_queue_task cant allocate iobuf blocks\n"); + printk(KERN_CRIT "blkmtd: write_queue_task cant allocate iobuf blocks\n"); free_kiovec(1, &iobuf); return 0; } @@ -456,7 +439,8 @@ mtdblkdev_write_queue_t *item; int i; DECLARE_WAITQUEUE(wait, current); - DEBUG(2, "blkmtd: queue_page_write: adding pagenr = %d pagecnt = %d\n", pagenr, pagecnt); + DEBUG(2, "blkmtd: queue_page_write: adding pagenr = %d pagecnt = %d\n", + pagenr, pagecnt); if(!pagecnt) return 0; @@ -499,7 +483,8 @@ schedule(); current->state = TASK_RUNNING; remove_wait_queue(&mtbd_sync_wq, &wait); - DEBUG(3, "blkmtd: queue_page_write: Queue has %d items in it\n", write_queue_cnt); + DEBUG(3, "blkmtd: queue_page_write: Queue has %d items in it\n", + write_queue_cnt); goto test_lock; } @@ -537,7 +522,8 @@ /* check readonly */ if(rawdevice->readonly) { - printk("blkmtd: error: trying to erase readonly device %s\n", device); + printk(KERN_ERR "blkmtd: error: trying to erase readonly device %s\n", + device); instr->state = MTD_ERASE_FAILED; goto erase_callback; } @@ -562,7 +548,8 @@ if(!numregions) { /* Not a valid erase block */ - printk("blkmtd: erase: invalid erase request 0x%lX @ 0x%08X\n", len, from); + printk(KERN_ERR "blkmtd: erase: invalid erase request 0x%lX @ 0x%08X\n", + len, from); instr->state = MTD_ERASE_FAILED; err = -EIO; } @@ -594,7 +581,8 @@ DEBUG(3, "blkmtd: erase: doing grab_cache_page() for page %d\n", pagenr); page = grab_cache_page(&rawdevice->as, pagenr); if(!page) { - DEBUG(3, "blkmtd: erase: grab_cache_page() failed for page %d\n", pagenr); + DEBUG(3, "blkmtd: erase: grab_cache_page() failed for page %d\n", + pagenr); kfree(pages); err = -EIO; instr->state = MTD_ERASE_FAILED; @@ -656,7 +644,8 @@ offset = from - (pagenr << PAGE_SHIFT); pages = (offset+len+PAGE_SIZE-1) >> PAGE_SHIFT; - DEBUG(3, "blkmtd: read: pagenr = %d offset = %d, pages = %d\n", pagenr, offset, pages); + DEBUG(3, "blkmtd: read: pagenr = %d offset = %d, pages = %d\n", + pagenr, offset, pages); /* just loop through each page, getting it via readpage() - slow but easy */ while(pages) { @@ -670,7 +659,7 @@ wait_on_page(page); if(!Page_Uptodate(page)) { /* error reading page */ - printk("blkmtd: read: page not uptodate\n"); + printk(KERN_ERR "blkmtd: read: page not uptodate\n"); page_cache_release(page); return -EIO; } @@ -725,7 +714,7 @@ /* handle readonly and out of range numbers */ if(rawdevice->readonly) { - printk("blkmtd: error: trying to write to a readonly device %s\n", device); + printk(KERN_ERR "blkmtd: error: trying to write to a readonly device %s\n", device); return -EROFS; } @@ -765,7 +754,8 @@ if(len3) pagecnt++; - DEBUG(3, "blkmtd: write: len1 = %d len2 = %d len3 = %d pagecnt = %d\n", len1, len2, len3, pagecnt); + DEBUG(3, "blkmtd: write: len1 = %d len2 = %d len3 = %d pagecnt = %d\n", + len1, len2, len3, pagecnt); /* get space for list of pages */ pages = kmalloc(pagecnt * sizeof(struct page *), GFP_KERNEL); @@ -778,7 +768,8 @@ /* do partial start region */ struct page *page; - DEBUG(3, "blkmtd: write: doing partial start, page = %d len = %d offset = %d\n", pagenr, len1, offset); + DEBUG(3, "blkmtd: write: doing partial start, page = %d len = %d offset = %d\n", + pagenr, len1, offset); page = read_cache_page(&rawdevice->as, pagenr, (filler_t *)blkmtd_readpage, rawdevice); if(IS_ERR(page)) { @@ -805,7 +796,7 @@ page = grab_cache_page(&rawdevice->as, pagenr); DEBUG(3, "blkmtd: write: got page %d from page cache\n", pagenr); if(!page) { - printk("blkmtd: write: cant grab cache page %d\n", pagenr); + printk(KERN_WARNING "blkmtd: write: cant grab cache page %d\n", pagenr); err = -EIO; goto write_err; } @@ -936,7 +927,8 @@ /* Cleanup and exit - sync the device and kill of the kernel thread */ -static void __exit cleanup_blkmtd(void) + +static void __devexit cleanup_blkmtd(void) { #ifdef BLKMTD_PROC_DEBUG if(blkmtd_proc) { @@ -970,11 +962,9 @@ UnlockPage(erase_page); __free_pages(erase_page, 0); } - printk("blkmtd: unloaded for %s\n", device); + printk(KERN_INFO "blkmtd: unloaded for %s\n", device); } -extern struct module __this_module; - #ifndef MODULE /* Handle kernel boot params */ @@ -983,6 +973,7 @@ static int __init param_blkmtd_device(char *str) { device = str; + printk("blkmtd device = %s", device); return 1; } @@ -1001,24 +992,9 @@ } -static int __init param_blkmtd_bs(char *str) -{ - bs = simple_strtol(str, NULL, 0); - return 1; -} - - -static int __init param_blkmtd_count(char *str) -{ - count = simple_strtol(str, NULL, 0); - return 1; -} - __setup("blkmtd_device=", param_blkmtd_device); __setup("blkmtd_erasesz=", param_blkmtd_erasesz); __setup("blkmtd_ro=", param_blkmtd_ro); -__setup("blkmtd_bs=", param_blkmtd_bs); -__setup("blkmtd_count=", param_blkmtd_count); #endif @@ -1062,7 +1038,7 @@ #endif int maj, min; - int i, blocksize, blocksize_bits; + int blocksize = BLOCK_SIZE, blocksize_bits; loff_t size = 0; int readonly = 0; int erase_size = CONFIG_MTD_BLKDEV_ERASESIZE; @@ -1070,10 +1046,12 @@ int err; int mode; int regions; + int i; + printk("blkmtd: starting\n"); /* Check args */ if(device == 0) { - printk("blkmtd: error, missing `device' name\n"); + printk(KERN_ERR "blkmtd: error, missing `device' name\n"); return -EINVAL; } @@ -1100,7 +1078,7 @@ file = filp_open(device, mode, 0); if(IS_ERR(file)) { - printk("blkmtd: error, cant open device %s\n", device); + printk(KERN_ERR "blkmtd: error, cant open device %s\n", device); DEBUG(2, "blkmtd: filp_open returned %ld\n", PTR_ERR(file)); return 1; } @@ -1109,7 +1087,7 @@ numbers */ inode = file->f_dentry->d_inode; if(!S_ISBLK(inode->i_mode)) { - printk("blkmtd: %s not a block device\n", device); + printk(KERN_ERR "blkmtd: %s not a block device\n", device); filp_close(file, NULL); return 1; } @@ -1119,31 +1097,25 @@ rdev = name_to_kdev_t(device); #endif - maj = MAJOR(rdev); - min = MINOR(rdev); - DEBUG(1, "blkmtd: found a block device major = %d, minor = %d\n", maj, min); - if(!rdev) { - printk("blkmtd: bad block device: `%s'\n", device); + printk(KERN_ERR "blkmtd: bad block device: `%s'\n", device); return 1; } + maj = MAJOR(rdev); + min = MINOR(rdev); + DEBUG(1, "blkmtd: found a block device major = %d, minor = %d\n", maj, min); + if(maj == MTD_BLOCK_MAJOR) { - printk("blkmtd: attempting to use an MTD device as a block device\n"); + printk(KERN_ERR "blkmtd: attempting to use an MTD device as a block device\n"); return 1; } DEBUG(1, "blkmtd: devname = %s\n", bdevname(rdev)); - blocksize = BLOCK_SIZE; - if(bs) { - blocksize = bs; - } else { - if (blksize_size[maj] && blksize_size[maj][min]) { - DEBUG(2, "blkmtd: blksize_size = %d\n", blksize_size[maj][min]); - blocksize = blksize_size[maj][min]; - } - } + if(blksize_size[maj] && blksize_size[maj][min]) + blocksize = blksize_size[maj][min]; + i = blocksize; blocksize_bits = 0; while(i != 1) { @@ -1151,18 +1123,12 @@ i >>= 1; } - if(count) { - size = count; - } else { - if (blk_size[maj]) { - size = ((loff_t) blk_size[maj][min] << BLOCK_SIZE_BITS) >> blocksize_bits; - } - } - size *= blocksize; + size = (loff_t) blk_size[maj][min] << blocksize_bits; + DEBUG(1, "blkmtd: size = %ld\n", (long int)size); if(size == 0) { - printk("blkmtd: cant determine size\n"); + printk(KERN_ERR "blkmtd: cant determine size\n"); return 1; } @@ -1241,10 +1207,8 @@ mtd_rawdevice->as.i_mmap = NULL; mtd_rawdevice->as.i_mmap_shared = NULL; mtd_rawdevice->as.gfp_mask = GFP_KERNEL; - -#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,2,0) mtd_rawdevice->mtd_info.module = THIS_MODULE; -#endif + if (add_mtd_device(&mtd_rawdevice->mtd_info)) { err = -EIO; goto init_err; @@ -1272,7 +1236,7 @@ DEBUG(2, "blkmtd: init: starting kernel task\n"); kernel_thread(write_queue_task, NULL, CLONE_FS | CLONE_FILES | CLONE_SIGHAND); DEBUG(2, "blkmtd: init: started\n"); - printk("blkmtd loaded: version = %s using %s erase_size = %dK %s\n", + printk(KERN_INFO "blkmtd loaded: version = %s using %s erase_size = %dK %s\n", VERSION, device, erase_size, (readonly) ? "(read-only)" : ""); } @@ -1282,7 +1246,7 @@ blkmtd_proc = create_proc_read_entry("blkmtd_debug", 0444, NULL, blkmtd_proc_read, NULL); if(blkmtd_proc == NULL) { - printk("Cant create /proc/blkmtd_debug\n"); + printk(KERN_ERR "Cant create /proc/blkmtd_debug\n"); } else { blkmtd_proc->owner = THIS_MODULE; } diff -urN orig/drivers/mtd/devices/docprobe.c linux/drivers/mtd/devices/docprobe.c --- orig/drivers/mtd/devices/docprobe.c Sun Oct 14 20:53:09 2001 +++ linux/drivers/mtd/devices/docprobe.c Fri May 31 11:48:55 2002 @@ -3,7 +3,7 @@ /* Probe routines common to all DoC devices */ /* (c) 1999 Machine Vision Holdings, Inc. */ /* Author: David Woodhouse */ -/* $Id: docprobe.c,v 1.30 2001/10/02 15:05:13 dwmw2 Exp $ */ +/* $Id: docprobe.c,v 1.32 2002/05/02 08:23:27 dwmw2 Exp $ */ @@ -67,7 +67,7 @@ static unsigned long doc_config_location = CONFIG_MTD_DOCPROBE_ADDRESS; MODULE_PARM(doc_config_location, "l"); - +MODULE_PARM_DESC(doc_config_location, "Physical memory address at which to probe for DiskOnChip"); static unsigned long __initdata doc_locations[] = { #if defined (__alpha__) || defined(__i386__) || defined(__x86_64__) @@ -84,7 +84,7 @@ 0xe0000, 0xe2000, 0xe4000, 0xe6000, 0xe8000, 0xea000, 0xec000, 0xee000, #endif /* CONFIG_MTD_DOCPROBE_HIGH */ -#elif defined(__ppc__) +#elif defined(__PPC__) 0xe4000000, #elif defined(CONFIG_MOMENCO_OCELOT) 0x2f000000, diff -urN orig/drivers/mtd/devices/mtdram.c linux/drivers/mtd/devices/mtdram.c --- orig/drivers/mtd/devices/mtdram.c Mon Aug 5 13:31:01 2002 +++ linux/drivers/mtd/devices/mtdram.c Mon Aug 5 13:44:03 2002 @@ -1,6 +1,6 @@ /* * mtdram - a test mtd device - * $Id: mtdram.c,v 1.26 2001/12/01 10:24:18 dwmw2 Exp $ + * $Id: mtdram.c,v 1.27 2002/03/28 15:05:19 dwmw2 Exp $ * Author: Alexander Larsson * * Copyright (c) 1999 Alexander Larsson @@ -28,7 +28,9 @@ static unsigned long total_size = CONFIG_MTDRAM_TOTAL_SIZE; static unsigned long erase_size = CONFIG_MTDRAM_ERASE_SIZE; MODULE_PARM(total_size,"l"); +MODULE_PARM_DESC(total_size, "Total device size in KiB"); MODULE_PARM(erase_size,"l"); +MODULE_PARM_DESC(erase_size, "Device erase block size in KiB"); #define MTDRAM_TOTAL_SIZE (total_size * 1024) #define MTDRAM_ERASE_SIZE (erase_size * 1024) #else diff -urN orig/drivers/mtd/devices/slram.c linux/drivers/mtd/devices/slram.c --- orig/drivers/mtd/devices/slram.c Sun Oct 14 20:53:09 2001 +++ linux/drivers/mtd/devices/slram.c Fri May 31 11:48:55 2002 @@ -1,6 +1,32 @@ /*====================================================================== - $Id: slram.c,v 1.25 2001/10/02 15:05:13 dwmw2 Exp $ + $Id: slram.c,v 1.26 2002/05/17 07:07:39 jochen Exp $ + + This driver provides a method to access memory not used by the kernel + itself (i.e. if the kernel commandline mem=xxx is used). To actually + use slram at least mtdblock or mtdchar is required (for block or + character device access). + + Usage: + + if compiled as loadable module: + modprobe slram map=,, + if statically linked into the kernel use the following kernel cmd.line + slram=,, + + : name of the device that will be listed in /proc/mtd + : start of the memory region, decimal or hex (0xabcdef) + : end of the memory region. It's possible to use +0x1234 + to specify the offset instead of the absolute address + + NOTE: + With slram it's only possible to map a contigous memory region. Therfore + if there's a device mapped somewhere in the region specified slram will + fail to load (see kernel log if modprobe fails). + + - + + Jochen Schaeuble ======================================================================*/ @@ -177,7 +203,7 @@ (*curmtd)->mtdinfo->write = slram_write; (*curmtd)->mtdinfo->module = THIS_MODULE; (*curmtd)->mtdinfo->type = MTD_RAM; - (*curmtd)->mtdinfo->erasesize = 0x10000; + (*curmtd)->mtdinfo->erasesize = 0x0; if (add_mtd_device((*curmtd)->mtdinfo)) { E("slram: Failed to register new device\n"); diff -urN orig/drivers/mtd/maps/Config.in linux/drivers/mtd/maps/Config.in --- orig/drivers/mtd/maps/Config.in Mon Aug 5 13:31:01 2002 +++ linux/drivers/mtd/maps/Config.in Fri Feb 21 15:17:33 2003 @@ -1,6 +1,6 @@ # drivers/mtd/maps/Config.in -# $Id: Config.in,v 1.28 2002/03/08 16:34:35 rkaiser Exp $ +# $Id: Config.in,v 1.32 2002/05/02 23:20:57 ahennessy Exp $ mainmenu_option next_comment @@ -41,14 +41,17 @@ dep_tristate ' System flash on MBX860 board' CONFIG_MTD_MBX860 $CONFIG_MTD_CFI dep_tristate ' CFI Flash device mapped on D-Box2' CONFIG_MTD_DBOX2 $CONFIG_MTD_CFI dep_tristate ' CFI Flash device mapping on FlagaDM' CONFIG_MTD_CFI_FLAGADM $CONFIG_MTD_CFI + dep_tristate ' CFI Flash device mapped on IBM Redwood-4/5' CONFIG_MTD_REDWOOD $CONFIG_MTD_CFI fi if [ "$CONFIG_MIPS" = "y" ]; then - dep_tristate ' Pb1000 boot flash device' CONFIG_MTD_PB1000 $CONFIG_MIPS_PB1000 + dep_tristate ' Pb1000 MTD support' CONFIG_MTD_PB1000 $CONFIG_MIPS_PB1000 dep_tristate ' Pb1500 MTD support' CONFIG_MTD_PB1500 $CONFIG_MIPS_PB1500 - if [ "$CONFIG_MTD_PB1500" = "y" -o "$CONFIG_MTD_PB1500" = "m" ]; then - bool ' Pb1500 boot flash device' CONFIG_MTD_PB1500_BOOT - bool ' Pb1500 user flash device (2nd 32MB bank)' CONFIG_MTD_PB1500_USER + dep_tristate ' Pb1100 MTD support' CONFIG_MTD_PB1100 $CONFIG_MIPS_PB1100 + if [ "$CONFIG_MTD_PB1500" = "y" -o "$CONFIG_MTD_PB1500" = "m" \ + -o "$CONFIG_MTD_PB1100" = "y" -o "$CONFIG_MTD_PB1100" = "m" ]; then + bool ' Pb[15]00 boot flash device' CONFIG_MTD_PB1500_BOOT + bool ' Pb[15]00 user flash device (2nd 32MB bank)' CONFIG_MTD_PB1500_USER fi dep_tristate ' Flash chip mapping on ITE QED-4N-S01B, Globespan IVR or custom board' CONFIG_MTD_CSTM_MIPS_IXX $CONFIG_MTD_CFI $CONFIG_MTD_JEDEC $CONFIG_MTD_PARTITIONS if [ "$CONFIG_MTD_CSTM_MIPS_IXX" = "y" -o "$CONFIG_MTD_CSTM_MIPS_IXX" = "m" ]; then @@ -74,8 +77,11 @@ dep_tristate ' CFI Flash device mapped on StrongARM SA11x0' CONFIG_MTD_SA1100 $CONFIG_MTD_CFI $CONFIG_ARCH_SA1100 $CONFIG_MTD_PARTITIONS dep_tristate ' CFI Flash device mapped on DC21285 Footbridge' CONFIG_MTD_DC21285 $CONFIG_MTD_CFI $CONFIG_ARCH_FOOTBRIDGE $CONFIG_MTD_PARTITIONS dep_tristate ' CFI Flash device mapped on the XScale IQ80310 board' CONFIG_MTD_IQ80310 $CONFIG_MTD_CFI $CONFIG_ARCH_IQ80310 - dep_tristate ' CFI Flash device mapped on Epxa10db' CONFIG_MTD_EPXA10DB $CONFIG_MTD_CFI $CONFIG_MTD_PARTITIONS $CONFIG_ARCH_CAMELOT + dep_tristate ' CFI Flash device mapped on the FortuNet board' CONFIG_MTD_FORTUNET $CONFIG_MTD_CFI $CONFIG_MTD_PARTITIONS $CONFIG_ARCH_FORTUNET + dep_tristate ' CFI Flash device mapped on Epxa' CONFIG_MTD_EPXA $CONFIG_MTD_CFI $CONFIG_MTD_PARTITIONS $CONFIG_ARCH_CAMELOT dep_tristate ' NV-RAM mapping AUTCPU12 board' CONFIG_MTD_AUTCPU12 $CONFIG_ARCH_AUTCPU12 + dep_tristate ' CFI Flash device mapped on EDB7312' CONFIG_MTD_EDB7312 $CONFIG_ARCH_EDB7212 $CONFIG_MTD_CFI + dep_tristate ' JEDEC Flash device mapped on impA7' CONFIG_MTD_IMPA7 $CONFIG_MTD_JEDECPROBE fi if [ "$CONFIG_ALPHA" = "y" ]; then dep_tristate ' Flash chip mapping on TSUNAMI' CONFIG_MTD_TSUNAMI $CONFIG_MTD_GENPROBE diff -urN orig/drivers/mtd/maps/Makefile linux/drivers/mtd/maps/Makefile --- orig/drivers/mtd/maps/Makefile Mon Aug 5 13:31:01 2002 +++ linux/drivers/mtd/maps/Makefile Fri Feb 21 15:17:33 2003 @@ -1,7 +1,7 @@ # # linux/drivers/maps/Makefile # -# $Id: Makefile,v 1.21 2002/02/22 09:34:44 gleixner Exp $ +# $Id: Makefile,v 1.25 2002/05/02 23:20:57 ahennessy Exp $ O_TARGET := mapslink.o @@ -13,7 +13,7 @@ obj-$(CONFIG_MTD_DC21285) += dc21285.o obj-$(CONFIG_MTD_DILNETPC) += dilnetpc.o obj-$(CONFIG_MTD_ELAN_104NC) += elan-104nc.o -obj-$(CONFIG_MTD_EPXA10DB) += epxa10db-flash.o +obj-$(CONFIG_MTD_EPXA) += epxa-flash.o obj-$(CONFIG_MTD_IQ80310) += iq80310.o obj-$(CONFIG_MTD_L440GX) += l440gx.o obj-$(CONFIG_MTD_AMD766ROM) += amd766rom.o @@ -27,6 +27,9 @@ obj-$(CONFIG_MTD_RPXLITE) += rpxlite.o obj-$(CONFIG_MTD_TQM8XXL) += tqm8xxl.o obj-$(CONFIG_MTD_SA1100) += sa1100-flash.o +ifeq ($(CONFIG_ASSABET_NEPONSET),y) + obj-$(CONFIG_MTD_SA1100) += neponset-flash.o +endif obj-$(CONFIG_MTD_SBC_GXX) += sbc_gxx.o obj-$(CONFIG_MTD_SC520CDP) += sc520cdp.o obj-$(CONFIG_MTD_NETSC520) += netsc520.o @@ -35,9 +38,14 @@ obj-$(CONFIG_MTD_DBOX2) += dbox2-flash.o obj-$(CONFIG_MTD_OCELOT) += ocelot.o obj-$(CONFIG_MTD_SOLUTIONENGINE)+= solutionengine.o +obj-$(CONFIG_MTD_FORTUNET) += fortunet.o obj-$(CONFIG_MTD_PCI) += pci.o obj-$(CONFIG_MTD_PB1000) += pb1xxx-flash.o obj-$(CONFIG_MTD_PB1500) += pb1xxx-flash.o +obj-$(CONFIG_MTD_PB1100) += pb1xxx-flash.o obj-$(CONFIG_MTD_AUTCPU12) += autcpu12-nvram.o +obj-$(CONFIG_MTD_EDB7312) += edb7312.o +obj-$(CONFIG_MTD_IMPA7) += impa7.o +obj-$(CONFIG_MTD_REDWOOD) += redwood.o include $(TOPDIR)/Rules.make diff -urN orig/drivers/mtd/maps/edb7312.c linux/drivers/mtd/maps/edb7312.c --- orig/drivers/mtd/maps/edb7312.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/mtd/maps/edb7312.c Tue Apr 30 14:12:04 2002 @@ -0,0 +1,223 @@ +/* + * $Id: edb7312.c,v 1.1 2002/04/30 13:12:04 mag Exp $ + * + * Handle mapping of the NOR flash on Cogent EDB7312 boards + * + * Copyright 2002 SYSGO Real-Time Solutions GmbH + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + */ + +#include +#include +#include +#include +#include +#include +#include + +#ifdef CONFIG_MTD_PARTITIONS +#include +#endif + +#define WINDOW_ADDR 0x00000000 /* physical properties of flash */ +#define WINDOW_SIZE 0x01000000 +#define BUSWIDTH 2 +#define FLASH_BLOCKSIZE_MAIN 0x20000 +#define FLASH_NUMBLOCKS_MAIN 128 +/* can be "cfi_probe", "jedec_probe", "map_rom", 0 }; */ +#define PROBETYPES { "cfi_probe", 0 } + +#define MSG_PREFIX "EDB7312-NOR:" /* prefix for our printk()'s */ +#define MTDID "edb7312-nor" /* for mtdparts= partitioning */ + +static struct mtd_info *mymtd; + +__u8 edb7312nor_read8(struct map_info *map, unsigned long ofs) +{ + return __raw_readb(map->map_priv_1 + ofs); +} + +__u16 edb7312nor_read16(struct map_info *map, unsigned long ofs) +{ + return __raw_readw(map->map_priv_1 + ofs); +} + +__u32 edb7312nor_read32(struct map_info *map, unsigned long ofs) +{ + return __raw_readl(map->map_priv_1 + ofs); +} + +#ifdef CFI_WORD_64 +__u64 edb7312nor_read64(struct map_info *map, unsigned long ofs) +{ + return __raw_readll(map->map_priv_1 + ofs); +} +#endif + +void edb7312nor_copy_from(struct map_info *map, void *to, unsigned long from, ssize_t len) +{ + memcpy_fromio(to, map->map_priv_1 + from, len); +} + +void edb7312nor_write8(struct map_info *map, __u8 d, unsigned long adr) +{ + __raw_writeb(d, map->map_priv_1 + adr); + mb(); +} + +void edb7312nor_write16(struct map_info *map, __u16 d, unsigned long adr) +{ + __raw_writew(d, map->map_priv_1 + adr); + mb(); +} + +void edb7312nor_write32(struct map_info *map, __u32 d, unsigned long adr) +{ + __raw_writel(d, map->map_priv_1 + adr); + mb(); +} + +#ifdef CFI_WORD_64 +void edb7312nor_write64(struct map_info *map, __u64 d, unsigned long adr) +{ + __raw_writell(d, map->map_priv_1 + adr); + mb(); +} +#endif + +void edb7312nor_copy_to(struct map_info *map, unsigned long to, const void *from, ssize_t len) +{ + memcpy_toio(map->map_priv_1 + to, from, len); +} + +struct map_info edb7312nor_map = { + name: "NOR flash on EDB7312", + size: WINDOW_SIZE, + buswidth: BUSWIDTH, + read8: edb7312nor_read8, + read16: edb7312nor_read16, + read32: edb7312nor_read32, +#ifdef CFI_WORD_64 + read64: edb7312nor_read64, +#endif + copy_from: edb7312nor_copy_from, + write8: edb7312nor_write8, + write16: edb7312nor_write16, + write32: edb7312nor_write32, +#ifdef CFI_WORD_64 + write64: edb7312nor_write64, +#endif + copy_to: edb7312nor_copy_to +}; + +#ifdef CONFIG_MTD_PARTITIONS + +/* + * MTD partitioning stuff + */ +static struct mtd_partition static_partitions[3] = +{ + { + name: "ARMboot", + size: 0x40000, + offset: 0 + }, + { + name: "Kernel", + size: 0x200000, + offset: 0x40000 + }, + { + name: "RootFS", + size: 0xDC0000, + offset: 0x240000 + }, +}; + +#define NB_OF(x) (sizeof (x) / sizeof (x[0])) + +#ifdef CONFIG_MTD_CMDLINE_PARTS +int parse_cmdline_partitions(struct mtd_info *master, + struct mtd_partition **pparts, + const char *mtd_id); +#endif + +#endif + +static int mtd_parts_nb = 0; +static struct mtd_partition *mtd_parts = 0; + +int __init init_edb7312nor(void) +{ + static const char *rom_probe_types[] = PROBETYPES; + const char **type; + const char *part_type = 0; + + printk(KERN_NOTICE MSG_PREFIX "0x%08x at 0x%08x\n", + WINDOW_SIZE, WINDOW_ADDR); + edb7312nor_map.map_priv_1 = (unsigned long) + ioremap(WINDOW_ADDR, WINDOW_SIZE); + + if (!edb7312nor_map.map_priv_1) { + printk(MSG_PREFIX "failed to ioremap\n"); + return -EIO; + } + + mymtd = 0; + type = rom_probe_types; + for(; !mymtd && *type; type++) { + mymtd = do_map_probe(*type, &edb7312nor_map); + } + if (mymtd) { + mymtd->module = THIS_MODULE; + +#ifdef CONFIG_MTD_PARTITIONS +#ifdef CONFIG_MTD_CMDLINE_PARTS + mtd_parts_nb = parse_cmdline_partitions(mymtd, &mtd_parts, MTDID); + if (mtd_parts_nb > 0) + part_type = "command line"; +#endif + if (mtd_parts_nb == 0) + { + mtd_parts = static_partitions; + mtd_parts_nb = NB_OF(static_partitions); + part_type = "static"; + } +#endif + add_mtd_device(mymtd); + if (mtd_parts_nb == 0) + printk(KERN_NOTICE MSG_PREFIX "no partition info available\n"); + else + { + printk(KERN_NOTICE MSG_PREFIX + "using %s partition definition\n", part_type); + add_mtd_partitions(mymtd, mtd_parts, mtd_parts_nb); + } + return 0; + } + + iounmap((void *)edb7312nor_map.map_priv_1); + return -ENXIO; +} + +static void __exit cleanup_edb7312nor(void) +{ + if (mymtd) { + del_mtd_device(mymtd); + map_destroy(mymtd); + } + if (edb7312nor_map.map_priv_1) { + iounmap((void *)edb7312nor_map.map_priv_1); + edb7312nor_map.map_priv_1 = 0; + } +} + +module_init(init_edb7312nor); +module_exit(cleanup_edb7312nor); + +MODULE_LICENSE("GPL"); +MODULE_AUTHOR("Marius Groeger "); +MODULE_DESCRIPTION("Generic configurable MTD map driver"); diff -urN orig/drivers/mtd/maps/epxa-flash.c linux/drivers/mtd/maps/epxa-flash.c --- orig/drivers/mtd/maps/epxa-flash.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/mtd/maps/epxa-flash.c Thu Feb 27 23:20:11 2003 @@ -0,0 +1,234 @@ +/* + * Flash memory access on EPXA based devices + * + * (C) 2000 Nicolas Pitre + * Copyright (C) 2001 Altera Corporation + * Copyright (C) 2001 Red Hat, Inc. + * + * $Id: epxa10db-flash.c,v 1.4 2002/08/22 10:46:19 cdavies Exp $ + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ + +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#ifdef CONFIG_EPXA10DB +#define BOARD_NAME "EPXA10DB" +#else +#define BOARD_NAME "EPXA1DB" +#endif + +static int nr_parts = 0; +static struct mtd_partition *parts; + +static struct mtd_info *mymtd; + +extern int parse_redboot_partitions(struct mtd_info *, struct mtd_partition **); +static int epxa_default_partitions(struct mtd_info *master, struct mtd_partition **pparts); + +static __u8 epxa_read8(struct map_info *map, unsigned long ofs) +{ + return __raw_readb(map->map_priv_1 + ofs); +} + +static __u16 epxa_read16(struct map_info *map, unsigned long ofs) +{ + return __raw_readw(map->map_priv_1 + ofs); +} + +static __u32 epxa_read32(struct map_info *map, unsigned long ofs) +{ + return __raw_readl(map->map_priv_1 + ofs); +} + +static void epxa_copy_from(struct map_info *map, void *to, unsigned long from, ssize_t len) +{ + memcpy_fromio(to, map->map_priv_1 + from, len); +} + +static void epxa_write8(struct map_info *map, __u8 d, unsigned long adr) +{ + __raw_writeb(d, map->map_priv_1 + adr); + mb(); +} + +static void epxa_write16(struct map_info *map, __u16 d, unsigned long adr) +{ + __raw_writew(d, map->map_priv_1 + adr); + mb(); +} + +static void epxa_write32(struct map_info *map, __u32 d, unsigned long adr) +{ + __raw_writel(d, map->map_priv_1 + adr); + mb(); +} + +static void epxa_copy_to(struct map_info *map, unsigned long to, const void *from, ssize_t len) +{ + memcpy_toio(map->map_priv_1 + to, from, len); +} + +static struct map_info epxa_map = { + .name = "EPXA flash", + .size = FLASH_SIZE, + .buswidth = 2, + .read8 = epxa_read8, + .read16 = epxa_read16, + .read32 = epxa_read32, + .copy_from = epxa_copy_from, + .write8 = epxa_write8, + .write16 = epxa_write16, + .write32 = epxa_write32, + .copy_to = epxa_copy_to +}; + +static int __init epxa_mtd_init(void) +{ + int i; + + printk(KERN_NOTICE "%s flash device: %x at %x\n", BOARD_NAME, FLASH_SIZE, FLASH_START); + epxa_map.map_priv_1 = (unsigned long)ioremap_nocache(FLASH_START, FLASH_SIZE); + if (!epxa_map.map_priv_1) { + printk("Failed to ioremap %s flash\n",BOARD_NAME); + return -EIO; + } + + mymtd = do_map_probe("cfi_probe", &epxa_map); + if (!mymtd) { + iounmap((void *)epxa_map.map_priv_1); + return -ENXIO; + } + + mymtd->module = THIS_MODULE; + + /* Unlock the flash device. */ + if(mymtd->unlock){ + for (i=0; inumeraseregions;i++){ + int j; + for(j=0;jeraseregions[i].numblocks;j++){ + mymtd->unlock(mymtd,mymtd->eraseregions[i].offset + j * mymtd->eraseregions[i].erasesize,mymtd->eraseregions[i].erasesize); + } + } + } + +#ifdef CONFIG_MTD_REDBOOT_PARTS + nr_parts = parse_redboot_partitions(mymtd, &parts); + + if (nr_parts > 0) { + add_mtd_partitions(mymtd, parts, nr_parts); + return 0; + } +#endif +#ifdef CONFIG_MTD_AFS_PARTS + nr_parts = parse_afs_partitions(mymtd, &parts); + + if (nr_parts > 0) { + add_mtd_partitions(mymtd, parts, nr_parts); + return 0; + } +#endif + + /* No recognised partitioning schemes found - use defaults */ + nr_parts = epxa_default_partitions(mymtd, &parts); + if (nr_parts > 0) { + add_mtd_partitions(mymtd, parts, nr_parts); + return 0; + } + + /* If all else fails... */ + add_mtd_device(mymtd); + return 0; +} + +static void __exit epxa_mtd_cleanup(void) +{ + if (mymtd) { + if (nr_parts) + del_mtd_partitions(mymtd); + else + del_mtd_device(mymtd); + map_destroy(mymtd); + } + if (epxa_map.map_priv_1) { + iounmap((void *)epxa_map.map_priv_1); + epxa_map.map_priv_1 = 0; + } +} + + +/* + * This will do for now, once we decide which bootldr we're finally + * going to use then we'll remove this function and do it properly + * + * Partions are currently (as offsets from base of flash): + * 0x00000000 - 0x003FFFFF - bootloader (!) + * 0x00400000 - 0x00FFFFFF - Flashdisk + */ + +static int __init epxa_default_partitions(struct mtd_info *master, struct mtd_partition **pparts) +{ + struct mtd_partition *parts; + int ret; + int npartitions = 0; + char *names; + const char *name = "jffs"; + + printk("Using default partitions for %s\n",BOARD_NAME); + npartitions=1; + parts = kmalloc(npartitions*sizeof(*parts)+strlen(name)+1, GFP_KERNEL); + if (!parts) { + ret = -ENOMEM; + goto out; + } + memzero(parts,npartitions*sizeof(*parts)+strlen(name)); + + names = (char *)&parts[npartitions]; + parts[0].name = names; + names += strlen(name) + 1; + strcpy(parts[0].name, name); + +#ifdef CONFIG_EPXA10DB_R2 + parts[0].size = FLASH_SIZE-0x00400000; + parts[0].offset = 0x00400000; +#elif defined CONFIG_EPXA10DB_R3 + parts[0].size = 0x00800000; + parts[0].offset = 0x00800000; +#else + parts[0].size = FLASH_SIZE-0x00180000; + parts[0].offset = 0x00180000; +#endif + ret = npartitions; + + out: + *pparts = parts; + return ret; +} + + +module_init(epxa_mtd_init); +module_exit(epxa_mtd_cleanup); + +MODULE_AUTHOR("Clive Davies"); +MODULE_DESCRIPTION("Altera epxa mtd flash map"); +MODULE_LICENSE("GPL"); diff -urN orig/drivers/mtd/maps/epxa10db-flash.c linux/drivers/mtd/maps/epxa10db-flash.c --- orig/drivers/mtd/maps/epxa10db-flash.c Mon Aug 5 13:31:01 2002 +++ linux/drivers/mtd/maps/epxa10db-flash.c Thu Jan 1 01:00:00 1970 @@ -1,219 +0,0 @@ -/* - * Flash memory access on EPXA based devices - * - * (C) 2000 Nicolas Pitre - * Copyright (C) 2001 Altera Corporation - * Copyright (C) 2001 Red Hat, Inc. - * - * $Id: epxa10db-flash.c,v 1.2 2001/12/19 13:00:19 jskov Exp $ - * - * This program is free software; you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation; either version 2 of the License, or - * (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program; if not, write to the Free Software - * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA - */ - -#include -#include -#include -#include -#include -#include -#include -#include - -#include - -static int nr_parts = 0; -static struct mtd_partition *parts; - -static struct mtd_info *mymtd; - -extern int parse_redboot_partitions(struct mtd_info *, struct mtd_partition **); -static int epxa10db_default_partitions(struct mtd_info *master, struct mtd_partition **pparts); - -static __u8 epxa10db_read8(struct map_info *map, unsigned long ofs) -{ - return __raw_readb(map->map_priv_1 + ofs); -} - -static __u16 epxa10db_read16(struct map_info *map, unsigned long ofs) -{ - return __raw_readw(map->map_priv_1 + ofs); -} - -static __u32 epxa10db_read32(struct map_info *map, unsigned long ofs) -{ - return __raw_readl(map->map_priv_1 + ofs); -} - -static void epxa10db_copy_from(struct map_info *map, void *to, unsigned long from, ssize_t len) -{ - memcpy_fromio(to, (void *)(map->map_priv_1 + from), len); -} - -static void epxa10db_write8(struct map_info *map, __u8 d, unsigned long adr) -{ - __raw_writeb(d, map->map_priv_1 + adr); - mb(); -} - -static void epxa10db_write16(struct map_info *map, __u16 d, unsigned long adr) -{ - __raw_writew(d, map->map_priv_1 + adr); - mb(); -} - -static void epxa10db_write32(struct map_info *map, __u32 d, unsigned long adr) -{ - __raw_writel(d, map->map_priv_1 + adr); - mb(); -} - -static void epxa10db_copy_to(struct map_info *map, unsigned long to, const void *from, ssize_t len) -{ - memcpy_toio((void *)(map->map_priv_1 + to), from, len); -} - - - -static struct map_info epxa10db_map = { - name: "EPXA10DB flash", - size: FLASH_SIZE, - buswidth: 2, - read8: epxa10db_read8, - read16: epxa10db_read16, - read32: epxa10db_read32, - copy_from: epxa10db_copy_from, - write8: epxa10db_write8, - write16: epxa10db_write16, - write32: epxa10db_write32, - copy_to: epxa10db_copy_to -}; - - -static int __init epxa10db_mtd_init(void) -{ - int i; - printk(KERN_NOTICE "Epxa10db flash device: %x at %x\n", FLASH_SIZE, FLASH_START); - epxa10db_map.map_priv_1 = (unsigned long)ioremap(FLASH_START, FLASH_SIZE); - if (!epxa10db_map.map_priv_1) { - printk("Failed to ioremap Epxa10db flash\n"); - return -EIO; - } - - mymtd = do_map_probe("cfi_probe", &epxa10db_map); - if (!mymtd) { - iounmap((void *)epxa10db_map.map_priv_1); - return -ENXIO; - } - - mymtd->module = THIS_MODULE; - - /* Unlock the flash device. */ - for (i=0; inumeraseregions;i++){ - int j; - for(j=0;jeraseregions[i].numblocks;j++){ - mymtd->unlock(mymtd,mymtd->eraseregions[i].offset + j * mymtd->eraseregions[i].erasesize,4); - } - } - -#ifdef CONFIG_MTD_REDBOOT_PARTS - nr_parts = parse_redboot_partitions(mymtd, &parts); - - if (nr_parts > 0) { - add_mtd_partitions(mymtd, parts, nr_parts); - return 0; - } -#endif -#ifdef CONFIG_MTD_AFS_PARTS - nr_parts = parse_afs_partitions(mymtd, &parts); - - if (nr_parts > 0) { - add_mtd_partitions(mymtd, parts, nr_parts); - return 0; - } -#endif - - /* No recognised partitioning schemes found - use defaults */ - nr_parts = epxa10db_default_partitions(mymtd, &parts); - if (nr_parts > 0) { - add_mtd_partitions(mymtd, parts, nr_parts); - return 0; - } - - /* If all else fails... */ - add_mtd_device(mymtd); - return 0; -} - -static void __exit epxa10db_mtd_cleanup(void) -{ - if (mymtd) { - if (nr_parts) - del_mtd_partitions(mymtd); - else - del_mtd_device(mymtd); - map_destroy(mymtd); - } - if (epxa10db_map.map_priv_1) { - iounmap((void *)epxa10db_map.map_priv_1); - epxa10db_map.map_priv_1 = 0; - } -} - - -/* - * This will do for now, once we decide which bootldr we're finally - * going to use then we'll remove this function and do it properly - * - * Partions are currently (as offsets from base of flash): - * 0x00000000 - 0x003FFFFF - bootloader (!) - * 0x00400000 - 0x00FFFFFF - Flashdisk - */ - -static int __init epxa10db_default_partitions(struct mtd_info *master, struct mtd_partition **pparts) -{ - struct mtd_partition *parts; - int ret, i; - int npartitions = 0; - char *names; - const char *name = "jffs"; - - printk("Using default partitions for epxa10db\n"); - npartitions=1; - parts = kmalloc(npartitions*sizeof(*parts)+strlen(name), GFP_KERNEL); - if (!parts) { - ret = -ENOMEM; - goto out; - } - i=0; - names = (char *)&parts[npartitions]; - parts[i].name = names; - names += strlen(name) + 1; - strcpy(parts[i].name, name); - - parts[i].size = FLASH_SIZE-0x00400000; - parts[i].offset = 0x00400000; - parts[i].mask_flags = 0; - - out: - *pparts = parts; - return npartitions; -} - -module_init(epxa10db_mtd_init); -module_exit(epxa10db_mtd_cleanup); - -MODULE_AUTHOR("Clive Davies"); -MODULE_DESCRIPTION("Altera epxa10db mtd flash map"); -MODULE_LICENSE("GPL"); diff -urN orig/drivers/mtd/maps/fortunet.c linux/drivers/mtd/maps/fortunet.c --- orig/drivers/mtd/maps/fortunet.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/mtd/maps/fortunet.c Thu Oct 24 13:52:13 2002 @@ -0,0 +1,431 @@ +/* fortunet.c memory map + */ + +#include +#include +#include +#include +#include +#include +#include + +#define MAX_NUM_REGIONS 4 +#define MAX_NUM_PARTITIONS 8 + +#define DEF_WINDOW_ADDR_PHY 0x00000000 +#define DEF_WINDOW_SIZE 0x00800000 // 8 Mega Bytes + +#define MTD_FORTUNET_PK "MTD FortuNet: " + +#define MAX_NAME_SIZE 128 + +struct map_region +{ + int window_addr_phyical; + int altbuswidth; + struct map_info map_info; + struct mtd_info *mymtd; + struct mtd_partition parts[MAX_NUM_PARTITIONS]; + char map_name[MAX_NAME_SIZE]; + int num_probes; + char probe_name[MAX_NAME_SIZE*2]; + char parts_name[MAX_NUM_PARTITIONS][MAX_NAME_SIZE]; +}; + +static struct map_region map_regions[MAX_NUM_REGIONS]; +static int map_regions_set[MAX_NUM_REGIONS] = {0,0,0,0}; +static int map_regions_parts[MAX_NUM_REGIONS] = {0,0,0,0}; + + +__u8 fortunet_read8(struct map_info *map, unsigned long ofs) +{ + return *(__u8 *)(map->map_priv_1 + ofs); +} + +__u16 fortunet_read16(struct map_info *map, unsigned long ofs) +{ + return *(__u16 *)(map->map_priv_1 + ofs); +} + +__u32 fortunet_read32(struct map_info *map, unsigned long ofs) +{ + return *(__u32 *)(map->map_priv_1 + ofs); +} + +void fortunet_copy_from(struct map_info *map, void *to, unsigned long from, ssize_t len) +{ + memcpy(to, (void *)(map->map_priv_1 + from), len); +} + +void fortunet_write8(struct map_info *map, __u8 d, unsigned long adr) +{ + *(__u8 *)(map->map_priv_1 + adr) = d; +} + +void fortunet_write16(struct map_info *map, __u16 d, unsigned long adr) +{ + *(__u16 *)(map->map_priv_1 + adr) = d; +} + +void fortunet_write32(struct map_info *map, __u32 d, unsigned long adr) +{ + *(__u32 *)(map->map_priv_1 + adr) = d; +} + +void fortunet_copy_to(struct map_info *map, unsigned long to, const void *from, ssize_t len) +{ + memcpy((void *)(map->map_priv_1 + to), from, len); +} + +__u8 d_fortunet_read8(struct map_info *map, unsigned long ofs) +{ + __u8 ret; + ret = *(__u8 *)(map->map_priv_1 + ofs); + printk("Debug %s: %02X = u8 * %08X\n",map->name,ret,ofs); + return ret; +} + +__u16 d_fortunet_read16(struct map_info *map, unsigned long ofs) +{ + __u16 ret; + ret = *(__u16 *)(map->map_priv_1 + ofs); + printk("Debug %s: %04X = u16 * %08X\n",map->name,ret,ofs); + return ret; +} + +__u32 d_fortunet_read32(struct map_info *map, unsigned long ofs) +{ + __u32 ret; + ret = *(__u32 *)(map->map_priv_1 + ofs); + printk("Debug %s: %08X = u32 * %08X\n",map->name,ret,ofs); + return ret; +} + +void d_fortunet_copy_from(struct map_info *map, void *to, unsigned long from, ssize_t len) +{ + memcpy(to, (void *)(map->map_priv_1 + from), len); +} + +void d_fortunet_write8(struct map_info *map, __u8 d, unsigned long adr) +{ + printk("Debug %s: u8 * %08X = %02X\n",map->name,adr,d); + *(__u8 *)(map->map_priv_1 + adr) = d; +} + +void d_fortunet_write16(struct map_info *map, __u16 d, unsigned long adr) +{ + printk("Debug %s: u16 * %08X = %04X\n",map->name,adr,d); + *(__u16 *)(map->map_priv_1 + adr) = d; +} + +void d_fortunet_write32(struct map_info *map, __u32 d, unsigned long adr) +{ + printk("Debug %s: u32 * %08X = %08X\n",map->name,adr,d); + *(__u32 *)(map->map_priv_1 + adr) = d; +} + +void d_fortunet_copy_to(struct map_info *map, unsigned long to, const void *from, ssize_t len) +{ + memcpy((void *)(map->map_priv_1 + to), from, len); +} + +struct map_info default_map = { + size: DEF_WINDOW_SIZE, + buswidth: 4, + map_priv_2: 0, + read8: fortunet_read8, + read16: fortunet_read16, + read32: fortunet_read32, + copy_from: fortunet_copy_from, + write8: fortunet_write8, + write16: fortunet_write16, + write32: fortunet_write32, + copy_to: fortunet_copy_to +}; + +static char * __init get_string_option(char *dest,int dest_size,char *sor) +{ + if(!dest_size) + return sor; + dest_size--; + while(*sor) + { + if(*sor==',') + { + sor++; + break; + } + else if(*sor=='\"') + { + sor++; + while(*sor) + { + if(*sor=='\"') + { + sor++; + break; + } + *dest = *sor; + dest++; + sor++; + dest_size--; + if(!dest_size) + { + *dest = 0; + return sor; + } + } + } + else + { + *dest = *sor; + dest++; + sor++; + dest_size--; + if(!dest_size) + { + *dest = 0; + return sor; + } + } + } + *dest = 0; + return sor; +} + +static int __init MTD_New_Region(char *line) +{ + char string[MAX_NAME_SIZE]; + int params[7]; + int pos; + char *liner; + liner = get_options (get_string_option(string,sizeof(string),line),7,params); + if(params[0]<1) + { + printk(MTD_FORTUNET_PK "Bad paramters for MTD Region " + " name,region-number[,base,size,buswidth,altbuswidth," + "options,probe_name...]\n"); + return 1; + } + if((params[1]<0)||(params[1]>=MAX_NUM_REGIONS)) + { + printk(MTD_FORTUNET_PK "Bad region index of %d only have 0..%u regions\n", + params[1],MAX_NUM_REGIONS-1); + return 1; + } + memset(&map_regions[params[1]],0,sizeof(map_regions[params[1]])); + memcpy(&map_regions[params[1]].map_info, + &default_map,sizeof(map_regions[params[1]].map_info)); + map_regions_set[params[1]] = 1; + map_regions[params[1]].window_addr_phyical = DEF_WINDOW_ADDR_PHY; + map_regions[params[1]].altbuswidth = 2; + map_regions[params[1]].mymtd = NULL; + map_regions[params[1]].map_info.name = map_regions[params[1]].map_name; + map_regions[params[1]].num_probes = 1; + strcpy(map_regions[params[1]].probe_name,"cfi_probe"); + strcpy(map_regions[params[1]].map_info.name,string); + if(params[0]>1) + { + map_regions[params[1]].window_addr_phyical = params[2]; + } + if(params[0]>2) + { + map_regions[params[1]].map_info.size = params[3]; + } + if(params[0]>3) + { + map_regions[params[1]].map_info.buswidth = params[4]; + } + if(params[0]>4) + { + map_regions[params[1]].altbuswidth = params[5]; + } + if(params[0]>5) + { + map_regions[params[1]].map_info.map_priv_2 = params[6]; + if(params[6]&1) + { // go into debug mode + map_regions[params[1]].map_info.read8 = d_fortunet_read8; + map_regions[params[1]].map_info.read16 = d_fortunet_read16; + map_regions[params[1]].map_info.read32 = d_fortunet_read32; + map_regions[params[1]].map_info.copy_from = d_fortunet_copy_from; + map_regions[params[1]].map_info.write8 = d_fortunet_write8; + map_regions[params[1]].map_info.write16 = d_fortunet_write16; + map_regions[params[1]].map_info.write32 = d_fortunet_write32; + map_regions[params[1]].map_info.copy_to = d_fortunet_copy_to; + } + if(*liner) + { + liner = get_string_option(map_regions[params[1]].probe_name, + sizeof(map_regions[params[1]].probe_name),liner); + map_regions[params[1]].num_probes = 1; + pos = strlen(map_regions[params[1]].probe_name)+1; + while((*liner)&&(pos+2=MAX_NUM_REGIONS)) + { + printk(MTD_FORTUNET_PK "Bad region index of %d only have 0..%u regions\n", + params[1],MAX_NUM_REGIONS-1); + return 1; + } + if(map_regions_parts[params[1]]>=MAX_NUM_PARTITIONS) + { + printk(MTD_FORTUNET_PK "Out of space for partion in this region\n"); + return 1; + } + map_regions[params[1]].parts[map_regions_parts[params[1]]].name = + map_regions[params[1]]. parts_name[map_regions_parts[params[1]]]; + strcpy(map_regions[params[1]].parts[map_regions_parts[params[1]]].name,string); + map_regions[params[1]].parts[map_regions_parts[params[1]]].size = + params[2]; + map_regions[params[1]].parts[map_regions_parts[params[1]]].offset = + params[3]; + map_regions[params[1]].parts[map_regions_parts[params[1]]].mask_flags = 0; + map_regions_parts[params[1]]++; + return 1; +} + +__setup("MTD_Region=", MTD_New_Region); +__setup("MTD_Partion=", MTD_New_Partion); + +int __init init_fortunet(void) +{ + int ix,iy,iz,ipos; + for(iy=ix=0;ixmodule = THIS_MODULE; + add_mtd_partitions(map_regions[ix].mymtd, + map_regions[ix].parts,map_regions_parts[ix]); + } + else + { + printk(KERN_NOTICE MTD_FORTUNET_PK + "No Flash in this region (%x:%u:%u) probed with %s\n", + map_regions[ix].window_addr_phyical, + map_regions[ix].map_info.buswidth, + map_regions[ix].altbuswidth, + map_regions[ix].probe_name); + } + } + } + if(iy) + return 0; + return -ENXIO; +} + +static void __exit cleanup_fortunet(void) +{ + int ix; + for(ix=0;ix +#include +#include +#include +#include +#include +#include + +#ifdef CONFIG_MTD_PARTITIONS +#include +#endif + +#define WINDOW_ADDR0 0x00000000 /* physical properties of flash */ +#define WINDOW_SIZE0 0x00800000 +#define WINDOW_ADDR1 0x10000000 /* physical properties of flash */ +#define WINDOW_SIZE1 0x00800000 +#define NUM_FLASHBANKS 2 +#define BUSWIDTH 4 + +/* can be { "cfi_probe", "jedec_probe", "map_rom", 0 }; */ +#define PROBETYPES { "jedec_probe", 0 } + +#define MSG_PREFIX "impA7:" /* prefix for our printk()'s */ +#define MTDID "impa7-%d" /* for mtdparts= partitioning */ + +static struct mtd_info *impa7_mtd[NUM_FLASHBANKS] = { 0 }; + +__u8 impa7_read8(struct map_info *map, unsigned long ofs) +{ + return __raw_readb(map->map_priv_1 + ofs); +} + +__u16 impa7_read16(struct map_info *map, unsigned long ofs) +{ + return __raw_readw(map->map_priv_1 + ofs); +} + +__u32 impa7_read32(struct map_info *map, unsigned long ofs) +{ + return __raw_readl(map->map_priv_1 + ofs); +} + +#ifdef CFI_WORD_64 +__u64 impa7_read64(struct map_info *map, unsigned long ofs) +{ + return __raw_readll(map->map_priv_1 + ofs); +} +#endif + +void impa7_copy_from(struct map_info *map, void *to, unsigned long from, ssize_t len) +{ + memcpy_fromio(to, map->map_priv_1 + from, len); +} + +void impa7_write8(struct map_info *map, __u8 d, unsigned long adr) +{ + __raw_writeb(d, map->map_priv_1 + adr); + mb(); +} + +void impa7_write16(struct map_info *map, __u16 d, unsigned long adr) +{ + __raw_writew(d, map->map_priv_1 + adr); + mb(); +} + +void impa7_write32(struct map_info *map, __u32 d, unsigned long adr) +{ + __raw_writel(d, map->map_priv_1 + adr); + mb(); +} + +#ifdef CFI_WORD_64 +void impa7_write64(struct map_info *map, __u64 d, unsigned long adr) +{ + __raw_writell(d, map->map_priv_1 + adr); + mb(); +} +#endif + +void impa7_copy_to(struct map_info *map, unsigned long to, const void *from, ssize_t len) +{ + memcpy_toio(map->map_priv_1 + to, from, len); +} + +static struct map_info impa7_map[NUM_FLASHBANKS] = { + { + name: "impA7 NOR Flash Bank #0", + size: WINDOW_SIZE0, + buswidth: BUSWIDTH, + read8: impa7_read8, + read16: impa7_read16, + read32: impa7_read32, +#ifdef CFI_WORD_64 + read64: impa7_read64, +#endif + copy_from: impa7_copy_from, + write8: impa7_write8, + write16: impa7_write16, + write32: impa7_write32, +#ifdef CFI_WORD_64 + write64: impa7_write64, +#endif + copy_to: impa7_copy_to + }, + { + name: "impA7 NOR Flash Bank #1", + size: WINDOW_SIZE1, + buswidth: BUSWIDTH, + read8: impa7_read8, + read16: impa7_read16, + read32: impa7_read32, +#ifdef CFI_WORD_64 + read64: impa7_read64, +#endif + copy_from: impa7_copy_from, + write8: impa7_write8, + write16: impa7_write16, + write32: impa7_write32, +#ifdef CFI_WORD_64 + write64: impa7_write64, +#endif + copy_to: impa7_copy_to + }, +}; + +#ifdef CONFIG_MTD_PARTITIONS + +/* + * MTD partitioning stuff + */ +static struct mtd_partition static_partitions[] = +{ + { + name: "FileSystem", + size: 0x800000, + offset: 0x00000000 + }, +}; + +#define NB_OF(x) (sizeof (x) / sizeof (x[0])) + +#ifdef CONFIG_MTD_CMDLINE_PARTS +int parse_cmdline_partitions(struct mtd_info *master, + struct mtd_partition **pparts, + const char *mtd_id); +#endif + +#endif + +static int mtd_parts_nb = 0; +static struct mtd_partition *mtd_parts = 0; + +int __init init_impa7(void) +{ + static const char *rom_probe_types[] = PROBETYPES; + const char **type; + const char *part_type = 0; + int i; + static struct { u_long addr; u_long size; } pt[NUM_FLASHBANKS] = { + { WINDOW_ADDR0, WINDOW_SIZE0 }, + { WINDOW_ADDR1, WINDOW_SIZE1 }, + }; + char mtdid[10]; + int devicesfound = 0; + + for(i=0; imodule = THIS_MODULE; + add_mtd_device(impa7_mtd[i]); + devicesfound++; +#ifdef CONFIG_MTD_PARTITIONS +#ifdef CONFIG_MTD_CMDLINE_PARTS + sprintf(mtdid, MTDID, i); + mtd_parts_nb = parse_cmdline_partitions(impa7_mtd[i], + &mtd_parts, + mtdid); + if (mtd_parts_nb > 0) + part_type = "command line"; +#endif + if (mtd_parts_nb <= 0) + { + mtd_parts = static_partitions; + mtd_parts_nb = NB_OF(static_partitions); + part_type = "static"; + } + if (mtd_parts_nb <= 0) + { + printk(KERN_NOTICE MSG_PREFIX + "no partition info available\n"); + } + else + { + printk(KERN_NOTICE MSG_PREFIX + "using %s partition definition\n", + part_type); + add_mtd_partitions(impa7_mtd[i], + mtd_parts, mtd_parts_nb); + } +#endif + } + else + iounmap((void *)impa7_map[i].map_priv_1); + } + return devicesfound == 0 ? -ENXIO : 0; +} + +static void __exit cleanup_impa7(void) +{ + int i; + for (i=0; i"); +MODULE_DESCRIPTION("MTD map driver for implementa impA7"); diff -urN orig/drivers/mtd/maps/neponset-flash.c linux/drivers/mtd/maps/neponset-flash.c --- orig/drivers/mtd/maps/neponset-flash.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/mtd/maps/neponset-flash.c Sat Mar 16 15:56:03 2002 @@ -0,0 +1,109 @@ +/* + * Flash memory access on SA11x0 based devices + * + * (C) 2000 Nicolas Pitre + * + * $Id: neponset-flash.c,v 1.18 2001/07/14 00:59:17 thockin Exp $ + */ + +#include +#include +#include +#include + +#include +#include +#include + +#include +#include +#include + +static __u8 read8(struct map_info *map, unsigned long ofs) +{ + return readb(map->map_priv_1 + ofs); +} + +static __u16 read16(struct map_info *map, unsigned long ofs) +{ + return readw(map->map_priv_1 + ofs); +} + +static __u32 read32(struct map_info *map, unsigned long ofs) +{ + return readl(map->map_priv_1 + ofs); +} + +static void copy_from(struct map_info *map, void *to, unsigned long from, ssize_t len) +{ + memcpy_fromio(to, map->map_priv_1 + from, len); +} + +static void write8(struct map_info *map, __u8 d, unsigned long adr) +{ + writeb(d, map->map_priv_1 + adr); +} + +static void write16(struct map_info *map, __u16 d, unsigned long adr) +{ + writew(d, map->map_priv_1 + adr); +} + +static void write32(struct map_info *map, __u32 d, unsigned long adr) +{ + writel(d, map->map_priv_1 + adr); +} + +static void copy_to(struct map_info *map, unsigned long to, const void *from, ssize_t len) +{ + memcpy_toio(map->map_priv_1 + to, from, len); +} + +#define MAX_SZ (32 * 1024 * 1024) + +static struct map_info neponset_map = { + name: "Neponset", + size: MAX_SZ, + buswidth: 4, + read8: read8, + read16: read16, + read32: read32, + copy_from: copy_from, + write8: write8, + write16: write16, + write32: write32, + copy_to: copy_to, +}; + +extern int parse_redboot_partitions(struct mtd_info *master, struct mtd_partition **pparts); +extern int parse_bootldr_partitions(struct mtd_info *master, struct mtd_partition **pparts); + +static struct mtd_info *neponset_mtd; + +int __init neponset_mtd_init(void) +{ + if (!machine_is_assabet() || !machine_has_neponset()) + return -ENODEV; + + neponset_map.map_priv_1 = (unsigned int)ioremap(0x08000000, MAX_SZ); + if (!neponset_map.map_priv_1) + return -ENOMEM; + + neponset_mtd = do_map_probe("cfi_probe", &neponset_map); + if (!neponset_mtd) + return -ENXIO; + neponset_mtd->module = THIS_MODULE; + add_mtd_device(neponset_mtd); + return 0; +} + +static void __exit neponset_mtd_cleanup(void) +{ + if (neponset_mtd) + map_destroy(neponset_mtd); + if (neponset_map.map_priv_1) + iounmap((void *)neponset_map.map_priv_1); +} + +module_init(neponset_mtd_init); +module_exit(neponset_mtd_cleanup); diff -urN orig/drivers/mtd/maps/pb1xxx-flash.c linux/drivers/mtd/maps/pb1xxx-flash.c --- orig/drivers/mtd/maps/pb1xxx-flash.c Mon Aug 5 13:31:01 2002 +++ linux/drivers/mtd/maps/pb1xxx-flash.c Sun Apr 14 13:02:44 2002 @@ -3,7 +3,7 @@ * * (C) 2001 Pete Popov * - * $Id: pb1xxx-flash.c,v 1.2 2002/02/14 19:36:45 ppopov Exp $ + * $Id: pb1xxx-flash.c,v 1.3 2002/04/12 20:48:31 ppopov Exp $ */ #include @@ -127,7 +127,7 @@ } }; -#elif defined(CONFIG_MIPS_PB1500) +#elif defined(CONFIG_MIPS_PB1500) || defined(CONFIG_MIPS_PB1100) static unsigned char flash_buswidth = 4; #if defined(CONFIG_MTD_PB1500_BOOT) && defined(CONFIG_MTD_PB1500_USER) diff -urN orig/drivers/mtd/maps/pci.c linux/drivers/mtd/maps/pci.c --- orig/drivers/mtd/maps/pci.c Mon Aug 5 13:31:01 2002 +++ linux/drivers/mtd/maps/pci.c Thu Jan 3 21:00:19 2002 @@ -361,7 +361,7 @@ static struct pci_driver mtd_pci_driver = { name: "MTD PCI", probe: mtd_pci_probe, - remove: __devexit_p(mtd_pci_remove), + remove: mtd_pci_remove, id_table: mtd_pci_ids, }; diff -urN orig/drivers/mtd/maps/physmap.c linux/drivers/mtd/maps/physmap.c --- orig/drivers/mtd/maps/physmap.c Sun Oct 14 20:53:09 2001 +++ linux/drivers/mtd/maps/physmap.c Fri May 31 11:48:56 2002 @@ -1,5 +1,5 @@ /* - * $Id: physmap.c,v 1.15 2001/10/02 15:05:14 dwmw2 Exp $ + * $Id: physmap.c,v 1.20 2002/04/30 09:14:50 mag Exp $ * * Normal mappings of chips in physical memory */ @@ -12,6 +12,9 @@ #include #include +#ifdef CONFIG_MTD_PARTITIONS +#include +#endif #define WINDOW_ADDR CONFIG_MTD_PHYSMAP_START #define WINDOW_SIZE CONFIG_MTD_PHYSMAP_LEN @@ -34,6 +37,13 @@ return __raw_readl(map->map_priv_1 + ofs); } +#ifdef CFI_WORD_64 +__u64 physmap_read64(struct map_info *map, unsigned long ofs) +{ + return __raw_readll(map->map_priv_1 + ofs); +} +#endif + void physmap_copy_from(struct map_info *map, void *to, unsigned long from, ssize_t len) { memcpy_fromio(to, map->map_priv_1 + from, len); @@ -57,6 +67,14 @@ mb(); } +#ifdef CFI_WORD_64 +void physmap_write64(struct map_info *map, __u64 d, unsigned long adr) +{ + __raw_writell(d, map->map_priv_1 + adr); + mb(); +} +#endif + void physmap_copy_to(struct map_info *map, unsigned long to, const void *from, ssize_t len) { memcpy_toio(map->map_priv_1 + to, from, len); @@ -69,15 +87,31 @@ read8: physmap_read8, read16: physmap_read16, read32: physmap_read32, +#ifdef CFI_WORD_64 + read64: physmap_read64, +#endif copy_from: physmap_copy_from, write8: physmap_write8, write16: physmap_write16, write32: physmap_write32, +#ifdef CFI_WORD_64 + write64: physmap_write64, +#endif copy_to: physmap_copy_to }; +#ifdef CONFIG_MTD_PARTITIONS +#ifdef CONFIG_MTD_CMDLINE_PARTS +static struct mtd_partition *mtd_parts = 0; +static int mtd_parts_nb = 0; +#endif +#endif + int __init init_physmap(void) { + static const char *rom_probe_types[] = { "cfi_probe", "jedec_probe", "map_rom", 0 }; + const char **type; + printk(KERN_NOTICE "physmap flash device: %x at %x\n", WINDOW_SIZE, WINDOW_ADDR); physmap_map.map_priv_1 = (unsigned long)ioremap(WINDOW_ADDR, WINDOW_SIZE); @@ -85,11 +119,28 @@ printk("Failed to ioremap\n"); return -EIO; } - mymtd = do_map_probe("cfi_probe", &physmap_map); + + mymtd = 0; + type = rom_probe_types; + for(; !mymtd && *type; type++) { + mymtd = do_map_probe(*type, &physmap_map); + } if (mymtd) { mymtd->module = THIS_MODULE; add_mtd_device(mymtd); +#ifdef CONFIG_MTD_PARTITIONS +#ifdef CONFIG_MTD_CMDLINE_PARTS + mtd_parts_nb = parse_cmdline_partitions(mymtd, &mtd_parts, + "phys"); + if (mtd_parts_nb > 0) + { + printk(KERN_NOTICE + "Using command line partition definition\n"); + add_mtd_partitions (mymtd, mtd_parts, mtd_parts_nb); + } +#endif +#endif return 0; } diff -urN orig/drivers/mtd/maps/redwood.c linux/drivers/mtd/maps/redwood.c --- orig/drivers/mtd/maps/redwood.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/mtd/maps/redwood.c Fri May 3 00:19:40 2002 @@ -0,0 +1,173 @@ +/* + * $Id: + * + * redwood.c - mapper for IBM Redwood-4/5 board. + * + * Copyright 2001 MontaVista Softare Inc. + * + * This program is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License as published by the + * Free Software Foundation; either version 2 of the License, or (at your + * option) any later version. + * + * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED + * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF + * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN + * NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT + * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF + * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON + * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF + * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + * + * You should have received a copy of the GNU General Public License along + * with this program; if not, write to the Free Software Foundation, Inc., + * 675 Mass Ave, Cambridge, MA 02139, USA. + * + * History: 12/17/2001 - Armin + * migrated to use do_map_probe + * + */ + +#include +#include +#include + +#include +#include +#include + +#include + +#define WINDOW_ADDR 0xffc00000 +#define WINDOW_SIZE 0x00400000 + +__u8 redwood_flash_read8(struct map_info *map, unsigned long ofs) +{ + return *(__u8 *)(map->map_priv_1 + ofs); +} + +__u16 redwood_flash_read16(struct map_info *map, unsigned long ofs) +{ + return *(__u16 *)(map->map_priv_1 + ofs); +} + +__u32 redwood_flash_read32(struct map_info *map, unsigned long ofs) +{ + return *(volatile unsigned int *)(map->map_priv_1 + ofs); +} + +void redwood_flash_copy_from(struct map_info *map, void *to, + unsigned long from, ssize_t len) +{ + memcpy(to, (void *)(map->map_priv_1 + from), len); +} + +void redwood_flash_write8(struct map_info *map, __u8 d, unsigned long adr) +{ + *(__u8 *)(map->map_priv_1 + adr) = d; +} + +void redwood_flash_write16(struct map_info *map, __u16 d, unsigned long adr) +{ + *(__u16 *)(map->map_priv_1 + adr) = d; +} + +void redwood_flash_write32(struct map_info *map, __u32 d, unsigned long adr) +{ + *(__u32 *)(map->map_priv_1 + adr) = d; +} + +void redwood_flash_copy_to(struct map_info *map, unsigned long to, + const void *from, ssize_t len) +{ + memcpy((void *)(map->map_priv_1 + to), from, len); +} + +struct map_info redwood_flash_map = { + name: "IBM Redwood", + size: WINDOW_SIZE, + buswidth: 2, + read8: redwood_flash_read8, + read16: redwood_flash_read16, + read32: redwood_flash_read32, + copy_from: redwood_flash_copy_from, + write8: redwood_flash_write8, + write16: redwood_flash_write16, + write32: redwood_flash_write32, + copy_to: redwood_flash_copy_to +}; + + +static struct mtd_partition redwood_flash_partitions[] = { + { + name: "Redwood OpenBIOS Vital Product Data", + offset: 0, + size: 0x10000, + mask_flags: MTD_WRITEABLE /* force read-only */ + }, + { + name: "Redwood kernel", + offset: 0x10000, + size: 0x200000 - 0x10000 + }, + { + name: "Redwood OpenBIOS non-volatile storage", + offset: 0x200000, + size: 0x10000, + mask_flags: MTD_WRITEABLE /* force read-only */ + }, + { + name: "Redwood filesystem", + offset: 0x210000, + size: 0x200000 - (0x10000 + 0x20000) + }, + { + name: "Redwood OpenBIOS", + offset: 0x3e0000, + size: 0x20000, + mask_flags: MTD_WRITEABLE /* force read-only */ + } +}; +#define NUM_REDWOOD_FLASH_PARTITIONS \ + (sizeof(redwood_flash_partitions)/sizeof(redwood_flash_partitions[0])) + +static struct mtd_info *redwood_mtd; + +int __init init_redwood_flash(void) +{ + printk(KERN_NOTICE "redwood: flash mapping: %x at %x\n", + WINDOW_SIZE, WINDOW_ADDR); + + redwood_flash_map.map_priv_1 = + (unsigned long)ioremap(WINDOW_ADDR, WINDOW_SIZE); + + if (!redwood_flash_map.map_priv_1) { + printk("init_redwood_flash: failed to ioremap\n"); + return -EIO; + } + + redwood_mtd = do_map_probe("cfi_probe",&redwood_flash_map); + + if (redwood_mtd) { + redwood_mtd->module = THIS_MODULE; + return add_mtd_partitions(redwood_mtd, + redwood_flash_partitions, + NUM_REDWOOD_FLASH_PARTITIONS); + } + + return -ENXIO; +} + +static void __exit cleanup_redwood_flash(void) +{ + if (redwood_mtd) { + del_mtd_partitions(redwood_mtd); + iounmap((void *)redwood_flash_map.map_priv_1); + map_destroy(redwood_mtd); + } +} + +module_init(init_redwood_flash); +module_exit(cleanup_redwood_flash); diff -urN orig/drivers/mtd/maps/sa1100-flash.c linux/drivers/mtd/maps/sa1100-flash.c --- orig/drivers/mtd/maps/sa1100-flash.c Mon Aug 5 13:31:01 2002 +++ linux/drivers/mtd/maps/sa1100-flash.c Thu Feb 27 23:20:10 2003 @@ -3,7 +3,7 @@ * * (C) 2000 Nicolas Pitre * - * $Id: sa1100-flash.c,v 1.26 2002/03/13 16:30:44 rmk Exp $ + * $Id: sa1100-flash.c,v 1.28 2002/05/07 13:48:38 abz Exp $ */ #include @@ -97,6 +97,32 @@ * entries. Thanks. */ +#ifdef CONFIG_SA1100_ADSAGC +#define ADSAGC_FLASH_SIZE 0x02000000 +static struct mtd_partition adsagc_partitions[] = { + { + name: "bootROM", + size: 0x80000, + offset: 0, + mask_flags: MTD_WRITEABLE, /* force read-only */ + }, { + name: "zImage", + size: 0x100000, + offset: MTDPART_OFS_APPEND, + mask_flags: MTD_WRITEABLE, /* force read-only */ + }, { + name: "ramdisk.gz", + size: 0x300000, + offset: MTDPART_OFS_APPEND, + mask_flags: MTD_WRITEABLE, /* force read-only */ + }, { + name: "User FS", + size: MTDPART_SIZ_FULL, + offset: MTDPART_OFS_APPEND, + } +}; +#endif + #ifdef CONFIG_SA1100_ADSBITSY #define ADSBITSY_FLASH_SIZE 0x02000000 static struct mtd_partition adsbitsy_partitions[] = { @@ -123,6 +149,32 @@ }; #endif +#ifdef CONFIG_SA1100_ADSBITSYPLUS +#define ADSBITSYPLUS_FLASH_SIZE 0x02000000 +static struct mtd_partition adsbitsyplus_partitions[] = { + { + name: "bootROM", + size: 0x80000, + offset: 0, + mask_flags: MTD_WRITEABLE, /* force read-only */ + }, { + name: "zImage", + size: 0x100000, + offset: MTDPART_OFS_APPEND, + mask_flags: MTD_WRITEABLE, /* force read-only */ + }, { + name: "ramdisk.gz", + size: 0x300000, + offset: MTDPART_OFS_APPEND, + mask_flags: MTD_WRITEABLE, /* force read-only */ + }, { + name: "User FS", + size: MTDPART_SIZ_FULL, + offset: MTDPART_OFS_APPEND, + } +}; +#endif + #ifdef CONFIG_SA1100_ASSABET /* Phase 4 Assabet has two 28F160B3 flash parts in bank 0: */ #define ASSABET4_FLASH_SIZE 0x00400000 @@ -393,27 +445,23 @@ static struct mtd_partition frodo_partitions[] = { { - name: "bootloader", + name: "Boot Loader", size: 0x00040000, - offset: 0x00000000, - mask_flags: MTD_WRITEABLE + offset: 0x00000000 }, { - name: "bootloader params", + name: "Parameter Block", size: 0x00040000, - offset: MTDPART_OFS_APPEND, - mask_flags: MTD_WRITEABLE + offset: MTDPART_OFS_APPEND }, { - name: "kernel", + name: "Linux Kernel", size: 0x00100000, - offset: MTDPART_OFS_APPEND, - mask_flags: MTD_WRITEABLE + offset: MTDPART_OFS_APPEND }, { - name: "ramdisk", - size: 0x00400000, - offset: MTDPART_OFS_APPEND, - mask_flags: MTD_WRITEABLE + name: "Ramdisk", + size: 0x00680000, + offset: MTDPART_OFS_APPEND }, { - name: "file system", + name: "Flash File System", size: MTDPART_SIZ_FULL, offset: MTDPART_OFS_APPEND } @@ -442,7 +490,7 @@ #endif #ifdef CONFIG_SA1100_GRAPHICSMASTER -#define GRAPHICSMASTER_FLASH_SIZE 0x01000000 +#define GRAPHICSMASTER_FLASH_SIZE 0x02000000 static struct mtd_partition graphicsmaster_partitions[] = { { name: "zImage", @@ -511,6 +559,38 @@ } #endif +#ifdef CONFIG_SA1100_HACKKIT +#define HACKKIT_FLASH_SIZE 0x01000000 +static struct mtd_partition hackkit_partitions[] = { + { + name: "BLOB", + size: 0x00040000, + offset: 0x00000000, + mask_flags: MTD_WRITEABLE, /* force read-only */ + }, { + name: "config", + size: 0x00040000, + offset: MTDPART_OFS_APPEND, + }, { + name: "kernel", + size: 0x00100000, + offset: MTDPART_OFS_APPEND, + }, { + name: "initrd", + size: 0x00180000, + offset: MTDPART_OFS_APPEND, + }, { + name: "rootfs", + size: 0x700000, + offset: MTDPART_OFS_APPEND, + }, { + name: "data", + size: MTDPART_SIZ_FULL, + offset: MTDPART_OFS_APPEND, + } +}; +#endif + #ifdef CONFIG_SA1100_HUW_WEBPANEL #define HUW_WEBPANEL_FLASH_SIZE 0x01000000 static struct mtd_partition huw_webpanel_partitions[] = { @@ -559,12 +639,12 @@ offset: 0x00540000, }, { name: "JORNADA720 usr local", - size: 0 /* will expand to the end of the flash */ + size: 0, /* will expand to the end of the flash */ offset: 0x00d00000, } }; -static void jornada720_set_vpp(int vpp) +static void jornada720_set_vpp(struct map_info *map, int vpp) { if (vpp) PPSR |= 0x80; @@ -575,6 +655,27 @@ #endif +#ifdef CONFIG_SA1100_NANOENGINE +/* nanoEngine has one 28F320B3B Flash part in bank 0: */ +#define NANOENGINE_FLASH_SIZE 0x00400000 +static struct mtd_partition nanoengine_partitions[] = { + { + name: "nanoEngine boot firmware and parameter table", + size: 0x00010000, /* 32K */ + offset: 0x00000000, + mask_flags: MTD_WRITEABLE, /* force read-only */ + },{ + name: "kernel/initrd reserved", + size: 0x002f0000, + offset: 0x00010000, + },{ + name: "experimental filesystem allocation", + size: 0x00100000, + offset: 0x00300000, + } +}; +#endif + #ifdef CONFIG_SA1100_PANGOLIN #define PANGOLIN_FLASH_SIZE 0x04000000 static struct mtd_partition pangolin_partitions[] = { @@ -703,6 +804,32 @@ }; #endif /* CONFIG_SA1100_SIMPAD */ +#ifdef CONFIG_SA1100_SIMPUTER +#define SIMPUTER_FLASH_SIZE 0x02000000 +static struct mtd_partition simputer_partitions[] = { + { + name: "blob+logo", + offset: 0, + size: 0x00040000 + }, + { + name: "kernel", + offset: MTDPART_OFS_APPEND, + size: 0x000C0000 + }, + { + name: "/(cramfs)", + offset: MTDPART_OFS_APPEND, + size: 0x00200000 + }, + { + name: "/usr/local(jffs2)", + offset: MTDPART_OFS_APPEND, + size: MTDPART_SIZ_FULL /* expand till the end */ + } +}; +#endif + #ifdef CONFIG_SA1100_STORK #define STORK_FLASH_SIZE 0x02000000 static struct mtd_partition stork_partitions[] = { @@ -770,7 +897,7 @@ #endif extern int parse_redboot_partitions(struct mtd_info *master, struct mtd_partition **pparts); -extern int parse_bootldr_partitions(struct mtd_info *master, struct mtd_partition **pparts); +extern int parse_cmdline_partitions(struct mtd_info *master, struct mtd_partition **pparts, char *); static struct mtd_partition *parsed_parts; static struct mtd_info *mymtd; @@ -791,6 +918,14 @@ */ part_type = "static"; +#ifdef CONFIG_SA1100_ADSAGC + if (machine_is_adsagc()) { + parts = adsagc_partitions; + nb_parts = ARRAY_SIZE(adsagc_partitions); + sa1100_map.size = ADSAGC_FLASH_SIZE; + sa1100_map.buswidth = (MSC1 & MSC_RBW) ? 2 : 4; + } +#endif #ifdef CONFIG_SA1100_ADSBITSY if (machine_is_adsbitsy()) { parts = adsbitsy_partitions; @@ -799,6 +934,14 @@ sa1100_map.buswidth = (MSC1 & MSC_RBW) ? 2 : 4; } #endif +#ifdef CONFIG_SA1100_ADSBITSYPLUS + if (machine_is_adsbitsyplus()) { + parts = adsbitsyplus_partitions; + nb_parts = ARRAY_SIZE(adsbitsyplus_partitions); + sa1100_map.size = ADSBITSYPLUS_FLASH_SIZE; + sa1100_map.buswidth = (MSC1 & MSC_RBW) ? 2 : 4; + } +#endif #ifdef CONFIG_SA1100_ASSABET if (machine_is_assabet()) { parts = assabet_partitions; @@ -848,6 +991,7 @@ sa1100_map.size = FRODO_FLASH_SIZE; base = 0x00000000; } +#endif #ifdef CONFIG_SA1100_GRAPHICSCLIENT if (machine_is_graphicsclient()) { parts = graphicsclient_partitions; @@ -872,6 +1016,13 @@ sa1100_map.set_vpp = h3600_set_vpp; } #endif +#ifdef CONFIG_SA1100_HACKKIT + if (machine_is_hackkit()) { + parts = hackkit_partitions; + nb_parts = ARRAY_SIZE(hackkit_partitions); + sa1100_map.size = HACKKIT_FLASH_SIZE; + } +#endif #ifdef CONFIG_SA1100_HUW_WEBPANEL if (machine_is_huw_webpanel()) { parts = huw_webpanel_partitions; @@ -887,6 +1038,13 @@ sa1100_map.set_vpp = jornada720_set_vpp; } #endif +#ifdef CONFIG_SA1100_NANOENGINE + if (machine_is_nanoengine()) { + parts = nanoengine_partitions; + nb_parts = ARRAY_SIZE(nanoengine_partitions); + sa1100_map.size = NANOENGINE_FLASH_SIZE; + } +#endif #ifdef CONFIG_SA1100_PANGOLIN if (machine_is_pangolin()) { parts = pangolin_partitions; @@ -922,6 +1080,13 @@ sa1100_map.size = SIMPAD_FLASH_SIZE; } #endif +#ifdef CONFIG_SA1100_SIMPUTER + if (machine_is_simputer()) { + parts = simputer_partitions; + nb_parts = ARRAY_SIZE(simputer_partitions); + sa1100_map.size = SIMPUTER_FLASH_SIZE; + } +#endif #ifdef CONFIG_SA1100_STORK if (machine_is_stork()) { parts = stork_partitions; @@ -956,7 +1121,9 @@ * specific machine settings might have been set above. */ printk(KERN_NOTICE "SA1100 flash: probing %d-bit flash bus\n", sa1100_map.buswidth*8); - mymtd = do_map_probe("cfi_probe", &sa1100_map); + mymtd = do_map_probe("jedec_probe", &sa1100_map); + if (!mymtd) + mymtd = do_map_probe("cfi_probe", &sa1100_map); ret = -ENXIO; if (!mymtd) goto out_err; @@ -975,11 +1142,11 @@ } } #endif -#ifdef CONFIG_MTD_BOOTLDR_PARTS +#ifdef CONFIG_MTD_CMDLINE_PARTS if (parsed_nr_parts == 0) { - int ret = parse_bootldr_partitions(mymtd, &parsed_parts); + int ret = parse_cmdline_partitions(mymtd, &parsed_parts, "sa1100"); if (ret > 0) { - part_type = "Compaq bootldr"; + part_type = "Command Line"; parsed_nr_parts = ret; } } diff -urN orig/drivers/mtd/maps/tqm8xxl.c linux/drivers/mtd/maps/tqm8xxl.c --- orig/drivers/mtd/maps/tqm8xxl.c Sun Oct 14 20:53:09 2001 +++ linux/drivers/mtd/maps/tqm8xxl.c Thu Jun 20 20:04:20 2002 @@ -2,7 +2,7 @@ * Handle mapping of the flash memory access routines * on TQM8xxL based devices. * - * $Id: tqm8xxl.c,v 1.3 2001/10/02 15:05:14 dwmw2 Exp $ + * $Id: tqm8xxl.c,v 1.4 2002/06/20 13:41:20 mag Exp $ * * based on rpxlite.c * @@ -91,20 +91,6 @@ memcpy_toio((void *)(map->map_priv_1 + to), from, len); } -struct map_info tqm8xxl_map = { - name: "TQM8xxL", - //size: WINDOW_SIZE, - buswidth: 4, - read8: tqm8xxl_read8, - read16: tqm8xxl_read16, - read32: tqm8xxl_read32, - copy_from: tqm8xxl_copy_from, - write8: tqm8xxl_write8, - write16: tqm8xxl_write16, - write32: tqm8xxl_write32, - copy_to: tqm8xxl_copy_to -}; - /* * Here are partition information for all known TQM8xxL series devices. * See include/linux/mtd/partitions.h for definition of the mtd_partition @@ -204,8 +190,9 @@ goto error_mem; } memset((void *)map_banks[idx]->name, 0, 16); - + sprintf(map_banks[idx]->name, "TQM8xxL%d", idx); + map_banks[idx]->size = flash_size; map_banks[idx]->buswidth = 4; map_banks[idx]->read8 = tqm8xxl_read8; map_banks[idx]->read16 = tqm8xxl_read16; diff -urN orig/drivers/mtd/mtdblock.c linux/drivers/mtd/mtdblock.c --- orig/drivers/mtd/mtdblock.c Mon Aug 5 13:31:02 2002 +++ linux/drivers/mtd/mtdblock.c Mon Aug 5 15:53:04 2002 @@ -1,7 +1,7 @@ /* * Direct MTD block device access * - * $Id: mtdblock.c,v 1.51 2001/11/20 11:42:33 dwmw2 Exp $ + * $Id: mtdblock.c,v 1.54 2002/05/03 16:46:24 dwmw2 Exp $ * * 02-nov-2000 Nicolas Pitre Added read-modify-write with cache */ @@ -22,15 +22,6 @@ #define DEVICE_OFF(device) #define DEVICE_NO_RANDOM #include -/* for old kernels... */ -#ifndef QUEUE_EMPTY -#define QUEUE_EMPTY (!CURRENT) -#endif -#if LINUX_VERSION_CODE < 0x20300 -#define QUEUE_PLUGGED (blk_dev[MAJOR_NR].plug_tq.sync) -#else -#define QUEUE_PLUGGED (blk_dev[MAJOR_NR].request_queue.plugged) -#endif #ifdef CONFIG_DEVFS_FS #include @@ -56,6 +47,8 @@ } *mtdblks[MAX_MTD_DEVICES]; static spinlock_t mtdblks_lock; +/* this lock is used just in kernels >= 2.5.x */ +static spinlock_t mtdblock_lock; static int mtd_sizes[MAX_MTD_DEVICES]; static int mtd_blksizes[MAX_MTD_DEVICES]; @@ -290,7 +283,7 @@ if (!inode) return -EINVAL; - dev = MINOR(inode->i_rdev); + dev = minor(inode->i_rdev); if (dev >= MAX_MTD_DEVICES) return -EINVAL; @@ -383,7 +376,7 @@ if (inode == NULL) release_return(-ENODEV); - dev = MINOR(inode->i_rdev); + dev = minor(inode->i_rdev); mtdblk = mtdblks[dev]; down(&mtdblk->cache_sem); @@ -413,10 +406,10 @@ /* * This is a special request_fn because it is executed in a process context - * to be able to sleep independently of the caller. The io_request_lock - * is held upon entry and exit. - * The head of our request queue is considered active so there is no need - * to dequeue requests before we are done. + * to be able to sleep independently of the caller. The + * io_request_lock (for <2.5) or queue_lock (for >=2.5) is held upon entry + * and exit. The head of our request queue is considered active so there is + * no need to dequeue requests before we are done. */ static void handle_mtdblock_request(void) { @@ -427,18 +420,21 @@ for (;;) { INIT_REQUEST; req = CURRENT; - spin_unlock_irq(&io_request_lock); - mtdblk = mtdblks[MINOR(req->rq_dev)]; + spin_unlock_irq(QUEUE_LOCK(QUEUE)); + mtdblk = mtdblks[minor(req->rq_dev)]; res = 0; - if (MINOR(req->rq_dev) >= MAX_MTD_DEVICES) + if (minor(req->rq_dev) >= MAX_MTD_DEVICES) panic(__FUNCTION__": minor out of bound"); + if (!IS_REQ_CMD(req)) + goto end_req; + if ((req->sector + req->current_nr_sectors) > (mtdblk->mtd->size >> 9)) goto end_req; // Handle the request - switch (req->cmd) + switch (rq_data_dir(req)) { int err; @@ -469,7 +465,7 @@ } end_req: - spin_lock_irq(&io_request_lock); + spin_lock_irq(QUEUE_LOCK(QUEUE)); end_request(res); } } @@ -483,34 +479,28 @@ struct task_struct *tsk = current; DECLARE_WAITQUEUE(wait, tsk); - tsk->session = 1; - tsk->pgrp = 1; /* we might get involved when memory gets low, so use PF_MEMALLOC */ tsk->flags |= PF_MEMALLOC; strcpy(tsk->comm, "mtdblockd"); - tsk->tty = NULL; spin_lock_irq(&tsk->sigmask_lock); sigfillset(&tsk->blocked); - recalc_sigpending(tsk); + recalc_sigpending(); spin_unlock_irq(&tsk->sigmask_lock); - exit_mm(tsk); - exit_files(tsk); - exit_sighand(tsk); - exit_fs(tsk); + daemonize(); while (!leaving) { add_wait_queue(&thr_wq, &wait); set_current_state(TASK_INTERRUPTIBLE); - spin_lock_irq(&io_request_lock); + spin_lock_irq(QUEUE_LOCK(QUEUE)); if (QUEUE_EMPTY || QUEUE_PLUGGED) { - spin_unlock_irq(&io_request_lock); + spin_unlock_irq(QUEUE_LOCK(QUEUE)); schedule(); remove_wait_queue(&thr_wq, &wait); } else { remove_wait_queue(&thr_wq, &wait); set_current_state(TASK_RUNNING); handle_mtdblock_request(); - spin_unlock_irq(&io_request_lock); + spin_unlock_irq(QUEUE_LOCK(QUEUE)); } } @@ -536,7 +526,7 @@ { struct mtdblk_dev *mtdblk; - mtdblk = mtdblks[MINOR(inode->i_rdev)]; + mtdblk = mtdblks[minor(inode->i_rdev)]; #ifdef PARANOIA if (!mtdblk) @@ -624,6 +614,9 @@ int i; spin_lock_init(&mtdblks_lock); + /* this lock is used just in kernels >= 2.5.x */ + spin_lock_init(&mtdblock_lock); + #ifdef CONFIG_DEVFS_FS if (devfs_register_blkdev(MTD_BLOCK_MAJOR, DEVICE_NAME, &mtd_fops)) { @@ -651,8 +644,9 @@ /* Allow the block size to default to BLOCK_SIZE. */ blksize_size[MAJOR_NR] = mtd_blksizes; blk_size[MAJOR_NR] = mtd_sizes; - - blk_init_queue(BLK_DEFAULT_QUEUE(MAJOR_NR), &mtdblock_request); + + BLK_INIT_QUEUE(BLK_DEFAULT_QUEUE(MAJOR_NR), &mtdblock_request, &mtdblock_lock); + kernel_thread (mtdblock_thread, NULL, CLONE_FS|CLONE_FILES|CLONE_SIGHAND); return 0; } diff -urN orig/drivers/mtd/mtdblock_ro.c linux/drivers/mtd/mtdblock_ro.c --- orig/drivers/mtd/mtdblock_ro.c Mon Aug 5 13:31:02 2002 +++ linux/drivers/mtd/mtdblock_ro.c Mon Aug 5 13:44:08 2002 @@ -1,5 +1,5 @@ /* - * $Id: mtdblock_ro.c,v 1.12 2001/11/20 11:42:33 dwmw2 Exp $ + * $Id: mtdblock_ro.c,v 1.13 2002/03/11 16:03:29 sioux Exp $ * * Read-only version of the mtdblock device, without the * read/erase/modify/writeback stuff @@ -45,8 +45,10 @@ #define BLK_DEC_USE_COUNT do {} while(0) #endif -static int mtd_sizes[MAX_MTD_DEVICES]; +/* this lock is used just in kernels >= 2.5.x */ +static spinlock_t mtdblock_ro_lock; +static int mtd_sizes[MAX_MTD_DEVICES]; static int mtdblock_open(struct inode *inode, struct file *file) { @@ -59,7 +61,7 @@ if (inode == 0) return -EINVAL; - dev = MINOR(inode->i_rdev); + dev = minor(inode->i_rdev); mtd = get_mtd_device(NULL, dev); if (!mtd) @@ -88,7 +90,7 @@ if (inode == NULL) release_return(-ENODEV); - dev = MINOR(inode->i_rdev); + dev = minor(inode->i_rdev); mtd = __get_mtd_device(NULL, dev); if (!mtd) { @@ -122,7 +124,7 @@ INIT_REQUEST; current_request = CURRENT; - if (MINOR(current_request->rq_dev) >= MAX_MTD_DEVICES) + if (minor(current_request->rq_dev) >= MAX_MTD_DEVICES) { printk("mtd: Unsupported device!\n"); end_request(0); @@ -131,9 +133,10 @@ // Grab our MTD structure - mtd = __get_mtd_device(NULL, MINOR(current_request->rq_dev)); + mtd = __get_mtd_device(NULL,minor(current_request->rq_dev)); if (!mtd) { - printk("MTD device %d doesn't appear to exist any more\n", CURRENT_DEV); + printk("MTD device %d doesn't appear to exist any more\n", + kdev_t_to_nr(CURRENT_DEV)); end_request(0); } @@ -141,7 +144,8 @@ (current_request->sector + current_request->nr_sectors) << 9 > mtd->size) { printk("mtd: Attempt to read past end of device!\n"); - printk("size: %x, sector: %lx, nr_sectors %lx\n", mtd->size, current_request->sector, current_request->nr_sectors); + printk("size: %x, sector: %lx, nr_sectors %lx\n", mtd->size, + current_request->sector, current_request->nr_sectors); end_request(0); continue; } @@ -152,11 +156,11 @@ /* Now drop the lock that the ll_rw_blk functions grabbed for us and process the request. This is necessary due to the extreme time we spend processing it. */ - spin_unlock_irq(&io_request_lock); + spin_unlock_irq(QUEUE_LOCK(QUEUE)); #endif // Handle the request - switch (current_request->cmd) + switch (rq_data_dir(current_request)) { size_t retlen; @@ -199,7 +203,7 @@ // Grab the lock and re-thread the item onto the linked list #if LINUX_VERSION_CODE >= KERNEL_VERSION(2,2,0) - spin_lock_irq(&io_request_lock); + spin_lock_irq(QUEUE_LOCK(QUEUE)); #endif end_request(res); } @@ -212,7 +216,7 @@ { struct mtd_info *mtd; - mtd = __get_mtd_device(NULL, MINOR(inode->i_rdev)); + mtd = __get_mtd_device(NULL, minor(inode->i_rdev)); if (!mtd) return -EINVAL; @@ -265,6 +269,9 @@ { int i; + /* this lock is used just in kernels >= 2.5.x */ + spin_lock_init(&mtdblock_ro_lock); + if (register_blkdev(MAJOR_NR,DEVICE_NAME,&mtd_fops)) { printk(KERN_NOTICE "Can't allocate major number %d for Memory Technology Devices.\n", MTD_BLOCK_MAJOR); @@ -280,7 +287,7 @@ blksize_size[MAJOR_NR] = NULL; blk_size[MAJOR_NR] = mtd_sizes; - blk_init_queue(BLK_DEFAULT_QUEUE(MAJOR_NR), &mtdblock_request); + BLK_INIT_QUEUE(BLK_DEFAULT_QUEUE(MAJOR_NR), &mtdblock_request, &mtdblock_ro_lock); return 0; } @@ -298,3 +305,4 @@ MODULE_LICENSE("GPL"); MODULE_AUTHOR("Erwin Authried et al."); MODULE_DESCRIPTION("Simple read-only block device emulation access to MTD devices"); + diff -urN orig/drivers/mtd/mtdchar-compat.c linux/drivers/mtd/mtdchar-compat.c --- orig/drivers/mtd/mtdchar-compat.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/mtd/mtdchar-compat.c Wed Mar 13 17:01:09 2002 @@ -0,0 +1,587 @@ +/* + * $Id: mtdchar-compat.c,v 1.2 2001/10/02 15:05:11 dwmw2 Exp $ + * + * Character-device access to raw MTD devices. + * + */ + + +#include +#include +#include +#include +#include +#include + +#ifdef CONFIG_DEVFS_FS +#include +static void mtd_notify_add(struct mtd_info* mtd); +static void mtd_notify_remove(struct mtd_info* mtd); + +static struct mtd_notifier notifier = { + add: mtd_notify_add, + remove: mtd_notify_remove, +}; + +static devfs_handle_t devfs_dir_handle; +static devfs_handle_t devfs_rw_handle[MAX_MTD_DEVICES]; +static devfs_handle_t devfs_ro_handle[MAX_MTD_DEVICES]; +#endif + +#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,2,0) +static loff_t mtd_lseek (struct file *file, loff_t offset, int orig) +#else +static int mtd_lseek (struct inode *inode, struct file *file, off_t offset, int orig) +#endif +{ + struct mtd_info *mtd=(struct mtd_info *)file->private_data; + + switch (orig) { + case 0: + /* SEEK_SET */ + file->f_pos = offset; + break; + case 1: + /* SEEK_CUR */ + file->f_pos += offset; + break; + case 2: + /* SEEK_END */ + file->f_pos =mtd->size + offset; + break; + default: + return -EINVAL; + } + + if (file->f_pos < 0) + file->f_pos = 0; + else if (file->f_pos >= mtd->size) + file->f_pos = mtd->size - 1; + + return file->f_pos; +} + + + +static int mtd_open(struct inode *inode, struct file *file) +{ + int minor = MINOR(inode->i_rdev); + int devnum = minor >> 1; + struct mtd_info *mtd; + + DEBUG(MTD_DEBUG_LEVEL0, "MTD_open\n"); + + if (devnum >= MAX_MTD_DEVICES) + return -ENODEV; + + /* You can't open the RO devices RW */ + if ((file->f_mode & 2) && (minor & 1)) + return -EACCES; + + mtd = get_mtd_device(NULL, devnum); + if (!mtd) + return -ENODEV; + if (MTD_ABSENT == mtd->type) { + put_mtd_device(mtd); + return -ENODEV; + } + + MOD_INC_USE_COUNT; + + file->private_data = mtd; + + /* You can't open it RW if it's not a writeable device */ + if ((file->f_mode & 2) && !(mtd->flags & MTD_WRITEABLE)) { + put_mtd_device(mtd); + MOD_DEC_USE_COUNT; + return -EACCES; + } + + return 0; +} /* mtd_open */ + +/*====================================================================*/ + +static release_t mtd_close(struct inode *inode, + struct file *file) +{ + struct mtd_info *mtd; + + DEBUG(MTD_DEBUG_LEVEL0, "MTD_close\n"); + + mtd = (struct mtd_info *)file->private_data; + + if (mtd->sync) + mtd->sync(mtd); + + put_mtd_device(mtd); + + MOD_DEC_USE_COUNT; + release_return(0); +} /* mtd_close */ + +#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,2,0) +#define FILE_POS *ppos +#else +#define FILE_POS file->f_pos +#endif + +/* FIXME: This _really_ needs to die. In 2.5, we should lock the + userspace buffer down and use it directly with readv/writev. +*/ +#define MAX_KMALLOC_SIZE 0x20000 + +#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,2,0) +static ssize_t mtd_read(struct file *file, char *buf, size_t count,loff_t *ppos) +#else +static int mtd_read(struct inode *inode,struct file *file, char *buf, int count) +#endif +{ + struct mtd_info *mtd = (struct mtd_info *)file->private_data; + size_t retlen=0; + size_t total_retlen=0; + int ret=0; +#ifndef NO_MM + int len; + char *kbuf; +#endif + DEBUG(MTD_DEBUG_LEVEL0,"MTD_read\n"); + + if (FILE_POS + count > mtd->size) + count = mtd->size - FILE_POS; + + if (!count) + return 0; + + /* FIXME: Use kiovec in 2.3 or 2.2+rawio, or at + * least split the IO into smaller chunks. + */ +#ifdef NO_MM + ret = MTD_READ(mtd, FILE_POS, count, &retlen, buf); + if (!ret) { + FILE_POS += retlen; + ret = retlen; + } + total_retlen = ret; +#else + while (count) { + if (count > MAX_KMALLOC_SIZE) + len = MAX_KMALLOC_SIZE; + else + len = count; + + kbuf=kmalloc(len,GFP_KERNEL); + if (!kbuf) + return -ENOMEM; + + ret = MTD_READ(mtd, FILE_POS, len, &retlen, kbuf); + if (!ret) { + FILE_POS += retlen; + if (copy_to_user(buf, kbuf, retlen)) { + kfree(kbuf); + return -EFAULT; + } + else + total_retlen += retlen; + + count -= retlen; + buf += retlen; + } + else { + kfree(kbuf); + return ret; + } + + kfree(kbuf); + } + +#endif + return total_retlen; +} /* mtd_read */ + +#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,2,0) +static ssize_t mtd_write(struct file *file, const char *buf, size_t count,loff_t *ppos) +#else +static read_write_t mtd_write(struct inode *inode,struct file *file, const char *buf, count_t count) +#endif +{ + struct mtd_info *mtd = (struct mtd_info *)file->private_data; + size_t retlen; + size_t total_retlen=0; + int ret=0; +#ifndef NO_MM + int len; + char *kbuf; +#endif + + DEBUG(MTD_DEBUG_LEVEL0,"MTD_write\n"); + + if (FILE_POS == mtd->size) + return -ENOSPC; + + if (FILE_POS + count > mtd->size) + count = mtd->size - FILE_POS; + + if (!count) + return 0; + +#ifdef NO_MM + ret = MTD_WRITE(mtd, FILE_POS, count, &retlen, buf); + if (!ret) { + FILE_POS += retlen; + ret = retlen; + } + total_retlen = ret; +#else + while (count) { + if (count > MAX_KMALLOC_SIZE) + len = MAX_KMALLOC_SIZE; + else + len = count; + + kbuf=kmalloc(len,GFP_KERNEL); + if (!kbuf) { + printk("kmalloc is null\n"); + return -ENOMEM; + } + + if (copy_from_user(kbuf, buf, len)) { + kfree(kbuf); + return -EFAULT; + } + + ret = (*(mtd->write))(mtd, FILE_POS, len, &retlen, kbuf); + if (!ret) { + FILE_POS += retlen; + total_retlen += retlen; + count -= retlen; + buf += retlen; + } + else { + kfree(kbuf); + return ret; + } + + kfree(kbuf); + } +#endif + return total_retlen; +} /* mtd_write */ + +/*====================================================================== + + IOCTL calls for getting device parameters. + +======================================================================*/ +static void mtd_erase_callback (struct erase_info *instr) +{ + wake_up((wait_queue_head_t *)instr->priv); +} + +static int mtd_ioctl(struct inode *inode, struct file *file, + u_int cmd, u_long arg) +{ + struct mtd_info *mtd = (struct mtd_info *)file->private_data; + int ret = 0; + u_long size; + + DEBUG(MTD_DEBUG_LEVEL0, "MTD_ioctl\n"); + + size = (cmd & IOCSIZE_MASK) >> IOCSIZE_SHIFT; + if (cmd & IOC_IN) { + ret = verify_area(VERIFY_READ, (char *)arg, size); + if (ret) return ret; + } + if (cmd & IOC_OUT) { + ret = verify_area(VERIFY_WRITE, (char *)arg, size); + if (ret) return ret; + } + + switch (cmd) { + case MEMGETREGIONCOUNT: + if (copy_to_user((int *) arg, &(mtd->numeraseregions), sizeof(int))) + return -EFAULT; + break; + + case MEMGETREGIONINFO: + { + struct region_info_user ur; + + if (copy_from_user( &ur, + (struct region_info_user *)arg, + sizeof(struct region_info_user))) { + return -EFAULT; + } + + if (ur.regionindex >= mtd->numeraseregions) + return -EINVAL; + if (copy_to_user((struct mtd_erase_region_info *) arg, + &(mtd->eraseregions[ur.regionindex]), + sizeof(struct mtd_erase_region_info))) + return -EFAULT; + break; + } + + case MEMGETINFO: + if (copy_to_user((struct mtd_info *)arg, mtd, + sizeof(struct mtd_info_user))) + return -EFAULT; + break; + + case MEMERASE: + { + struct erase_info *erase=kmalloc(sizeof(struct erase_info),GFP_KERNEL); + if (!erase) + ret = -ENOMEM; + else { + wait_queue_head_t waitq; + DECLARE_WAITQUEUE(wait, current); + + init_waitqueue_head(&waitq); + + memset (erase,0,sizeof(struct erase_info)); + if (copy_from_user(&erase->addr, (u_long *)arg, + 2 * sizeof(u_long))) { + kfree(erase); + return -EFAULT; + } + erase->mtd = mtd; + erase->callback = mtd_erase_callback; + erase->priv = (unsigned long)&waitq; + + /* + FIXME: Allow INTERRUPTIBLE. Which means + not having the wait_queue head on the stack. + + If the wq_head is on the stack, and we + leave because we got interrupted, then the + wq_head is no longer there when the + callback routine tries to wake us up. + */ + ret = mtd->erase(mtd, erase); + if (!ret) { + set_current_state(TASK_UNINTERRUPTIBLE); + add_wait_queue(&waitq, &wait); + if (erase->state != MTD_ERASE_DONE && + erase->state != MTD_ERASE_FAILED) + schedule(); + remove_wait_queue(&waitq, &wait); + set_current_state(TASK_RUNNING); + + ret = (erase->state == MTD_ERASE_FAILED)?-EIO:0; + } + kfree(erase); + } + break; + } + + case MEMWRITEOOB: + { + struct mtd_oob_buf buf; + void *databuf; + ssize_t retlen; + + if (copy_from_user(&buf, (struct mtd_oob_buf *)arg, sizeof(struct mtd_oob_buf))) + return -EFAULT; + + if (buf.length > 0x4096) + return -EINVAL; + + if (!mtd->write_oob) + ret = -EOPNOTSUPP; + else + ret = verify_area(VERIFY_READ, (char *)buf.ptr, buf.length); + + if (ret) + return ret; + + databuf = kmalloc(buf.length, GFP_KERNEL); + if (!databuf) + return -ENOMEM; + + if (copy_from_user(databuf, buf.ptr, buf.length)) { + kfree(databuf); + return -EFAULT; + } + + ret = (mtd->write_oob)(mtd, buf.start, buf.length, &retlen, databuf); + + if (copy_to_user((void *)arg + sizeof(u_int32_t), &retlen, sizeof(u_int32_t))) + ret = -EFAULT; + + kfree(databuf); + break; + + } + + case MEMREADOOB: + { + struct mtd_oob_buf buf; + void *databuf; + ssize_t retlen; + + if (copy_from_user(&buf, (struct mtd_oob_buf *)arg, sizeof(struct mtd_oob_buf))) + return -EFAULT; + + if (buf.length > 0x4096) + return -EINVAL; + + if (!mtd->read_oob) + ret = -EOPNOTSUPP; + else + ret = verify_area(VERIFY_WRITE, (char *)buf.ptr, buf.length); + + if (ret) + return ret; + + databuf = kmalloc(buf.length, GFP_KERNEL); + if (!databuf) + return -ENOMEM; + + ret = (mtd->read_oob)(mtd, buf.start, buf.length, &retlen, databuf); + + if (copy_to_user((void *)arg + sizeof(u_int32_t), &retlen, sizeof(u_int32_t))) + ret = -EFAULT; + else if (retlen && copy_to_user(buf.ptr, databuf, retlen)) + ret = -EFAULT; + + kfree(databuf); + break; + } + + case MEMLOCK: + { + unsigned long adrs[2]; + + if (copy_from_user(adrs ,(void *)arg, 2* sizeof(unsigned long))) + return -EFAULT; + + if (!mtd->lock) + ret = -EOPNOTSUPP; + else + ret = mtd->lock(mtd, adrs[0], adrs[1]); + break; + } + + case MEMUNLOCK: + { + unsigned long adrs[2]; + + if (copy_from_user(adrs, (void *)arg, 2* sizeof(unsigned long))) + return -EFAULT; + + if (!mtd->unlock) + ret = -EOPNOTSUPP; + else + ret = mtd->unlock(mtd, adrs[0], adrs[1]); + break; + } + + + default: + DEBUG(MTD_DEBUG_LEVEL0, "Invalid ioctl %x (MEMGETINFO = %x)\n", cmd, MEMGETINFO); + ret = -ENOTTY; + } + + return ret; +} /* memory_ioctl */ + +static struct file_operations mtd_fops = { +#if LINUX_VERSION_CODE >= 0x20300 /* Someone knows when these made their debut? */ + owner: THIS_MODULE, +#endif +#if LINUX_VERSION_CODE >=0x20200 + llseek: mtd_lseek, /* lseek */ +#else + lseek: mtd_lseek, /* lseek */ +#endif + read: mtd_read, /* read */ + write: mtd_write, /* write */ + ioctl: mtd_ioctl, /* ioctl */ + open: mtd_open, /* open */ + release: mtd_close, /* release */ +}; + + +#ifdef CONFIG_DEVFS_FS +/* Notification that a new device has been added. Create the devfs entry for + * it. */ + +static void mtd_notify_add(struct mtd_info* mtd) +{ + char name[8]; + + if (!mtd) + return; + + sprintf(name, "%d", mtd->index); + devfs_rw_handle[mtd->index] = devfs_register(devfs_dir_handle, name, + DEVFS_FL_DEFAULT, MTD_CHAR_MAJOR, mtd->index*2, + S_IFCHR | S_IRUGO | S_IWUGO, + &mtd_fops, NULL); + + sprintf(name, "%dro", mtd->index); + devfs_ro_handle[mtd->index] = devfs_register(devfs_dir_handle, name, + DEVFS_FL_DEFAULT, MTD_CHAR_MAJOR, mtd->index*2+1, + S_IFCHR | S_IRUGO | S_IWUGO, + &mtd_fops, NULL); +} + +static void mtd_notify_remove(struct mtd_info* mtd) +{ + if (!mtd) + return; + + devfs_unregister(devfs_rw_handle[mtd->index]); + devfs_unregister(devfs_ro_handle[mtd->index]); +} +#endif + +#if LINUX_VERSION_CODE < 0x20212 && defined(MODULE) +#define init_mtdchar init_module +#define cleanup_mtdchar cleanup_module +#endif + +mod_init_t init_mtdchar(void) +{ +#ifdef CONFIG_DEVFS_FS + if (devfs_register_chrdev(MTD_CHAR_MAJOR, "mtd", &mtd_fops)) + { + printk(KERN_NOTICE "Can't allocate major number %d for Memory Technology Devices.\n", + MTD_CHAR_MAJOR); + return -EAGAIN; + } + + devfs_dir_handle = devfs_mk_dir(NULL, "mtd", NULL); + + register_mtd_user(¬ifier); +#else + if (register_chrdev(MTD_CHAR_MAJOR, "mtd", &mtd_fops)) + { + printk(KERN_NOTICE "Can't allocate major number %d for Memory Technology Devices.\n", + MTD_CHAR_MAJOR); + return -EAGAIN; + } +#endif + + return 0; +} + +mod_exit_t cleanup_mtdchar(void) +{ +#ifdef CONFIG_DEVFS_FS + unregister_mtd_user(¬ifier); + devfs_unregister(devfs_dir_handle); + devfs_unregister_chrdev(MTD_CHAR_MAJOR, "mtd"); +#else + unregister_chrdev(MTD_CHAR_MAJOR, "mtd"); +#endif +} + +module_init(init_mtdchar); +module_exit(cleanup_mtdchar); + + +MODULE_LICENSE("GPL"); +MODULE_AUTHOR("David Woodhouse "); +MODULE_DESCRIPTION("Direct character-device access to MTD devices"); diff -urN orig/drivers/mtd/mtdconcat.c linux/drivers/mtd/mtdconcat.c --- orig/drivers/mtd/mtdconcat.c Mon Aug 5 13:31:02 2002 +++ linux/drivers/mtd/mtdconcat.c Fri May 31 11:48:54 2002 @@ -5,7 +5,7 @@ * * This code is GPL * - * $Id: mtdconcat.c,v 1.2 2002/03/22 08:45:22 dwmw2 Exp $ + * $Id: mtdconcat.c,v 1.3 2002/05/21 21:04:25 dwmw2 Exp $ */ #include @@ -303,11 +303,14 @@ */ erase->addr = 0; } + kfree(erase); + if (err) + return err; + instr->state = MTD_ERASE_DONE; if (instr->callback) instr->callback(instr); - kfree(erase); - return err; + return 0; } static int concat_lock (struct mtd_info *mtd, loff_t ofs, size_t len) diff -urN orig/drivers/mtd/mtdpart.c linux/drivers/mtd/mtdpart.c --- orig/drivers/mtd/mtdpart.c Mon Aug 5 13:31:03 2002 +++ linux/drivers/mtd/mtdpart.c Fri Feb 21 15:17:33 2003 @@ -6,7 +6,6 @@ * This code is GPL * * $Id: mtdpart.c,v 1.27 2002/03/08 16:34:35 rkaiser Exp $ - * - with protection register access removed until that code is merged in 2.4. * * 02-21-2002 Thomas Gleixner * added support for read_oob, write_oob @@ -223,10 +222,14 @@ slave->mtd.type = master->type; slave->mtd.flags = master->flags & ~parts[i].mask_flags; slave->mtd.size = parts[i].size; + slave->mtd.priv = master->priv; slave->mtd.oobblock = master->oobblock; slave->mtd.oobsize = master->oobsize; slave->mtd.ecctype = master->ecctype; slave->mtd.eccsize = master->eccsize; + slave->mtd.read_user_prot_reg = master->read_user_prot_reg; + slave->mtd.read_fact_prot_reg = master->read_fact_prot_reg; + slave->mtd.write_user_prot_reg = master->write_user_prot_reg; slave->mtd.name = parts[i].name; slave->mtd.bank_size = master->bank_size; diff -urN orig/drivers/mtd/nand/Config.in linux/drivers/mtd/nand/Config.in --- orig/drivers/mtd/nand/Config.in Sun Oct 14 20:53:09 2001 +++ linux/drivers/mtd/nand/Config.in Fri May 31 11:48:57 2002 @@ -1,6 +1,6 @@ # drivers/mtd/nand/Config.in -# $Id: Config.in,v 1.4 2001/09/19 09:35:23 dwmw2 Exp $ +# $Id: Config.in,v 1.8 2002/04/30 09:34:16 mag Exp $ mainmenu_option next_comment @@ -10,9 +10,21 @@ if [ "$CONFIG_MTD_NAND" = "y" -o "$CONFIG_MTD_NAND" = "m" ]; then bool ' Enable ECC correction algorithm' CONFIG_MTD_NAND_ECC bool ' Verify NAND page writes' CONFIG_MTD_NAND_VERIFY_WRITE + if [ "$CONFIG_JFFS2_FS" = "y" -o "$CONFIG_JFFS2_FS" = "m" ]; then + bool ' Use JFFS2 layout for OOB area' CONFIG_MTD_NAND_ECC_JFFS2 + fi fi + if [ "$CONFIG_ARM" = "y" -a "$CONFIG_ARCH_P720T" = "y" ]; then dep_tristate ' NAND Flash device on SPIA board' CONFIG_MTD_NAND_SPIA $CONFIG_MTD_NAND +fi + +if [ "$CONFIG_ARCH_AUTCPU12" = "y" ]; then + dep_tristate ' SmartMedia Card on AUTCPU12 board' CONFIG_MTD_NAND_AUTCPU12 $CONFIG_MTD_NAND +fi + +if [ "$CONFIG_ARCH_EDB7312" = "y" ]; then + dep_tristate ' NAND Flash device on EDP7312 board' CONFIG_MTD_NAND_EDB7312 $CONFIG_MTD_NAND fi endmenu diff -urN orig/drivers/mtd/nand/Makefile linux/drivers/mtd/nand/Makefile --- orig/drivers/mtd/nand/Makefile Sun Oct 14 20:53:09 2001 +++ linux/drivers/mtd/nand/Makefile Fri May 31 11:48:57 2002 @@ -1,7 +1,7 @@ # # linux/drivers/nand/Makefile # -# $Id: Makefile,v 1.5 2001/09/19 22:39:59 dwmw2 Exp $ +# $Id: Makefile,v 1.7 2002/04/30 09:34:16 mag Exp $ O_TARGET := nandlink.o @@ -12,5 +12,7 @@ obj-$(CONFIG_MTD_NAND) += $(nandobjs-y) obj-$(CONFIG_MTD_NAND_SPIA) += spia.o +obj-$(CONFIG_MTD_NAND_AUTCPU12) += autcpu12.o +obj-$(CONFIG_MTD_NAND_EDB7312) += edb7312.o include $(TOPDIR)/Rules.make diff -urN orig/drivers/mtd/nand/autcpu12.c linux/drivers/mtd/nand/autcpu12.c --- orig/drivers/mtd/nand/autcpu12.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/mtd/nand/autcpu12.c Wed Mar 13 17:00:52 2002 @@ -0,0 +1,251 @@ +/* + * drivers/mtd/autcpu12.c + * + * Copyright (c) 2001 Thomas Gleixner (gleixner@autronix.de) + * + * Derived from drivers/mtd/spia.c + * Copyright (C) 2000 Steven J. Hill (sjhill@cotw.com) + * + * $Id: autcpu12.c,v 1.3 2002/02/26 17:58:24 gleixner Exp $ + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + * + * Overview: + * This is a device driver for the NAND flash device found on the + * autronix autcpu12 board, which is a SmartMediaCard. It supports + * 16MB, 32MB and 64MB cards. + * + * 02-12-2002 TG Cleanup of module params + * + * 02-20-2002 TG adjusted for different rd/wr adress support + * added support for read device ready/busy line + * added page_cache + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +/* + * MTD structure for AUTCPU12 board + */ +static struct mtd_info *autcpu12_mtd = NULL; + +/* + * Module stuff + */ +#if LINUX_VERSION_CODE < 0x20212 && defined(MODULE) +#define autcpu12_init init_module +#define autcpu12_cleanup cleanup_module +#endif + +static int autcpu12_io_base = CS89712_VIRT_BASE; +static int autcpu12_fio_pbase = AUTCPU12_PHYS_SMC; +static int autcpu12_fio_ctrl = AUTCPU12_SMC_SELECT_OFFSET; +static int autcpu12_pedr = AUTCPU12_SMC_PORT_OFFSET; +static int autcpu12_fio_base; + +#ifdef MODULE +MODULE_PARM(autcpu12_fio_pbase, "i"); +MODULE_PARM(autcpu12_fio_ctrl, "i"); +MODULE_PARM(autcpu12_pedr, "i"); + +__setup("autcpu12_fio_pbase=",autcpu12_fio_pbase); +__setup("autcpu12_fio_ctrl=",autcpu12_fio_ctrl); +__setup("autcpu12_pedr=",autcpu12_pedr); +#endif + +/* + * Define partitions for flash devices + */ + +static struct mtd_partition partition_info16k[] = { + { name: "AUTCPU12 flash partition 1", + offset: 0, + size: 8 * SZ_1M }, + { name: "AUTCPU12 flash partition 2", + offset: 8 * SZ_1M, + size: 8 * SZ_1M }, +}; + +static struct mtd_partition partition_info32k[] = { + { name: "AUTCPU12 flash partition 1", + offset: 0, + size: 16 * SZ_1M }, + { name: "AUTCPU12 flash partition 2", + offset: 16 * SZ_1M, + size: 24 * SZ_1M }, +}; + +static struct mtd_partition partition_info64k[] = { + { name: "AUTCPU12 flash partition 1", + offset: 0, + size: 16 * SZ_1M }, + { name: "AUTCPU12 flash partition 2", + offset: 16 * SZ_1M, + size: 48 * SZ_1M}, +}; + +#define NUM_PARTITIONS16K 2 +#define NUM_PARTITIONS32K 3 +#define NUM_PARTITIONS64K 4 + +/* + * hardware specific access to control-lines +*/ +void autcpu12_hwcontrol(int cmd) +{ + + switch(cmd){ + + case NAND_CTL_SETCLE: (*(volatile unsigned char *) (autcpu12_io_base + autcpu12_pedr)) |= AUTCPU12_SMC_CLE; break; + case NAND_CTL_CLRCLE: (*(volatile unsigned char *) (autcpu12_io_base + autcpu12_pedr)) &= ~AUTCPU12_SMC_CLE; break; + + case NAND_CTL_SETALE: (*(volatile unsigned char *) (autcpu12_io_base + autcpu12_pedr)) |= AUTCPU12_SMC_ALE; break; + case NAND_CTL_CLRALE: (*(volatile unsigned char *) (autcpu12_io_base + autcpu12_pedr)) &= ~AUTCPU12_SMC_ALE; break; + + case NAND_CTL_SETNCE: (*(volatile unsigned char *) (autcpu12_fio_base + autcpu12_fio_ctrl)) = 0x01; break; + case NAND_CTL_CLRNCE: (*(volatile unsigned char *) (autcpu12_fio_base + autcpu12_fio_ctrl)) = 0x00; break; + } +} + +/* +* read device ready pin +*/ +int autcpu12_device_ready(void) +{ + + return ( (*(volatile unsigned char *) (autcpu12_io_base + autcpu12_pedr)) & AUTCPU12_SMC_RDY) ? 1 : 0; + +} +/* + * Main initialization routine + */ +int __init autcpu12_init (void) +{ + struct nand_chip *this; + int err = 0; + + /* Allocate memory for MTD device structure and private data */ + autcpu12_mtd = kmalloc (sizeof(struct mtd_info) + sizeof (struct nand_chip), + GFP_KERNEL); + if (!autcpu12_mtd) { + printk ("Unable to allocate AUTCPU12 NAND MTD device structure.\n"); + err = -ENOMEM; + goto out; + } + + /* map physical adress */ + autcpu12_fio_base=(unsigned long)ioremap(autcpu12_fio_pbase,SZ_1K); + if(!autcpu12_fio_base){ + printk("Ioremap autcpu12 SmartMedia Card failed\n"); + err = -EIO; + goto out_mtd; + } + + /* Get pointer to private data */ + this = (struct nand_chip *) (&autcpu12_mtd[1]); + + /* Initialize structures */ + memset((char *) autcpu12_mtd, 0, sizeof(struct mtd_info)); + memset((char *) this, 0, sizeof(struct nand_chip)); + + /* Link the private data with the MTD structure */ + autcpu12_mtd->priv = this; + + /* Set address of NAND IO lines */ + this->IO_ADDR_R = autcpu12_fio_base; + this->IO_ADDR_W = autcpu12_fio_base; + this->hwcontrol = autcpu12_hwcontrol; + this->dev_ready = autcpu12_device_ready; + /* 20 us command delay time */ + this->chip_delay = 20; + + /* Scan to find existance of the device */ + if (nand_scan (autcpu12_mtd)) { + err = -ENXIO; + goto out_ior; + } + + /* Allocate memory for internal data buffer */ + this->data_buf = kmalloc (sizeof(u_char) * (autcpu12_mtd->oobblock + autcpu12_mtd->oobsize), GFP_KERNEL); + if (!this->data_buf) { + printk ("Unable to allocate NAND data buffer for AUTCPU12.\n"); + err = -ENOMEM; + goto out_ior; + } + + /* Allocate memory for internal data buffer */ + this->data_cache = kmalloc (sizeof(u_char) * (autcpu12_mtd->oobblock + autcpu12_mtd->oobsize), GFP_KERNEL); + if (!this->data_cache) { + printk ("Unable to allocate NAND data cache for AUTCPU12.\n"); + err = -ENOMEM; + goto out_buf; + } + this->cache_page = -1; + + /* Register the partitions */ + switch(autcpu12_mtd->size){ + case SZ_16M: add_mtd_partitions(autcpu12_mtd, partition_info16k, NUM_PARTITIONS16K); break; + case SZ_32M: add_mtd_partitions(autcpu12_mtd, partition_info32k, NUM_PARTITIONS32K); break; + case SZ_64M: add_mtd_partitions(autcpu12_mtd, partition_info64k, NUM_PARTITIONS64K); break; + default: { + printk ("Unsupported SmartMedia device\n"); + err = -ENXIO; + goto out_cac; + } + } + goto out; + +out_cac: + kfree (this->data_cache); +out_buf: + kfree (this->data_buf); +out_ior: + iounmap((void *)autcpu12_fio_base); +out_mtd: + kfree (autcpu12_mtd); +out: + return err; +} + +module_init(autcpu12_init); + +/* + * Clean up routine + */ +#ifdef MODULE +static void __exit autcpu12_cleanup (void) +{ + struct nand_chip *this = (struct nand_chip *) &autcpu12_mtd[1]; + + /* Unregister partitions */ + del_mtd_partitions(autcpu12_mtd); + + /* Unregister the device */ + del_mtd_device (autcpu12_mtd); + + /* Free internal data buffers */ + kfree (this->data_buf); + kfree (this->data_cache); + + /* unmap physical adress */ + iounmap((void *)autcpu12_fio_base); + + /* Free the MTD device structure */ + kfree (autcpu12_mtd); +} +module_exit(autcpu12_cleanup); +#endif + +MODULE_LICENSE("GPL"); +MODULE_AUTHOR("Thomas Gleixner "); +MODULE_DESCRIPTION("Glue layer for SmartMediaCard on autronix autcpu12"); diff -urN orig/drivers/mtd/nand/nand.c linux/drivers/mtd/nand/nand.c --- orig/drivers/mtd/nand/nand.c Sun Oct 14 20:53:09 2001 +++ linux/drivers/mtd/nand/nand.c Fri May 31 11:48:57 2002 @@ -3,7 +3,51 @@ * * Copyright (C) 2000 Steven J. Hill (sjhill@cotw.com) * - * $Id: nand.c,v 1.12 2001/10/02 15:05:14 dwmw2 Exp $ + * 10-29-2001 Thomas Gleixner (gleixner@autronix.de) + * - Changed nand_chip structure for controlline function to + * support different hardware structures (Access to + * controllines ALE,CLE,NCE via hardware specific function. + * - exit out of "failed erase block" changed, to avoid + * driver hangup + * - init_waitqueue_head added in function nand_scan !! + * + * 01-30-2002 Thomas Gleixner (gleixner@autronix.de) + * change in nand_writev to block invalid vecs entries + * + * 02-11-2002 Thomas Gleixner (gleixner@autronix.de) + * - major rewrite to avoid duplicated code + * common nand_write_page function + * common get_chip function + * - added oob_config structure for out of band layouts + * - write_oob changed for partial programming + * - read cache for faster access for subsequent reads + * from the same page. + * - support for different read/write address + * - support for device ready/busy line + * - read oob for more than one page enabled + * + * 02-27-2002 Thomas Gleixner (gleixner@autronix.de) + * - command-delay can be programmed + * - fixed exit from erase with callback-function enabled + * + * 03-21-2002 Thomas Gleixner (gleixner@autronix.de) + * - DEBUG improvements provided by Elizabeth Clarke + * (eclarke@aminocom.com) + * - added zero check for this->chip_delay + * + * 04-03-2002 Thomas Gleixner (gleixner@autronix.de) + * - added added hw-driver supplied command and wait functions + * - changed blocking for erase (erase suspend enabled) + * - check pointers before accessing flash provided by + * John Hall (john.hall@optionexist.co.uk) + * + * 04-09-2002 Thomas Gleixner (gleixner@autronix.de) + * - nand_wait repaired + * + * 04-28-2002 Thomas Gleixner (gleixner@autronix.de) + * - OOB config defines moved to nand.h + * + * $Id: nand.c,v 1.24 2002/04/28 13:39:48 gleixner Exp $ * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as @@ -31,79 +75,83 @@ /* * Macros for low-level register control */ -#define NAND_CTRL (*(volatile unsigned char *) \ - ((struct nand_chip *) mtd->priv)->CTRL_ADDR) -#define nand_select() NAND_CTRL &= ~this->NCE; \ - nand_command(mtd, NAND_CMD_RESET, -1, -1); \ - udelay (10); -#define nand_deselect() NAND_CTRL |= ~this->NCE; +#define nand_select() this->hwcontrol(NAND_CTL_SETNCE); \ + this->cmdfunc(mtd, NAND_CMD_RESET, -1, -1); + +#define nand_deselect() this->hwcontrol(NAND_CTL_CLRNCE); /* + * Definition of the out of band configuration structure + */ +struct nand_oob_config { + int ecc_pos[6]; /* position of ECC bytes inside oob */ + int badblock_pos; /* position of bad block flag inside oob -1 = inactive */ + int eccvalid_pos; /* position of ECC valid flag inside oob -1 = inactive */ +}; + + +static struct nand_oob_config oob_config; +/* * NAND low-level MTD interface functions */ -static int nand_read (struct mtd_info *mtd, loff_t from, size_t len, - size_t *retlen, u_char *buf); +static int nand_read (struct mtd_info *mtd, loff_t from, size_t len, size_t * retlen, u_char * buf); static int nand_read_ecc (struct mtd_info *mtd, loff_t from, size_t len, - size_t *retlen, u_char *buf, u_char *ecc_code); -static int nand_read_oob (struct mtd_info *mtd, loff_t from, size_t len, - size_t *retlen, u_char *buf); -static int nand_write (struct mtd_info *mtd, loff_t to, size_t len, - size_t *retlen, const u_char *buf); + size_t * retlen, u_char * buf, u_char * ecc_code); +static int nand_read_oob (struct mtd_info *mtd, loff_t from, size_t len, size_t * retlen, u_char * buf); +static int nand_write (struct mtd_info *mtd, loff_t to, size_t len, size_t * retlen, const u_char * buf); static int nand_write_ecc (struct mtd_info *mtd, loff_t to, size_t len, - size_t *retlen, const u_char *buf, - u_char *ecc_code); -static int nand_write_oob (struct mtd_info *mtd, loff_t to, size_t len, - size_t *retlen, const u_char *buf); + size_t * retlen, const u_char * buf, u_char * ecc_code); +static int nand_write_oob (struct mtd_info *mtd, loff_t to, size_t len, size_t * retlen, const u_char * buf); static int nand_writev (struct mtd_info *mtd, const struct iovec *vecs, - unsigned long count, loff_t to, size_t *retlen); + unsigned long count, loff_t to, size_t * retlen); static int nand_erase (struct mtd_info *mtd, struct erase_info *instr); static void nand_sync (struct mtd_info *mtd); +static int nand_write_page (struct mtd_info *mtd, struct nand_chip *this, + int page, int col, int last, u_char * ecc_code); /* * Send command to NAND device */ -static void nand_command (struct mtd_info *mtd, unsigned command, - int column, int page_addr) +static void nand_command (struct mtd_info *mtd, unsigned command, int column, int page_addr) { register struct nand_chip *this = mtd->priv; - register unsigned long NAND_IO_ADDR = this->IO_ADDR; + register unsigned long NAND_IO_ADDR = this->IO_ADDR_W; /* Begin command latch cycle */ - NAND_CTRL |= this->CLE; + this->hwcontrol (NAND_CTL_SETCLE); /* * Write out the command to the device. */ - if (command != NAND_CMD_SEQIN) + if (command != NAND_CMD_SEQIN) writeb (command, NAND_IO_ADDR); else { if (mtd->oobblock == 256 && column >= 256) { column -= 256; - writeb(NAND_CMD_RESET, NAND_IO_ADDR); - writeb(NAND_CMD_READOOB, NAND_IO_ADDR); - writeb(NAND_CMD_SEQIN, NAND_IO_ADDR); - } - else if (mtd->oobblock == 512 && column >= 256) { + writeb (NAND_CMD_RESET, NAND_IO_ADDR); + writeb (NAND_CMD_READOOB, NAND_IO_ADDR); + writeb (NAND_CMD_SEQIN, NAND_IO_ADDR); + } else if (mtd->oobblock == 512 && column >= 256) { if (column < 512) { column -= 256; - writeb(NAND_CMD_READ1, NAND_IO_ADDR); - writeb(NAND_CMD_SEQIN, NAND_IO_ADDR); - } - else { + writeb (NAND_CMD_READ1, NAND_IO_ADDR); + writeb (NAND_CMD_SEQIN, NAND_IO_ADDR); + } else { column -= 512; - writeb(NAND_CMD_READOOB, NAND_IO_ADDR); - writeb(NAND_CMD_SEQIN, NAND_IO_ADDR); + writeb (NAND_CMD_READOOB, NAND_IO_ADDR); + writeb (NAND_CMD_SEQIN, NAND_IO_ADDR); } - } - else { - writeb(NAND_CMD_READ0, NAND_IO_ADDR); - writeb(NAND_CMD_SEQIN, NAND_IO_ADDR); + } else { + writeb (NAND_CMD_READ0, NAND_IO_ADDR); + writeb (NAND_CMD_SEQIN, NAND_IO_ADDR); } } /* Set ALE and clear CLE to start address cycle */ - NAND_CTRL &= ~this->CLE; - NAND_CTRL |= this->ALE; + this->hwcontrol (NAND_CTL_CLRCLE); + + if (column != -1 || page_addr != -1) + this->hwcontrol (NAND_CTL_SETALE); /* Serially input address */ if (column != -1) @@ -113,27 +161,251 @@ writeb ((unsigned char) ((page_addr >> 8) & 0xff), NAND_IO_ADDR); /* One more address cycle for higher density devices */ if (mtd->size & 0x0c000000) { - writeb ((unsigned char) ((page_addr >> 16) & 0x0f), - NAND_IO_ADDR); + writeb ((unsigned char) ((page_addr >> 16) & 0x0f), NAND_IO_ADDR); } } - /* Latch in address */ - NAND_CTRL &= ~this->ALE; + if (column != -1 || page_addr != -1) + this->hwcontrol (NAND_CTL_CLRALE); + + /* + * If we don't have access to the busy pin, we apply the given + * command delay + */ + if (!this->dev_ready) { + udelay (this->chip_delay); + return; + } + + /* program and erase have their own busy handlers */ + switch (command) { + case NAND_CMD_PAGEPROG: + case NAND_CMD_ERASE1: + case NAND_CMD_ERASE2: + udelay (this->chip_delay); + return; + } + /* wait until command is processed */ + while (!this->dev_ready()); +} + +/* + * Get chip for selected access + */ +static inline void nand_get_chip (struct nand_chip *this, int new_state, int *erase_state) +{ + + DECLARE_WAITQUEUE (wait, current); + + /* + * Grab the lock and see if the device is available + * For erasing, we keep the spinlock until the + * erase command is written. + */ + +retry: + spin_lock_bh (&this->chip_lock); + + if (this->state == FL_READY) { + this->state = new_state; + if (new_state != FL_ERASING) + spin_unlock_bh (&this->chip_lock); + return; + } + + if (this->state == FL_ERASING) { + if (new_state != FL_ERASING) { + this->state = new_state; + spin_unlock_bh (&this->chip_lock); + return; + } + } + + set_current_state (TASK_UNINTERRUPTIBLE); + add_wait_queue (&this->wq, &wait); + spin_unlock_bh (&this->chip_lock); + schedule (); + remove_wait_queue (&this->wq, &wait); + goto retry; +} + +/* + * Wait for command done. This applies to erase and program only + * Erase can take up to 400ms and program up to 20ms according to + * general NAND and SmartMedia specs + * +*/ +static int nand_wait(struct mtd_info *mtd, struct nand_chip *this, int state) +{ + + DECLARE_WAITQUEUE (wait, current); + unsigned long timeo = jiffies; + int status; + + if (state == FL_ERASING) + timeo += (HZ * 400) / 1000; + else + timeo += (HZ * 20) / 1000; + + spin_lock_bh (&this->chip_lock); + while (time_before(jiffies, timeo)) { + /* Check, if we were interrupted */ + if (this->state != state) { + spin_unlock_bh (&this->chip_lock); + return 0; + } + if (this->dev_ready) { + if (this->dev_ready ()) + break; + } + this->cmdfunc (mtd, NAND_CMD_STATUS, -1, -1); + if (readb (this->IO_ADDR_R) & 0x40) + break; + + set_current_state (TASK_UNINTERRUPTIBLE); + add_wait_queue (&this->wq, &wait); + spin_unlock_bh (&this->chip_lock); + schedule_timeout (1); + remove_wait_queue (&this->wq, &wait); + spin_lock_bh (&this->chip_lock); + } + this->cmdfunc (mtd, NAND_CMD_STATUS, -1, -1); + status = (int) readb (this->IO_ADDR_R); + spin_unlock_bh (&this->chip_lock); - /* Pause for 15us */ - udelay (15); + return status; +} + +/* + * Nand_page_program function is used for write and writev ! + */ +static int nand_write_page (struct mtd_info *mtd, struct nand_chip *this, + int page, int col, int last, u_char * ecc_code) +{ + + int i, status; +#ifdef CONFIG_MTD_NAND_VERIFY_WRITE +#ifdef CONFIG_MTD_NAND_ECC + int ecc_bytes = (mtd->oobblock == 512) ? 6 : 3; +#endif +#endif + /* pad oob area */ + for (i = mtd->oobblock; i < mtd->oobblock + mtd->oobsize; i++) + this->data_buf[i] = 0xff; + +#ifdef CONFIG_MTD_NAND_ECC + /* Zero out the ECC array */ + for (i = 0; i < 6; i++) + ecc_code[i] = 0x00; + + /* Read back previous written data, if col > 0 */ + if (col) { + this->cmdfunc (mtd, NAND_CMD_READ0, col, page); + for (i = 0; i < col; i++) + this->data_buf[i] = readb (this->IO_ADDR_R); + } + + /* Calculate and write the ECC if we have enough data */ + if ((col < mtd->eccsize) && (last >= mtd->eccsize)) { + nand_calculate_ecc (&this->data_buf[0], &(ecc_code[0])); + for (i = 0; i < 3; i++) + this->data_buf[(mtd->oobblock + oob_config.ecc_pos[i])] = ecc_code[i]; + if (oob_config.eccvalid_pos != -1) + this->data_buf[mtd->oobblock + oob_config.eccvalid_pos] = 0xf0; + } + + /* Calculate and write the second ECC if we have enough data */ + if ((mtd->oobblock == 512) && (last == mtd->oobblock)) { + nand_calculate_ecc (&this->data_buf[256], &(ecc_code[3])); + for (i = 3; i < 6; i++) + this->data_buf[(mtd->oobblock + oob_config.ecc_pos[i])] = ecc_code[i]; + if (oob_config.eccvalid_pos != -1) + this->data_buf[mtd->oobblock + oob_config.eccvalid_pos] &= 0x0f; + } +#endif + /* Prepad for partial page programming !!! */ + for (i = 0; i < col; i++) + this->data_buf[i] = 0xff; + + /* Postpad for partial page programming !!! oob is already padded */ + for (i = last; i < mtd->oobblock; i++) + this->data_buf[i] = 0xff; + + /* Send command to begin auto page programming */ + this->cmdfunc (mtd, NAND_CMD_SEQIN, 0x00, page); + + /* Write out complete page of data */ + for (i = 0; i < (mtd->oobblock + mtd->oobsize); i++) + writeb (this->data_buf[i], this->IO_ADDR_W); + + /* Send command to actually program the data */ + this->cmdfunc (mtd, NAND_CMD_PAGEPROG, -1, -1); + + /* call wait ready function */ + status = this->waitfunc (mtd, this, FL_WRITING); + + /* See if device thinks it succeeded */ + if (status & 0x01) { + DEBUG (MTD_DEBUG_LEVEL0, "nand_write_ecc: " "Failed write, page 0x%08x, ", page); + return -EIO; + } +#ifdef CONFIG_MTD_NAND_VERIFY_WRITE + /* + * The NAND device assumes that it is always writing to + * a cleanly erased page. Hence, it performs its internal + * write verification only on bits that transitioned from + * 1 to 0. The device does NOT verify the whole page on a + * byte by byte basis. It is possible that the page was + * not completely erased or the page is becoming unusable + * due to wear. The read with ECC would catch the error + * later when the ECC page check fails, but we would rather + * catch it early in the page write stage. Better to write + * no data than invalid data. + */ + + /* Send command to read back the page */ + if (col < mtd->eccsize) + this->cmdfunc (mtd, NAND_CMD_READ0, col, page); + else + this->cmdfunc (mtd, NAND_CMD_READ1, col - 256, page); + + /* Loop through and verify the data */ + for (i = col; i < last; i++) { + if (this->data_buf[i] != readb (this->IO_ADDR_R)) { + DEBUG (MTD_DEBUG_LEVEL0, "nand_write_ecc: " "Failed write verify, page 0x%08x ", page); + return -EIO; + } + } + +#ifdef CONFIG_MTD_NAND_ECC + /* + * We also want to check that the ECC bytes wrote + * correctly for the same reasons stated above. + */ + this->cmdfunc (mtd, NAND_CMD_READOOB, 0x00, page); + for (i = 0; i < mtd->oobsize; i++) + this->data_buf[i] = readb (this->IO_ADDR_R); + for (i = 0; i < ecc_bytes; i++) { + if ((this->data_buf[(oob_config.ecc_pos[i])] != ecc_code[i]) && ecc_code[i]) { + DEBUG (MTD_DEBUG_LEVEL0, + "nand_write_ecc: Failed ECC write " + "verify, page 0x%08x, " "%6i bytes were succesful\n", page, i); + return -EIO; + } + } +#endif +#endif + return 0; } /* * NAND read */ -static int nand_read (struct mtd_info *mtd, loff_t from, size_t len, - size_t *retlen, u_char *buf) +static int nand_read (struct mtd_info *mtd, loff_t from, size_t len, size_t * retlen, u_char * buf) { #ifdef CONFIG_MTD_NAND_ECC struct nand_chip *this = mtd->priv; - + return nand_read_ecc (mtd, from, len, retlen, buf, this->ecc_code_buf); #else return nand_read_ecc (mtd, from, len, retlen, buf, NULL); @@ -144,54 +416,28 @@ * NAND read with ECC */ static int nand_read_ecc (struct mtd_info *mtd, loff_t from, size_t len, - size_t *retlen, u_char *buf, u_char *ecc_code) + size_t * retlen, u_char * buf, u_char * ecc_code) { - int j, col, page, state; + int j, col, page; int erase_state = 0; + int ecc_status = 0, ecc_failed = 0; struct nand_chip *this = mtd->priv; - DECLARE_WAITQUEUE(wait, current); + u_char *data_poi; #ifdef CONFIG_MTD_NAND_ECC - int ecc_result; u_char ecc_calc[6]; #endif - DEBUG (MTD_DEBUG_LEVEL3, - "nand_read_ecc: from = 0x%08x, len = %i\n", (unsigned int) from, - (int) len); + DEBUG (MTD_DEBUG_LEVEL3, "nand_read_ecc: from = 0x%08x, len = %i\n", (unsigned int) from, (int) len); /* Do not allow reads past end of device */ if ((from + len) > mtd->size) { - DEBUG (MTD_DEBUG_LEVEL0, - "nand_read_ecc: Attempt read beyond end of device\n"); + DEBUG (MTD_DEBUG_LEVEL0, "nand_read_ecc: Attempt read beyond end of device\n"); *retlen = 0; return -EINVAL; } /* Grab the lock and see if the device is available */ -retry: - spin_lock_bh (&this->chip_lock); - - switch (this->state) { - case FL_READY: - this->state = FL_READING; - spin_unlock_bh (&this->chip_lock); - break; - - case FL_ERASING: - this->state = FL_READING; - erase_state = 1; - spin_unlock_bh (&this->chip_lock); - break; - - default: - set_current_state (TASK_UNINTERRUPTIBLE); - add_wait_queue (&this->wq, &wait); - spin_unlock_bh (&this->chip_lock); - schedule(); - - remove_wait_queue (&this->wq, &wait); - goto retry; - }; + nand_get_chip (this, FL_READING, &erase_state); /* First we calculate the starting page */ page = from >> this->page_shift; @@ -199,12 +445,6 @@ /* Get raw starting column */ col = from & (mtd->oobblock - 1); - /* State machine for devices having pages larger than 256 bytes */ - state = (col < mtd->eccsize) ? 0 : 1; - - /* Calculate column address within ECC block context */ - col = (col >= mtd->eccsize) ? (col - mtd->eccsize) : col; - /* Initialize return value */ *retlen = 0; @@ -215,106 +455,95 @@ while (*retlen < len) { #ifdef CONFIG_MTD_NAND_ECC + + /* Do we have this page in cache ? */ + if (this->cache_page == page) + goto readdata; + /* Send the read command */ - if (!state) - nand_command (mtd, NAND_CMD_READ0, 0x00, page); - else - nand_command (mtd, NAND_CMD_READ1, 0x00, page); - - /* Read in a block big enough for ECC */ - for (j=0 ; j < mtd->eccsize ; j++) - this->data_buf[j] = readb (this->IO_ADDR); - - /* Read in the out-of-band data */ - if (!state) { - nand_command (mtd, NAND_CMD_READOOB, 0x00, page); - for (j=0 ; j<3 ; j++) - ecc_code[j] = readb(this->IO_ADDR); - nand_command (mtd, NAND_CMD_READ0, 0x00, page); - } - else { - nand_command (mtd, NAND_CMD_READOOB, 0x03, page); - for (j=3 ; j<6 ; j++) - ecc_code[j] = readb(this->IO_ADDR); - nand_command (mtd, NAND_CMD_READ0, 0x00, page); - } + this->cmdfunc (mtd, NAND_CMD_READ0, 0x00, page); + /* Read in a page + oob data */ + for (j = 0; j < mtd->oobblock + mtd->oobsize; j++) + this->data_buf[j] = readb (this->IO_ADDR_R); + this->cmdfunc (mtd, NAND_CMD_READ0, 0x00, page); + + /* copy data into cache, for read out of cache and if ecc fails */ + if (this->data_cache) + memcpy (this->data_cache, this->data_buf, mtd->oobblock + mtd->oobsize); + + /* Pick the ECC bytes out of the oob data */ + for (j = 0; j < 6; j++) + ecc_code[j] = this->data_buf[(mtd->oobblock + oob_config.ecc_pos[j])]; /* Calculate the ECC and verify it */ - if (!state) { - nand_calculate_ecc (&this->data_buf[0], - &ecc_calc[0]); - ecc_result = nand_correct_data (&this->data_buf[0], - &ecc_code[0], &ecc_calc[0]); - } - else { - nand_calculate_ecc (&this->data_buf[0], - &ecc_calc[3]); - ecc_result = nand_correct_data (&this->data_buf[0], - &ecc_code[3], &ecc_calc[3]); - } - if (ecc_result == -1) { - DEBUG (MTD_DEBUG_LEVEL0, - "nand_read_ecc: " \ - "Failed ECC read, page 0x%08x\n", page); - nand_deselect (); - spin_lock_bh (&this->chip_lock); - if (erase_state) - this->state = FL_ERASING; - else - this->state = FL_READY; - wake_up (&this->wq); - spin_unlock_bh (&this->chip_lock); - return -EIO; + /* If block was not written with ECC, skip ECC */ + if (oob_config.eccvalid_pos != -1 && + (this->data_buf[mtd->oobblock + oob_config.eccvalid_pos] & 0x0f) != 0x0f) { + + nand_calculate_ecc (&this->data_buf[0], &ecc_calc[0]); + switch (nand_correct_data (&this->data_buf[0], &ecc_code[0], &ecc_calc[0])) { + case -1: + DEBUG (MTD_DEBUG_LEVEL0, "nand_read_ecc: " "Failed ECC read, page 0x%08x\n", page); + ecc_failed++; + break; + case 1: + case 2: /* transfer ECC corrected data to cache */ + memcpy (this->data_cache, this->data_buf, 256); + break; + } } + if (oob_config.eccvalid_pos != -1 && + mtd->oobblock == 512 && (this->data_buf[mtd->oobblock + oob_config.eccvalid_pos] & 0xf0) != 0xf0) { + + nand_calculate_ecc (&this->data_buf[256], &ecc_calc[3]); + switch (nand_correct_data (&this->data_buf[256], &ecc_code[3], &ecc_calc[3])) { + case -1: + DEBUG (MTD_DEBUG_LEVEL0, "nand_read_ecc: " "Failed ECC read, page 0x%08x\n", page); + ecc_failed++; + break; + case 1: + case 2: /* transfer ECC corrected data to cache */ + if (this->data_cache) + memcpy (&this->data_cache[256], &this->data_buf[256], 256); + break; + } + } +readdata: /* Read the data from ECC data buffer into return buffer */ - if ((*retlen + (mtd->eccsize - col)) >= len) { + data_poi = (this->data_cache) ? this->data_cache : this->data_buf; + data_poi += col; + if ((*retlen + (mtd->oobblock - col)) >= len) { while (*retlen < len) - buf[(*retlen)++] = this->data_buf[col++]; - /* We're done */ - continue; + buf[(*retlen)++] = *data_poi++; + } else { + for (j = col; j < mtd->oobblock; j++) + buf[(*retlen)++] = *data_poi++; } - else - for (j=col ; j < mtd->eccsize ; j++) - buf[(*retlen)++] = this->data_buf[j]; + /* Set cache page address, invalidate, if ecc_failed */ + this->cache_page = (this->data_cache && !ecc_failed) ? page : -1; + + ecc_status += ecc_failed; + ecc_failed = 0; + #else /* Send the read command */ - if (!state) - nand_command (mtd, NAND_CMD_READ0, col, page); - else - nand_command (mtd, NAND_CMD_READ1, col, page); + this->cmdfunc (mtd, NAND_CMD_READ0, col, page); - /* Read the data directly into the return buffer */ - if ((*retlen + (mtd->eccsize - col)) >= len) { + /* Read the data directly into the return buffer */ + if ((*retlen + (mtd->oobblock - col)) >= len) { while (*retlen < len) - buf[(*retlen)++] = readb (this->IO_ADDR); + buf[(*retlen)++] = readb (this->IO_ADDR_R); /* We're done */ continue; - } - else - for (j=col ; j < mtd->eccsize ; j++) - buf[(*retlen)++] = readb (this->IO_ADDR); + } else + for (j = col; j < mtd->oobblock; j++) + buf[(*retlen)++] = readb (this->IO_ADDR_R); #endif - - /* - * If the amount of data to be read is greater than - * (256 - col), then all subsequent reads will take - * place on page or half-page (in the case of 512 byte - * page devices) aligned boundaries and the column - * address will be zero. Setting the column address to - * to zero after the first read allows us to simplify - * the reading of data and the if/else statements above. - */ - if (col) - col = 0x00; - + /* For subsequent reads align to page boundary. */ + col = 0; /* Increment page address */ - if ((mtd->oobblock == 256) || state) - page++; - - /* Toggle state machine */ - if (mtd->oobblock == 512) - state = state ? 0 : 1; + page++; } /* De-select the NAND device */ @@ -322,31 +551,28 @@ /* Wake up anyone waiting on the device */ spin_lock_bh (&this->chip_lock); - if (erase_state) - this->state = FL_ERASING; - else - this->state = FL_READY; + this->state = FL_READY; wake_up (&this->wq); spin_unlock_bh (&this->chip_lock); - - /* Return happy */ - return 0; + + /* + * Return success, if no ECC failures, else -EIO + * fs driver will take care of that, because + * retlen == desired len and result == -EIO + */ + return ecc_status ? -EIO : 0; } /* * NAND read out-of-band */ -static int nand_read_oob (struct mtd_info *mtd, loff_t from, size_t len, - size_t *retlen, u_char *buf) +static int nand_read_oob (struct mtd_info *mtd, loff_t from, size_t len, size_t * retlen, u_char * buf) { int i, col, page; int erase_state = 0; struct nand_chip *this = mtd->priv; - DECLARE_WAITQUEUE(wait, current); - - DEBUG (MTD_DEBUG_LEVEL3, - "nand_read_oob: from = 0x%08x, len = %i\n", (unsigned int) from, - (int) len); + + DEBUG (MTD_DEBUG_LEVEL3, "nand_read_oob: from = 0x%08x, len = %i\n", (unsigned int) from, (int) len); /* Shift to get page */ page = ((int) from) >> this->page_shift; @@ -357,59 +583,43 @@ /* Initialize return length value */ *retlen = 0; - /* Do not allow read past end of page */ - if ((col + len) > mtd->oobsize) { - DEBUG (MTD_DEBUG_LEVEL0, - "nand_read_oob: Attempt read past end of page " \ - "0x%08x, column %i, length %i\n", page, col, len); + /* Do not allow reads past end of device */ + if ((from + len) > mtd->size) { + DEBUG (MTD_DEBUG_LEVEL0, "nand_read_oob: Attempt read beyond end of device\n"); + *retlen = 0; return -EINVAL; } -retry: /* Grab the lock and see if the device is available */ - spin_lock_bh (&this->chip_lock); - - switch (this->state) { - case FL_READY: - this->state = FL_READING; - spin_unlock_bh (&this->chip_lock); - break; - - case FL_ERASING: - this->state = FL_READING; - erase_state = 1; - spin_unlock_bh (&this->chip_lock); - break; - - default: - set_current_state (TASK_UNINTERRUPTIBLE); - add_wait_queue (&this->wq, &wait); - spin_unlock_bh (&this->chip_lock); - schedule(); - - remove_wait_queue (&this->wq, &wait); - goto retry; - }; + nand_get_chip (this, FL_READING, &erase_state); - /* Select the NAND device */ - nand_select (); + /* can we read out of cache ? */ + if (this->cache_page == page && (col + len <= mtd->oobsize)) { + /* Read the data */ + memcpy (buf, &this->data_cache[mtd->oobblock + col], len); + } else { - /* Send the read command */ - nand_command (mtd, NAND_CMD_READOOB, col, page); + /* Select the NAND device */ + nand_select (); - /* Read the data */ - for (i = 0 ; i < len ; i++) - buf[i] = readb (this->IO_ADDR); - - /* De-select the NAND device */ - nand_deselect (); + /* Send the read command */ + this->cmdfunc (mtd, NAND_CMD_READOOB, col, page); + /* + * Read the data, if we read more than one page + * oob data, let the device transfer the data ! + */ + for (i = 0; i < len; i++) { + buf[i] = readb (this->IO_ADDR_R); + if ((col++ & (mtd->oobsize - 1)) == (mtd->oobsize - 1)) + udelay (this->chip_delay); + } + /* De-select the NAND device */ + nand_deselect (); + } /* Wake up anyone waiting on the device */ spin_lock_bh (&this->chip_lock); - if (erase_state) - this->state = FL_ERASING; - else - this->state = FL_READY; + this->state = FL_READY; wake_up (&this->wq); spin_unlock_bh (&this->chip_lock); @@ -421,12 +631,11 @@ /* * NAND write */ -static int nand_write (struct mtd_info *mtd, loff_t to, size_t len, - size_t *retlen, const u_char *buf) +static int nand_write (struct mtd_info *mtd, loff_t to, size_t len, size_t * retlen, const u_char * buf) { #ifdef CONFIG_MTD_NAND_ECC struct nand_chip *this = mtd->priv; - + return nand_write_ecc (mtd, to, len, retlen, buf, this->ecc_code_buf); #else return nand_write_ecc (mtd, to, len, retlen, buf, NULL); @@ -437,46 +646,21 @@ * NAND write with ECC */ static int nand_write_ecc (struct mtd_info *mtd, loff_t to, size_t len, - size_t *retlen, const u_char *buf, - u_char *ecc_code) + size_t * retlen, const u_char * buf, u_char * ecc_code) { - int i, page, col, cnt, status; + int i, page, col, cnt, ret = 0; struct nand_chip *this = mtd->priv; - DECLARE_WAITQUEUE(wait, current); -#ifdef CONFIG_MTD_NAND_ECC - int ecc_bytes = (mtd->oobblock == 512) ? 6 : 3; -#endif - DEBUG (MTD_DEBUG_LEVEL3, - "nand_write_ecc: to = 0x%08x, len = %i\n", (unsigned int) to, - (int) len); + DEBUG (MTD_DEBUG_LEVEL3, "nand_write_ecc: to = 0x%08x, len = %i\n", (unsigned int) to, (int) len); - /* Do not allow write past end of page */ + /* Do not allow write past end of device */ if ((to + len) > mtd->size) { - DEBUG (MTD_DEBUG_LEVEL0, - "nand_write_ecc: Attempted write past end of device\n"); + DEBUG (MTD_DEBUG_LEVEL0, "nand_write_oob: Attempt to write past end of page\n"); return -EINVAL; } -retry: /* Grab the lock and see if the device is available */ - spin_lock_bh (&this->chip_lock); - - switch (this->state) { - case FL_READY: - this->state = FL_WRITING; - spin_unlock_bh (&this->chip_lock); - break; - - default: - set_current_state (TASK_UNINTERRUPTIBLE); - add_wait_queue (&this->wq, &wait); - spin_unlock_bh (&this->chip_lock); - schedule(); - - remove_wait_queue (&this->wq, &wait); - goto retry; - }; + nand_get_chip (this, FL_WRITING, NULL); /* Shift to get page */ page = ((int) to) >> this->page_shift; @@ -491,184 +675,34 @@ nand_select (); /* Check the WP bit */ - nand_command (mtd, NAND_CMD_STATUS, -1, -1); - if (!(readb (this->IO_ADDR) & 0x80)) { - DEBUG (MTD_DEBUG_LEVEL0, - "nand_write_ecc: Device is write protected!!!\n"); - nand_deselect (); - spin_lock_bh (&this->chip_lock); - this->state = FL_READY; - wake_up (&this->wq); - spin_unlock_bh (&this->chip_lock); - return -EIO; + this->cmdfunc (mtd, NAND_CMD_STATUS, -1, -1); + if (!(readb (this->IO_ADDR_R) & 0x80)) { + DEBUG (MTD_DEBUG_LEVEL0, "nand_write_ecc: Device is write protected!!!\n"); + ret = -EIO; + goto out; } /* Loop until all data is written */ while (*retlen < len) { + /* Invalidate cache, if we write to this page */ + if (this->cache_page == page) + this->cache_page = -1; + /* Write data into buffer */ if ((col + len) >= mtd->oobblock) - for(i=col, cnt=0 ; i < mtd->oobblock ; i++, cnt++) + for (i = col, cnt = 0; i < mtd->oobblock; i++, cnt++) this->data_buf[i] = buf[(*retlen + cnt)]; else - for(i=col, cnt=0 ; cnt < (len - *retlen) ; i++, cnt++) + for (i = col, cnt = 0; cnt < (len - *retlen); i++, cnt++) this->data_buf[i] = buf[(*retlen + cnt)]; - -#ifdef CONFIG_MTD_NAND_ECC - /* Zero out the ECC array */ - for (i=0 ; i < 6 ; i++) - ecc_code[i] = 0x00; - - /* Calculate and write the ECC if we have enough data */ - if ((col < mtd->eccsize) && - ((col + (len - *retlen)) >= mtd->eccsize)) { - nand_command (mtd, NAND_CMD_READ0, col, page); - for (i=0 ; i < col ; i++) - this->data_buf[i] = readb (this->IO_ADDR); - nand_calculate_ecc (&this->data_buf[0], &ecc_code[0]); - for (i=0 ; i<3 ; i++) - this->data_buf[(mtd->oobblock + i)] = - ecc_code[i]; - } - /* Calculate and write the second ECC if we have enough data */ - if ((mtd->oobblock == 512) && - ((col + (len - *retlen)) >= mtd->oobblock)) { - nand_calculate_ecc (&this->data_buf[256], &ecc_code[3]); - for (i=3 ; i<6 ; i++) - this->data_buf[(mtd->oobblock + i)] = - ecc_code[i]; - } + /* We use the same function for write and writev !) */ + ret = nand_write_page (mtd, this, page, col, i, ecc_code); + if (ret) + goto out; - /* Write ones for partial page programming */ - for (i=ecc_bytes ; i < mtd->oobsize ; i++) - this->data_buf[(mtd->oobblock + i)] = 0xff; -#else - /* Write ones for partial page programming */ - for (i=mtd->oobblock ; i < (mtd->oobblock + mtd->oobsize) ; i++) - this->data_buf[i] = 0xff; -#endif - - /* Write pre-padding bytes into buffer */ - for (i=0 ; i < col ; i++) - this->data_buf[i] = 0xff; - - /* Write post-padding bytes into buffer */ - if ((col + (len - *retlen)) < mtd->oobblock) { - for(i=(col + cnt) ; i < mtd->oobblock ; i++) - this->data_buf[i] = 0xff; - } - - /* Send command to begin auto page programming */ - nand_command (mtd, NAND_CMD_SEQIN, 0x00, page); - - /* Write out complete page of data */ - for (i=0 ; i < (mtd->oobblock + mtd->oobsize) ; i++) - writeb (this->data_buf[i], this->IO_ADDR); - - /* Send command to actually program the data */ - nand_command (mtd, NAND_CMD_PAGEPROG, -1, -1); - - /* - * Wait for program operation to complete. This could - * take up to 3000us (3ms) on some devices, so we try - * and exit as quickly as possible. - */ - status = 0; - for (i=0 ; i<24 ; i++) { - /* Delay for 125us */ - udelay (125); - - /* Check the status */ - nand_command (mtd, NAND_CMD_STATUS, -1, -1); - status = (int) readb (this->IO_ADDR); - if (status & 0x40) - break; - } - - /* See if device thinks it succeeded */ - if (status & 0x01) { - DEBUG (MTD_DEBUG_LEVEL0, - "nand_write_ecc: " \ - "Failed write, page 0x%08x, " \ - "%6i bytes were succesful\n", page, *retlen); - nand_deselect (); - spin_lock_bh (&this->chip_lock); - this->state = FL_READY; - wake_up (&this->wq); - spin_unlock_bh (&this->chip_lock); - return -EIO; - } - -#ifdef CONFIG_MTD_NAND_VERIFY_WRITE - /* - * The NAND device assumes that it is always writing to - * a cleanly erased page. Hence, it performs its internal - * write verification only on bits that transitioned from - * 1 to 0. The device does NOT verify the whole page on a - * byte by byte basis. It is possible that the page was - * not completely erased or the page is becoming unusable - * due to wear. The read with ECC would catch the error - * later when the ECC page check fails, but we would rather - * catch it early in the page write stage. Better to write - * no data than invalid data. - */ - - /* Send command to read back the page */ - if (col < mtd->eccsize) - nand_command (mtd, NAND_CMD_READ0, col, page); - else - nand_command (mtd, NAND_CMD_READ1, col - 256, page); - - /* Loop through and verify the data */ - for (i=col ; i < cnt ; i++) { - if (this->data_buf[i] != readb (this->IO_ADDR)) { - DEBUG (MTD_DEBUG_LEVEL0, - "nand_write_ecc: " \ - "Failed write verify, page 0x%08x, " \ - "%6i bytes were succesful\n", - page, *retlen); - nand_deselect (); - spin_lock_bh (&this->chip_lock); - this->state = FL_READY; - wake_up (&this->wq); - spin_unlock_bh (&this->chip_lock); - return -EIO; - } - } - -#ifdef CONFIG_MTD_NAND_ECC - /* - * We also want to check that the ECC bytes wrote - * correctly for the same reasons stated above. - */ - nand_command (mtd, NAND_CMD_READOOB, 0x00, page); - for (i=0 ; i < ecc_bytes ; i++) { - if ((readb (this->IO_ADDR) != ecc_code[i]) && - ecc_code[i]) { - DEBUG (MTD_DEBUG_LEVEL0, - "nand_write_ecc: Failed ECC write " \ - "verify, page 0x%08x, " \ - "%6i bytes were succesful\n", - page, i); - nand_deselect (); - spin_lock_bh (&this->chip_lock); - this->state = FL_READY; - wake_up (&this->wq); - spin_unlock_bh (&this->chip_lock); - return -EIO; - } - } -#endif - -#endif - - /* - * If we are writing a large amount of data and/or it - * crosses page or half-page boundaries, we set the - * the column to zero. It simplifies the program logic. - */ - if (col) - col = 0x00; + /* Next data start at page boundary */ + col = 0; /* Update written bytes count */ *retlen += cnt; @@ -677,6 +711,10 @@ page++; } + /* Return happy */ + *retlen = len; + +out: /* De-select the NAND device */ nand_deselect (); @@ -686,24 +724,18 @@ wake_up (&this->wq); spin_unlock_bh (&this->chip_lock); - /* Return happy */ - *retlen = len; - return 0; + return ret; } /* * NAND write out-of-band */ -static int nand_write_oob (struct mtd_info *mtd, loff_t to, size_t len, - size_t *retlen, const u_char *buf) +static int nand_write_oob (struct mtd_info *mtd, loff_t to, size_t len, size_t * retlen, const u_char * buf) { - int i, column, page, status; + int i, column, page, status, ret = 0; struct nand_chip *this = mtd->priv; - DECLARE_WAITQUEUE(wait, current); - - DEBUG (MTD_DEBUG_LEVEL3, - "nand_write_oob: to = 0x%08x, len = %i\n", (unsigned int) to, - (int) len); + + DEBUG (MTD_DEBUG_LEVEL3, "nand_write_oob: to = 0x%08x, len = %i\n", (unsigned int) to, (int) len); /* Shift to get page */ page = ((int) to) >> this->page_shift; @@ -716,105 +748,69 @@ /* Do not allow write past end of page */ if ((column + len) > mtd->oobsize) { - DEBUG (MTD_DEBUG_LEVEL0, - "nand_write_oob: Attempt to write past end of page\n"); + DEBUG (MTD_DEBUG_LEVEL0, "nand_write_oob: Attempt to write past end of page\n"); return -EINVAL; } -retry: /* Grab the lock and see if the device is available */ - spin_lock_bh (&this->chip_lock); - - switch (this->state) { - case FL_READY: - this->state = FL_WRITING; - spin_unlock_bh (&this->chip_lock); - break; - - default: - set_current_state (TASK_UNINTERRUPTIBLE); - add_wait_queue (&this->wq, &wait); - spin_unlock_bh (&this->chip_lock); - schedule(); - - remove_wait_queue (&this->wq, &wait); - goto retry; - }; + nand_get_chip (this, FL_WRITING, NULL); /* Select the NAND device */ nand_select (); /* Check the WP bit */ - nand_command (mtd, NAND_CMD_STATUS, -1, -1); - if (!(readb (this->IO_ADDR) & 0x80)) { - DEBUG (MTD_DEBUG_LEVEL0, - "nand_write_oob: Device is write protected!!!\n"); - nand_deselect (); - spin_lock_bh (&this->chip_lock); - this->state = FL_READY; - wake_up (&this->wq); - spin_unlock_bh (&this->chip_lock); - return -EIO; - } + this->cmdfunc (mtd, NAND_CMD_STATUS, -1, -1); + if (!(readb (this->IO_ADDR_R) & 0x80)) { + DEBUG (MTD_DEBUG_LEVEL0, "nand_write_oob: Device is write protected!!!\n"); + ret = -EIO; + goto out; + } + + /* Invalidate cache, if we write to this page */ + if (this->cache_page == page) + this->cache_page = -1; + + /* Write ones for partial page programming */ + for (i = mtd->oobblock; i < (mtd->oobblock + mtd->oobsize); i++) + this->data_buf[i] = 0xff; + + /* Transfer data */ + for (i = 0; i < len; i++) + this->data_buf[i + mtd->oobblock + column] = buf[i]; /* Write out desired data */ - nand_command (mtd, NAND_CMD_SEQIN, column + 512, page); - for (i=0 ; iIO_ADDR); + this->cmdfunc (mtd, NAND_CMD_SEQIN, mtd->oobblock, page); + for (i = 0; i < mtd->oobsize; i++) + writeb (this->data_buf[i + mtd->oobblock], this->IO_ADDR_W); /* Send command to program the OOB data */ - nand_command (mtd, NAND_CMD_PAGEPROG, -1, -1); + this->cmdfunc (mtd, NAND_CMD_PAGEPROG, -1, -1); - /* - * Wait for program operation to complete. This could - * take up to 3000us (3ms) on some devices, so we try - * and exit as quickly as possible. - */ - status = 0; - for (i=0 ; i<24 ; i++) { - /* Delay for 125us */ - udelay (125); - - /* Check the status */ - nand_command (mtd, NAND_CMD_STATUS, -1, -1); - status = (int) readb (this->IO_ADDR); - if (status & 0x40) - break; - } + status = this->waitfunc (mtd, this, FL_WRITING); /* See if device thinks it succeeded */ if (status & 0x01) { - DEBUG (MTD_DEBUG_LEVEL0, - "nand_write_oob: " \ - "Failed write, page 0x%08x\n", page); - nand_deselect (); - spin_lock_bh (&this->chip_lock); - this->state = FL_READY; - wake_up (&this->wq); - spin_unlock_bh (&this->chip_lock); - return -EIO; + DEBUG (MTD_DEBUG_LEVEL0, "nand_write_oob: " "Failed write, page 0x%08x\n", page); + ret = -EIO; + goto out; } - #ifdef CONFIG_MTD_NAND_VERIFY_WRITE /* Send command to read back the data */ - nand_command (mtd, NAND_CMD_READOOB, column, page); + this->cmdfunc (mtd, NAND_CMD_READOOB, column, page); /* Loop through and verify the data */ - for (i=0 ; iIO_ADDR)) { - DEBUG (MTD_DEBUG_LEVEL0, - "nand_write_oob: " \ - "Failed write verify, page 0x%08x\n", page); - nand_deselect (); - spin_lock_bh (&this->chip_lock); - this->state = FL_READY; - wake_up (&this->wq); - spin_unlock_bh (&this->chip_lock); - return -EIO; + for (i = 0; i < len; i++) { + if (buf[i] != readb (this->IO_ADDR_R)) { + DEBUG (MTD_DEBUG_LEVEL0, "nand_write_oob: " "Failed write verify, page 0x%08x\n", page); + ret = -EIO; + goto out; } } #endif + /* Return happy */ + *retlen = len; +out: /* De-select the NAND device */ nand_deselect (); @@ -824,59 +820,33 @@ wake_up (&this->wq); spin_unlock_bh (&this->chip_lock); - /* Return happy */ - *retlen = len; - return 0; + return ret; } /* * NAND write with iovec */ -static int nand_writev (struct mtd_info *mtd, const struct iovec *vecs, - unsigned long count, loff_t to, size_t *retlen) +static int nand_writev (struct mtd_info *mtd, const struct iovec *vecs, unsigned long count, loff_t to, size_t * retlen) { - int i, page, col, cnt, len, total_len, status; + int i, page, col, cnt, len, total_len, ret = 0; struct nand_chip *this = mtd->priv; - DECLARE_WAITQUEUE(wait, current); -#ifdef CONFIG_MTD_NAND_ECC - int ecc_bytes = (mtd->oobblock == 512) ? 6 : 3; -#endif /* Calculate total length of data */ total_len = 0; - for (i=0 ; i < count ; i++) + for (i = 0; i < count; i++) total_len += (int) vecs[i].iov_len; DEBUG (MTD_DEBUG_LEVEL3, - "nand_writev: to = 0x%08x, len = %i\n", (unsigned int) to, - (unsigned int) total_len); + "nand_writev: to = 0x%08x, len = %i, count = %ld\n", (unsigned int) to, (unsigned int) total_len, count); /* Do not allow write past end of page */ if ((to + total_len) > mtd->size) { - DEBUG (MTD_DEBUG_LEVEL0, - "nand_writev: Attempted write past end of device\n"); + DEBUG (MTD_DEBUG_LEVEL0, "nand_writev: Attempted write past end of device\n"); return -EINVAL; } -retry: /* Grab the lock and see if the device is available */ - spin_lock_bh (&this->chip_lock); - - switch (this->state) { - case FL_READY: - this->state = FL_WRITING; - spin_unlock_bh (&this->chip_lock); - break; - - default: - set_current_state (TASK_UNINTERRUPTIBLE); - add_wait_queue (&this->wq, &wait); - spin_unlock_bh (&this->chip_lock); - schedule(); - - remove_wait_queue (&this->wq, &wait); - goto retry; - }; + nand_get_chip (this, FL_WRITING, NULL); /* Shift to get page */ page = ((int) to) >> this->page_shift; @@ -891,181 +861,46 @@ nand_select (); /* Check the WP bit */ - nand_command (mtd, NAND_CMD_STATUS, -1, -1); - if (!(readb (this->IO_ADDR) & 0x80)) { - DEBUG (MTD_DEBUG_LEVEL0, - "nand_writev: Device is write protected!!!\n"); - nand_deselect (); - spin_lock_bh (&this->chip_lock); - this->state = FL_READY; - wake_up (&this->wq); - spin_unlock_bh (&this->chip_lock); - return -EIO; + this->cmdfunc (mtd, NAND_CMD_STATUS, -1, -1); + if (!(readb (this->IO_ADDR_R) & 0x80)) { + DEBUG (MTD_DEBUG_LEVEL0, "nand_writev: Device is write protected!!!\n"); + ret = -EIO; + goto out; } /* Loop until all iovecs' data has been written */ cnt = col; len = 0; while (count) { + /* Invalidate cache, if we write to this page */ + if (this->cache_page == page) + this->cache_page = -1; + /* Do any need pre-fill for partial page programming */ - for (i=0 ; i < cnt ; i++) + for (i = 0; i < cnt; i++) this->data_buf[i] = 0xff; /* * Read data out of each tuple until we have a full page * to write or we've read all the tuples. */ + while ((cnt < mtd->oobblock) && count) { - this->data_buf[cnt++] = - ((u_char *) vecs->iov_base)[len++]; + if (vecs->iov_base != NULL && vecs->iov_len) { + this->data_buf[cnt++] = ((u_char *) vecs->iov_base)[len++]; + } if (len >= (int) vecs->iov_len) { vecs++; len = 0; count--; } } - - /* Do any need post-fill for partial page programming */ - for (i=cnt ; i < mtd->oobblock ; i++) - this->data_buf[i] = 0xff; - -#ifdef CONFIG_MTD_NAND_ECC - /* Zero out the ECC array */ - for (i=0 ; i < 6 ; i++) - this->ecc_code_buf[i] = 0x00; - - /* Calculate and write the first ECC */ - if (col >= mtd->eccsize) { - nand_command (mtd, NAND_CMD_READ0, col, page); - for (i=0 ; i < col ; i++) - this->data_buf[i] = readb (this->IO_ADDR); - nand_calculate_ecc (&this->data_buf[0], - &(this->ecc_code_buf[0])); - for (i=0 ; i<3 ; i++) - this->data_buf[(mtd->oobblock + i)] = - this->ecc_code_buf[i]; - } - - /* Calculate and write the second ECC */ - if ((mtd->oobblock == 512) && (cnt == mtd->oobblock)) { - nand_calculate_ecc (&this->data_buf[256], - &(this->ecc_code_buf[3])); - for (i=3 ; i<6 ; i++) - this->data_buf[(mtd->oobblock + i)] = - this->ecc_code_buf[i]; - } - - /* Write ones for partial page programming */ - for (i=ecc_bytes ; i < mtd->oobsize ; i++) - this->data_buf[(mtd->oobblock + i)] = 0xff; -#else - /* Write ones for partial page programming */ - for (i=mtd->oobblock ; i < (mtd->oobblock + mtd->oobsize) ; i++) - this->data_buf[i] = 0xff; -#endif - /* Send command to begin auto page programming */ - nand_command (mtd, NAND_CMD_SEQIN, 0x00, page); - /* Write out complete page of data */ - for (i=0 ; i < (mtd->oobblock + mtd->oobsize) ; i++) - writeb (this->data_buf[i], this->IO_ADDR); - - /* Send command to actually program the data */ - nand_command (mtd, NAND_CMD_PAGEPROG, -1, -1); - - /* - * Wait for program operation to complete. This could - * take up to 3000us (3ms) on some devices, so we try - * and exit as quickly as possible. - */ - status = 0; - for (i=0 ; i<24 ; i++) { - /* Delay for 125us */ - udelay (125); - - /* Check the status */ - nand_command (mtd, NAND_CMD_STATUS, -1, -1); - status = (int) readb (this->IO_ADDR); - if (status & 0x40) - break; - } - - /* See if device thinks it succeeded */ - if (status & 0x01) { - DEBUG (MTD_DEBUG_LEVEL0, - "nand_writev: " \ - "Failed write, page 0x%08x, " \ - "%6i bytes were succesful\n", page, *retlen); - nand_deselect (); - spin_lock_bh (&this->chip_lock); - this->state = FL_READY; - wake_up (&this->wq); - spin_unlock_bh (&this->chip_lock); - return -EIO; - } - -#ifdef CONFIG_MTD_NAND_VERIFY_WRITE - /* - * The NAND device assumes that it is always writing to - * a cleanly erased page. Hence, it performs its internal - * write verification only on bits that transitioned from - * 1 to 0. The device does NOT verify the whole page on a - * byte by byte basis. It is possible that the page was - * not completely erased or the page is becoming unusable - * due to wear. The read with ECC would catch the error - * later when the ECC page check fails, but we would rather - * catch it early in the page write stage. Better to write - * no data than invalid data. - */ - - /* Send command to read back the page */ - if (col < mtd->eccsize) - nand_command (mtd, NAND_CMD_READ0, col, page); - else - nand_command (mtd, NAND_CMD_READ1, col - 256, page); - - /* Loop through and verify the data */ - for (i=col ; i < cnt ; i++) { - if (this->data_buf[i] != readb (this->IO_ADDR)) { - DEBUG (MTD_DEBUG_LEVEL0, - "nand_writev: " \ - "Failed write verify, page 0x%08x, " \ - "%6i bytes were succesful\n", - page, *retlen); - nand_deselect (); - spin_lock_bh (&this->chip_lock); - this->state = FL_READY; - wake_up (&this->wq); - spin_unlock_bh (&this->chip_lock); - return -EIO; - } - } - -#ifdef CONFIG_MTD_NAND_ECC - /* - * We also want to check that the ECC bytes wrote - * correctly for the same reasons stated above. - */ - nand_command (mtd, NAND_CMD_READOOB, 0x00, page); - for (i=0 ; i < ecc_bytes ; i++) { - if ((readb (this->IO_ADDR) != this->ecc_code_buf[i]) && - this->ecc_code_buf[i]) { - DEBUG (MTD_DEBUG_LEVEL0, - "nand_writev: Failed ECC write " \ - "verify, page 0x%08x, " \ - "%6i bytes were succesful\n", - page, i); - nand_deselect (); - spin_lock_bh (&this->chip_lock); - this->state = FL_READY; - wake_up (&this->wq); - spin_unlock_bh (&this->chip_lock); - return -EIO; - } - } -#endif + /* We use the same function for write and writev !) */ + ret = nand_write_page (mtd, this, page, col, cnt, this->ecc_code_buf); + if (ret) + goto out; -#endif /* Update written bytes count */ *retlen += (cnt - col); @@ -1076,6 +911,7 @@ page++; } +out: /* De-select the NAND device */ nand_deselect (); @@ -1086,7 +922,7 @@ spin_unlock_bh (&this->chip_lock); /* Return happy */ - return 0; + return ret; } /* @@ -1094,53 +930,33 @@ */ static int nand_erase (struct mtd_info *mtd, struct erase_info *instr) { - int i, page, len, status, pages_per_block; + int page, len, status, pages_per_block, ret; struct nand_chip *this = mtd->priv; - DECLARE_WAITQUEUE(wait, current); + DECLARE_WAITQUEUE (wait, current); DEBUG (MTD_DEBUG_LEVEL3, - "nand_erase: start = 0x%08x, len = %i\n", - (unsigned int) instr->addr, (unsigned int) instr->len); + "nand_erase: start = 0x%08x, len = %i\n", (unsigned int) instr->addr, (unsigned int) instr->len); /* Start address must align on block boundary */ if (instr->addr & (mtd->erasesize - 1)) { - DEBUG (MTD_DEBUG_LEVEL0, - "nand_erase: Unaligned address\n"); + DEBUG (MTD_DEBUG_LEVEL0, "nand_erase: Unaligned address\n"); return -EINVAL; } /* Length must align on block boundary */ if (instr->len & (mtd->erasesize - 1)) { - DEBUG (MTD_DEBUG_LEVEL0, - "nand_erase: Length not block aligned\n"); + DEBUG (MTD_DEBUG_LEVEL0, "nand_erase: Length not block aligned\n"); return -EINVAL; } /* Do not allow erase past end of device */ if ((instr->len + instr->addr) > mtd->size) { - DEBUG (MTD_DEBUG_LEVEL0, - "nand_erase: Erase past end of device\n"); + DEBUG (MTD_DEBUG_LEVEL0, "nand_erase: Erase past end of device\n"); return -EINVAL; } -retry: /* Grab the lock and see if the device is available */ - spin_lock_bh (&this->chip_lock); - - switch (this->state) { - case FL_READY: - this->state = FL_ERASING; - break; - - default: - set_current_state (TASK_UNINTERRUPTIBLE); - add_wait_queue (&this->wq, &wait); - spin_unlock_bh (&this->chip_lock); - schedule(); - - remove_wait_queue (&this->wq, &wait); - goto retry; - }; + nand_get_chip (this, FL_ERASING, NULL); /* Shift to get first page */ page = (int) (instr->addr >> this->page_shift); @@ -1152,81 +968,84 @@ nand_select (); /* Check the WP bit */ - nand_command (mtd, NAND_CMD_STATUS, -1, -1); - if (!(readb (this->IO_ADDR) & 0x80)) { - DEBUG (MTD_DEBUG_LEVEL0, - "nand_erase: Device is write protected!!!\n"); - nand_deselect (); - this->state = FL_READY; - spin_unlock_bh (&this->chip_lock); - return -EIO; + this->cmdfunc (mtd, NAND_CMD_STATUS, -1, -1); + if (!(readb (this->IO_ADDR_R) & 0x80)) { + DEBUG (MTD_DEBUG_LEVEL0, "nand_erase: Device is write protected!!!\n"); + instr->state = MTD_ERASE_FAILED; + goto erase_exit; } /* Loop through the pages */ len = instr->len; + + instr->state = MTD_ERASING; + while (len) { + if (oob_config.badblock_pos != -1) { + /* Check if we have a bad block, we do not erase bad blocks ! */ + this->cmdfunc (mtd, NAND_CMD_READOOB, oob_config.badblock_pos, page); + if (readb (this->IO_ADDR_R) != 0xff) { + printk (KERN_WARNING "nand_erase: attempt to erase a bad block at page 0x%08x\n", page); + instr->state = MTD_ERASE_FAILED; + goto erase_exit; + } + } + /* Send commands to erase a page */ - nand_command(mtd, NAND_CMD_ERASE1, -1, page); - nand_command(mtd, NAND_CMD_ERASE2, -1, -1); + this->cmdfunc (mtd, NAND_CMD_ERASE1, -1, page); + this->cmdfunc (mtd, NAND_CMD_ERASE2, -1, -1); - /* - * Wait for program operation to complete. This could - * take up to 4000us (4ms) on some devices, so we try - * and exit as quickly as possible. - */ - status = 0; - for (i=0 ; i<32 ; i++) { - /* Delay for 125us */ - udelay (125); - - /* Check the status */ - nand_command (mtd, NAND_CMD_STATUS, -1, -1); - status = (int) readb (this->IO_ADDR); - if (status & 0x40) - break; - } + spin_unlock_bh (&this->chip_lock); + status = this->waitfunc (mtd, this, FL_ERASING); + /* Get spinlock, in case we exit */ + spin_lock_bh (&this->chip_lock); /* See if block erase succeeded */ if (status & 0x01) { - DEBUG (MTD_DEBUG_LEVEL0, - "nand_erase: " \ - "Failed erase, page 0x%08x\n", page); - nand_deselect (); - this->state = FL_READY; - spin_unlock_bh (&this->chip_lock); - return -EIO; + DEBUG (MTD_DEBUG_LEVEL0, "nand_erase: " "Failed erase, page 0x%08x\n", page); + instr->state = MTD_ERASE_FAILED; + goto erase_exit; + } + + /* Check, if we were interupted */ + if (this->state == FL_ERASING) { + /* Invalidate cache, if last_page is inside erase-block */ + if (this->cache_page >= page && this->cache_page < (page + pages_per_block)) + this->cache_page = -1; + + /* Increment page address and decrement length */ + len -= mtd->erasesize; + page += pages_per_block; } - - /* Increment page address and decrement length */ - len -= mtd->erasesize; - page += pages_per_block; - /* Release the spin lock */ spin_unlock_bh (&this->chip_lock); - erase_retry: - /* Check the state and sleep if it changed */ spin_lock_bh (&this->chip_lock); - if (this->state == FL_ERASING) { + /* Check the state and sleep if it changed */ + if (this->state == FL_ERASING || this->state == FL_READY) { + /* Select the NAND device again, if we were interrupted */ + this->state = FL_ERASING; + nand_select (); continue; - } - else { + } else { set_current_state (TASK_UNINTERRUPTIBLE); add_wait_queue (&this->wq, &wait); spin_unlock_bh (&this->chip_lock); - schedule(); - + schedule (); remove_wait_queue (&this->wq, &wait); goto erase_retry; } } - spin_unlock_bh (&this->chip_lock); + instr->state = MTD_ERASE_DONE; +erase_exit: /* De-select the NAND device */ nand_deselect (); + spin_unlock_bh (&this->chip_lock); + ret = instr->state == MTD_ERASE_DONE ? 0 : -EIO;; /* Do call back function */ - if (instr->callback) + if (!ret && instr->callback) instr->callback (instr); /* The device is ready */ @@ -1234,8 +1053,8 @@ this->state = FL_READY; spin_unlock_bh (&this->chip_lock); - /* Return happy */ - return 0; + /* Return more or less happy */ + return ret; } /* @@ -1244,16 +1063,16 @@ static void nand_sync (struct mtd_info *mtd) { struct nand_chip *this = mtd->priv; - DECLARE_WAITQUEUE(wait, current); + DECLARE_WAITQUEUE (wait, current); DEBUG (MTD_DEBUG_LEVEL3, "nand_sync: called\n"); retry: /* Grab the spinlock */ - spin_lock_bh(&this->chip_lock); + spin_lock_bh (&this->chip_lock); /* See what's going on */ - switch(this->state) { + switch (this->state) { case FL_READY: case FL_SYNCING: this->state = FL_SYNCING; @@ -1270,7 +1089,7 @@ goto retry; } - /* Lock the device */ + /* Lock the device */ spin_lock_bh (&this->chip_lock); /* Set the device to be ready again */ @@ -1279,7 +1098,7 @@ wake_up (&this->wq); } - /* Unlock the device */ + /* Unlock the device */ spin_unlock_bh (&this->chip_lock); } @@ -1291,20 +1110,31 @@ int i, nand_maf_id, nand_dev_id; struct nand_chip *this = mtd->priv; + /* check for proper chip_delay setup, set 20us if not */ + if (!this->chip_delay) + this->chip_delay = 20; + + /* check, if a user supplied command function given */ + if (this->cmdfunc == NULL) + this->cmdfunc = nand_command; + + /* check, if a user supplied wait function given */ + if (this->waitfunc == NULL) + this->waitfunc = nand_wait; + /* Select the device */ nand_select (); /* Send the command for reading device ID */ - nand_command (mtd, NAND_CMD_READID, 0x00, -1); + this->cmdfunc (mtd, NAND_CMD_READID, 0x00, -1); /* Read manufacturer and device IDs */ - nand_maf_id = readb (this->IO_ADDR); - nand_dev_id = readb (this->IO_ADDR); + nand_maf_id = readb (this->IO_ADDR_R); + nand_dev_id = readb (this->IO_ADDR_R); /* Print and store flash device information */ for (i = 0; nand_flash_ids[i].name != NULL; i++) { - if (nand_maf_id == nand_flash_ids[i].manufacture_id && - nand_dev_id == nand_flash_ids[i].model_id) { + if (nand_maf_id == nand_flash_ids[i].manufacture_id && nand_dev_id == nand_flash_ids[i].model_id) { if (!mtd->size) { mtd->name = nand_flash_ids[i].name; mtd->erasesize = nand_flash_ids[i].erasesize; @@ -1314,26 +1144,47 @@ mtd->oobblock = 256; mtd->oobsize = 8; this->page_shift = 8; - } - else { + } else { mtd->oobblock = 512; mtd->oobsize = 16; this->page_shift = 9; } } - printk (KERN_INFO "NAND device: Manufacture ID:" \ - " 0x%02x, Chip ID: 0x%02x (%s)\n", - nand_maf_id, nand_dev_id, mtd->name); + printk (KERN_INFO "NAND device: Manufacture ID:" + " 0x%02x, Chip ID: 0x%02x (%s)\n", nand_maf_id, nand_dev_id, mtd->name); break; } } - /* Initialize state and spinlock */ + /* Initialize state, waitqueue and spinlock */ this->state = FL_READY; - spin_lock_init(&this->chip_lock); + init_waitqueue_head (&this->wq); + spin_lock_init (&this->chip_lock); /* De-select the device */ nand_deselect (); + /* + * Preset out of band configuration + */ +#ifdef CONFIG_MTD_NAND_ECC_JFFS2 + oob_config.ecc_pos[0] = NAND_JFFS2_OOB_ECCPOS0; + oob_config.ecc_pos[1] = NAND_JFFS2_OOB_ECCPOS1; + oob_config.ecc_pos[2] = NAND_JFFS2_OOB_ECCPOS2; + oob_config.ecc_pos[3] = NAND_JFFS2_OOB_ECCPOS3; + oob_config.ecc_pos[4] = NAND_JFFS2_OOB_ECCPOS4; + oob_config.ecc_pos[5] = NAND_JFFS2_OOB_ECCPOS5; + oob_config.badblock_pos = 5; + oob_config.eccvalid_pos = 4; +#else + oob_config.ecc_pos[0] = NAND_NOOB_ECCPOS0; + oob_config.ecc_pos[1] = NAND_NOOB_ECCPOS1; + oob_config.ecc_pos[2] = NAND_NOOB_ECCPOS2; + oob_config.ecc_pos[3] = NAND_NOOB_ECCPOS3; + oob_config.ecc_pos[4] = NAND_NOOB_ECCPOS4; + oob_config.ecc_pos[5] = NAND_NOOB_ECCPOS5; + oob_config.badblock_pos = NAND_NOOB_BADBPOS; + oob_config.eccvalid_pos = NAND_NOOB_ECCVPOS; +#endif /* Print warning message for no device */ if (!mtd->size) { @@ -1367,8 +1218,8 @@ return 0; } -EXPORT_SYMBOL(nand_scan); +EXPORT_SYMBOL (nand_scan); -MODULE_LICENSE("GPL"); -MODULE_AUTHOR("Steven J. Hill IO_ADDR = spia_fio_base; - this->CTRL_ADDR = spia_io_base + spia_pedr; - this->CLE = 0x01; - this->ALE = 0x02; - this->NCE = 0x04; + this->IO_ADDR_R = spia_fio_base; + this->IO_ADDR_W = spia_fio_base; + /* Set address of hardware control function */ + this->hwcontrol = spia_hwcontrol; + /* 15 us command delay time */ + this->chip_delay = 15; /* Scan to find existence of the device */ if (nand_scan (spia_mtd)) { @@ -127,6 +152,16 @@ return -ENOMEM; } + /* Allocate memory for internal data buffer */ + this->data_cache = kmalloc (sizeof(u_char) * (spia_mtd->oobblock + spia_mtd->oobsize), GFP_KERNEL); + if (!this->data_cache) { + printk ("Unable to allocate NAND data cache for SPIA.\n"); + kfree (this->data_buf); + kfree (spia_mtd); + return = -ENOMEM; + } + this->cache_page = -1; + /* Register the partitions */ add_mtd_partitions(spia_mtd, partition_info, NUM_PARTITIONS); @@ -148,6 +183,7 @@ /* Free internal data buffer */ kfree (this->data_buf); + kfree (this->page_cache); /* Free the MTD device structure */ kfree (spia_mtd); diff -urN orig/drivers/mtd/nftlcore.c linux/drivers/mtd/nftlcore.c --- orig/drivers/mtd/nftlcore.c Mon Aug 5 13:31:03 2002 +++ linux/drivers/mtd/nftlcore.c Mon Aug 5 13:44:09 2002 @@ -1,7 +1,7 @@ /* Linux driver for NAND Flash Translation Layer */ /* (c) 1999 Machine Vision Holdings, Inc. */ /* Author: David Woodhouse */ -/* $Id: nftlcore.c,v 1.85 2001/11/20 11:42:33 dwmw2 Exp $ */ +/* $Id: nftlcore.c,v 1.86 2002/03/06 14:38:17 dwmw2 Exp $ */ /* The contents of this file are distributed under the GNU General @@ -118,9 +118,6 @@ init_MUTEX(&nftl->mutex); - /* get physical parameters */ - nftl->EraseSize = mtd->erasesize; - nftl->nb_blocks = mtd->size / mtd->erasesize; nftl->mtd = mtd; if (NFTL_mount(nftl) < 0) { @@ -1062,7 +1059,7 @@ int i; #ifdef PRERELEASE - printk(KERN_INFO "NFTL driver: nftlcore.c $Revision: 1.85 $, nftlmount.c %s\n", nftlmountrev); + printk(KERN_INFO "NFTL driver: nftlcore.c $Revision: 1.86 $, nftlmount.c %s\n", nftlmountrev); #endif if (register_blkdev(MAJOR_NR, "nftl", &nftl_fops)){ diff -urN orig/drivers/mtd/nftlmount.c linux/drivers/mtd/nftlmount.c --- orig/drivers/mtd/nftlmount.c Wed Feb 27 14:25:03 2002 +++ linux/drivers/mtd/nftlmount.c Wed Mar 13 15:54:50 2002 @@ -4,7 +4,7 @@ * Author: Fabrice Bellard (fabrice.bellard@netgem.com) * Copyright (C) 2000 Netgem S.A. * - * $Id: nftlmount.c,v 1.25 2001/11/30 16:46:27 dwmw2 Exp $ + * $Id: nftlmount.c,v 1.28 2002/03/13 07:31:25 dwmw2 Exp $ * * 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 @@ -39,7 +39,7 @@ #define SECTORSIZE 512 -char nftlmountrev[]="$Revision: 1.25 $"; +char nftlmountrev[]="$Revision: 1.28 $"; /* find_boot_record: Find the NFTL Media Header and its Spare copy which contains the * various device information of the NFTL partition and Bad Unit Table. Update @@ -56,6 +56,11 @@ struct NFTLMediaHeader *mh = &nftl->MediaHdr; unsigned int i; + /* Assume logical EraseSize == physical erasesize for starting the scan. + We'll sort it out later if we find a MediaHeader which says otherwise */ + nftl->EraseSize = nftl->mtd->erasesize; + nftl->nb_blocks = nftl->mtd->size / nftl->EraseSize; + nftl->MediaUnit = BLOCK_NIL; nftl->SpareMediaUnit = BLOCK_NIL; @@ -122,7 +127,6 @@ continue; } #endif - /* OK, we like it. */ if (boot_record_count) { @@ -137,6 +141,10 @@ if (boot_record_count == 1) nftl->SpareMediaUnit = block; + /* Mark this boot record (NFTL MediaHeader) block as reserved */ + nftl->ReplUnitTable[block] = BLOCK_RESERVED; + + boot_record_count++; continue; } @@ -145,12 +153,18 @@ memcpy(mh, buf, sizeof(struct NFTLMediaHeader)); /* Do some sanity checks on it */ - if (mh->UnitSizeFactor != 0xff) { - printk(KERN_NOTICE "Sorry, we don't support UnitSizeFactor " - "of != 1 yet.\n"); + if (mh->UnitSizeFactor == 0) { + printk(KERN_NOTICE "NFTL: UnitSizeFactor 0x00 detected. This violates the spec but we think we know what it means...\n"); + } else if (mh->UnitSizeFactor < 0xfc) { + printk(KERN_NOTICE "Sorry, we don't support UnitSizeFactor 0x%02x\n", + mh->UnitSizeFactor); return -1; + } else if (mh->UnitSizeFactor != 0xff) { + printk(KERN_NOTICE "WARNING: Support for NFTL with UnitSizeFactor 0x%02x is experimental\n", + mh->UnitSizeFactor); + nftl->EraseSize = nftl->mtd->erasesize << (0xff - mh->UnitSizeFactor); + nftl->nb_blocks = nftl->mtd->size / nftl->EraseSize; } - nftl->nb_boot_blocks = le16_to_cpu(mh->FirstPhysicalEUN); if ((nftl->nb_boot_blocks + 2) >= nftl->nb_blocks) { printk(KERN_NOTICE "NFTL Media Header sanity check failed:\n"); @@ -168,11 +182,39 @@ } nftl->nr_sects = nftl->numvunits * (nftl->EraseSize / SECTORSIZE); - + /* If we're not using the last sectors in the device for some reason, reduce nb_blocks accordingly so we forget they're there */ nftl->nb_blocks = le16_to_cpu(mh->NumEraseUnits) + le16_to_cpu(mh->FirstPhysicalEUN); + /* XXX: will be suppressed */ + nftl->lastEUN = nftl->nb_blocks - 1; + + /* memory alloc */ + nftl->EUNtable = kmalloc(nftl->nb_blocks * sizeof(u16), GFP_KERNEL); + if (!nftl->EUNtable) { + printk(KERN_NOTICE "NFTL: allocation of EUNtable failed\n"); + return -ENOMEM; + } + + nftl->ReplUnitTable = kmalloc(nftl->nb_blocks * sizeof(u16), GFP_KERNEL); + if (!nftl->ReplUnitTable) { + kfree(nftl->EUNtable); + printk(KERN_NOTICE "NFTL: allocation of ReplUnitTable failed\n"); + return -ENOMEM; + } + + /* mark the bios blocks (blocks before NFTL MediaHeader) as reserved */ + for (i = 0; i < nftl->nb_boot_blocks; i++) + nftl->ReplUnitTable[i] = BLOCK_RESERVED; + /* mark all remaining blocks as potentially containing data */ + for (; i < nftl->nb_blocks; i++) { + nftl->ReplUnitTable[i] = BLOCK_NOTEXPLORED; + } + + /* Mark this boot record (NFTL MediaHeader) block as reserved */ + nftl->ReplUnitTable[block] = BLOCK_RESERVED; + /* read the Bad Erase Unit Table and modify ReplUnitTable[] accordingly */ for (i = 0; i < nftl->nb_blocks; i++) { if ((i & (SECTORSIZE - 1)) == 0) { @@ -182,6 +224,8 @@ &retlen, buf, (char *)&oob)) < 0) { printk(KERN_NOTICE "Read of bad sector table failed (err %d)\n", ret); + kfree(nftl->ReplUnitTable); + kfree(nftl->EUNtable); return -1; } } @@ -505,41 +549,11 @@ struct nftl_uci1 h1; int retlen; - /* XXX: will be suppressed */ - s->lastEUN = s->nb_blocks - 1; - - /* memory alloc */ - s->EUNtable = kmalloc(s->nb_blocks * sizeof(u16), GFP_KERNEL); - s->ReplUnitTable = kmalloc(s->nb_blocks * sizeof(u16), GFP_KERNEL); - if (!s->EUNtable || !s->ReplUnitTable) { - fail: - if (s->EUNtable) - kfree(s->EUNtable); - if (s->ReplUnitTable) - kfree(s->ReplUnitTable); - return -1; - } - - /* mark all blocks as potentially containing data */ - for (i = 0; i < s->nb_blocks; i++) { - s->ReplUnitTable[i] = BLOCK_NOTEXPLORED; - } - /* search for NFTL MediaHeader and Spare NFTL Media Header */ if (find_boot_record(s) < 0) { printk("Could not find valid boot record\n"); - goto fail; + return -1; } - - /* mark the bios blocks (blocks before NFTL MediaHeader) as reserved */ - for (i = 0; i < s->nb_boot_blocks; i++) - s->ReplUnitTable[i] = BLOCK_RESERVED; - - /* also mark the boot records (NFTL MediaHeader) blocks as reserved */ - if (s->MediaUnit != BLOCK_NIL) - s->ReplUnitTable[s->MediaUnit] = BLOCK_RESERVED; - if (s->SpareMediaUnit != BLOCK_NIL) - s->ReplUnitTable[s->SpareMediaUnit] = BLOCK_RESERVED; /* init the logical to physical table */ for (i = 0; i < s->nb_blocks; i++) { diff -urN orig/drivers/net/8390.h linux/drivers/net/8390.h --- orig/drivers/net/8390.h Mon Aug 5 13:31:03 2002 +++ linux/drivers/net/8390.h Wed Sep 11 17:30:15 2002 @@ -117,8 +117,7 @@ #if defined(CONFIG_MAC) || \ defined(CONFIG_ARIADNE2) || defined(CONFIG_ARIADNE2_MODULE) || \ - defined(CONFIG_HYDRA) || defined(CONFIG_HYDRA_MODULE) || \ - defined(CONFIG_ARM_ETHERH) || defined(CONFIG_ARM_ETHERH_MODULE) + defined(CONFIG_HYDRA) || defined(CONFIG_HYDRA_MODULE) #define EI_SHIFT(x) (ei_local->reg_offset[x]) #undef inb #undef inb_p @@ -130,6 +129,8 @@ #define inb_p(port) in_8(port) #define outb_p(val,port) out_8(port,val) +#elif defined(CONFIG_ARM_ETHERH) || defined(CONFIG_ARM_ETHERH_MODULE) +#define EI_SHIFT(x) (ei_local->reg_offset[x]) #else #define EI_SHIFT(x) (x) #endif diff -urN orig/drivers/net/Config.in linux/drivers/net/Config.in --- orig/drivers/net/Config.in Mon Aug 5 13:31:03 2002 +++ linux/drivers/net/Config.in Fri Jan 24 16:13:09 2003 @@ -26,9 +26,13 @@ if [ "$CONFIG_NET_ETHERNET" = "y" ]; then if [ "$CONFIG_ARM" = "y" ]; then dep_bool ' ARM EBSA110 AM79C961A support' CONFIG_ARM_AM79C961A $CONFIG_ARCH_EBSA110 + tristate ' Cirrus Logic CS8900A support' CONFIG_ARM_CIRRUS if [ "$CONFIG_ARCH_ACORN" = "y" ]; then source drivers/acorn/net/Config.in fi + fi + if [ "$CONFIG_ARCH_CAMELOT" = "y" ]; then + tristate ' Altera Ether00 support' CONFIG_ETHER00 fi if [ "$CONFIG_PPC" = "y" ]; then dep_tristate ' MACE (Power Mac ethernet) support' CONFIG_MACE $CONFIG_ALL_PPC diff -urN orig/drivers/net/Makefile linux/drivers/net/Makefile --- orig/drivers/net/Makefile Mon Aug 5 13:31:03 2002 +++ linux/drivers/net/Makefile Sun Sep 15 20:23:41 2002 @@ -209,10 +209,12 @@ obj-$(CONFIG_HYDRA) += hydra.o 8390.o obj-$(CONFIG_ARIADNE) += ariadne.o obj-$(CONFIG_CS89x0) += cs89x0.o +obj-$(CONFIG_ARM_CIRRUS) += cirrus.o obj-$(CONFIG_MACSONIC) += macsonic.o obj-$(CONFIG_MACMACE) += macmace.o obj-$(CONFIG_MAC89x0) += mac89x0.o obj-$(CONFIG_TUN) += tun.o +obj-$(CONFIG_ETHER00) +=ether00.o obj-$(CONFIG_DL2K) += dl2k.o ifeq ($(CONFIG_ARCH_ACORN),y) diff -urN orig/drivers/net/am79c961a.c linux/drivers/net/am79c961a.c --- orig/drivers/net/am79c961a.c Mon Aug 5 13:31:04 2002 +++ linux/drivers/net/am79c961a.c Fri Jan 10 13:23:17 2003 @@ -75,6 +75,17 @@ " : : "r" (val), "r" (reg), "r" (ISAIO_BASE + 0x0464)); } +static inline unsigned short read_ireg(u_long base_addr, u_int reg) +{ + u_short v; + __asm__( + "str%?h %1, [%2] @ NAT_RAP\n\t" + "str%?h %0, [%2, #8] @ NET_IDP\n\t" + : "=r" (v) + : "r" (reg), "r" (ISAIO_BASE + 0x0464)); + return v; +} + #define am_writeword(dev,off,val) __raw_writew(val, ISAMEM_BASE + ((off) << 1)) #define am_readword(dev,off) __raw_readw(ISAMEM_BASE + ((off) << 1)) @@ -254,9 +265,27 @@ write_rreg (dev->base_addr, BASERXH, 0); write_rreg (dev->base_addr, CSR0, CSR0_STOP); write_rreg (dev->base_addr, CSR3, CSR3_IDONM|CSR3_BABLM|CSR3_DXSUFLO); + write_rreg (dev->base_addr, CSR4, CSR4_APAD_XMIT); write_rreg (dev->base_addr, CSR0, CSR0_IENA|CSR0_STRT); } +static void am79c961_timer(unsigned long data) +{ + struct net_device *dev = (struct net_device *)data; + struct dev_priv *priv = (struct dev_priv *)dev->priv; + unsigned int lnkstat, carrier; + + lnkstat = read_ireg(dev->base_addr, ISALED0) & ISALED0_LNKST; + carrier = netif_carrier_ok(dev); + + if (lnkstat && !carrier) + netif_carrier_on(dev); + else if (!lnkstat && carrier) + netif_carrier_off(dev); + + mod_timer(&priv->timer, jiffies + 5*HZ); +} + /* * Open/initialize the board. */ @@ -274,6 +303,11 @@ am79c961_init_for_open(dev); + netif_carrier_off(dev); + + priv->timer.expires = jiffies; + add_timer(&priv->timer); + netif_start_queue(dev); return 0; @@ -288,7 +322,10 @@ struct dev_priv *priv = (struct dev_priv *)dev->priv; unsigned long flags; + del_timer_sync(&priv->timer); + netif_stop_queue(dev); + netif_carrier_off(dev); spin_lock_irqsave(priv->chip_lock, flags); write_rreg (dev->base_addr, CSR0, CSR0_STOP); @@ -409,7 +446,7 @@ am79c961_sendpacket(struct sk_buff *skb, struct net_device *dev) { struct dev_priv *priv = (struct dev_priv *)dev->priv; - unsigned int length = ETH_ZLEN < skb->len ? skb->len : ETH_ZLEN; + unsigned int length = skb->len; unsigned int hdraddr, bufaddr; unsigned int head; unsigned long flags; @@ -511,6 +548,7 @@ am79c961_tx(struct net_device *dev, struct dev_priv *priv) { do { + signed short len; u_int hdraddr; u_int status; @@ -546,6 +584,8 @@ continue; } priv->stats.tx_packets ++; + len = am_readword (dev, hdraddr + 4); + priv->stats.tx_bytes += -len; } while (priv->txtail != priv->txhead); netif_wake_queue(dev); @@ -578,10 +618,10 @@ { struct dev_priv *priv = (struct dev_priv *)dev->priv; - spin_lock_irq(priv->chip_lock); + spin_lock_irq(&priv->chip_lock); write_rreg (dev->base_addr, CSR0, CSR0_STOP); write_rreg (dev->base_addr, CSR3, CSR3_MASKALL); - spin_unlock_irq(priv->chip_lock); + spin_unlock_irq(&priv->chip_lock); am79c961_ramtest(dev, 0x66); am79c961_ramtest(dev, 0x99); @@ -645,6 +685,11 @@ dev->dev_addr[i] = inb(dev->base_addr + i * 2) & 0xff; printk (i == 5 ? "%02x\n" : "%02x:", dev->dev_addr[i]); } + + spin_lock_init(&priv->chip_lock); + init_timer(&priv->timer); + priv->timer.data = (unsigned long)dev; + priv->timer.function = am79c961_timer; if (am79c961_hw_init(dev)) goto release; diff -urN orig/drivers/net/am79c961a.h linux/drivers/net/am79c961a.h --- orig/drivers/net/am79c961a.h Tue Oct 3 20:08:34 2000 +++ linux/drivers/net/am79c961a.h Fri Jan 10 13:02:04 2003 @@ -58,6 +58,10 @@ #define CSR3_BABLM 0x4000 #define CSR3_MASKALL 0x5F00 +#define CSR4 4 +#define CSR4_ASTRP_RCV 0x0400 +#define CSR4_APAD_XMIT 0x0800 + #define CTRL1 5 #define CTRL1_SPND 0x0001 @@ -112,6 +116,9 @@ #define TST_UFLO 0x4000 #define TST_BUFF 0x8000 +#define ISALED0 0x0004 +#define ISALED0_LNKST 0x8000 + struct dev_priv { struct net_device_stats stats; unsigned long rxbuffer[RX_BUFFERS]; @@ -123,6 +130,7 @@ unsigned long rxhdr; unsigned long txhdr; spinlock_t chip_lock; + struct timer_list timer; }; extern int am79c961_probe (struct net_device *dev); diff -urN orig/drivers/net/cirrus.c linux/drivers/net/cirrus.c --- orig/drivers/net/cirrus.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/net/cirrus.c Sat Feb 1 15:40:29 2003 @@ -0,0 +1,692 @@ + +/* + * linux/drivers/net/cirrus.c + * + * Author: Abraham van der Merwe + * + * A Cirrus Logic CS8900A driver for Linux + * based on the cs89x0 driver written by Russell Nelson, + * Donald Becker, and others. + * + * This source code is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * version 2 as published by the Free Software Foundation. + */ + +/* + * At the moment the driver does not support memory mode operation. + * It is trivial to implement this, but not worth the effort. + */ + +/* + * TODO: + * + * 1. If !ready in send_start(), queue buffer and send it in interrupt handler + * when we receive a BufEvent with Rdy4Tx, send it again. dangerous! + * 2. how do we prevent interrupt handler destroying integrity of get_stats()? + * 3. Change reset code to check status. + * 4. Implement set_mac_address and remove fake mac address + * 5. Link status detection stuff + * 6. Write utility to write EEPROM, do self testing, etc. + * 7. Implement DMA routines (I need a board w/ DMA support for that) + * 8. Power management + * 9. Add support for multiple ethernet chips + * 10. Add support for other cs89xx chips (need hardware for that) + */ + +#include +#include +#include + +#include +#include +#include +#include +#include +#include + +#include +#include +#include + +#include +#include +#include + +#include "cirrus.h" + +/* #define DEBUG */ +/* #define FULL_DUPLEX */ + +#ifdef CONFIG_SA1100_FRODO +# define CIRRUS_DEFAULT_IO FRODO_ETH_IO + 0x300 +# define CIRRUS_DEFAULT_IRQ FRODO_ETH_IRQ +#elif CONFIG_SA1100_CERF +# define CIRRUS_DEFAULT_IO CERF_ETH_IO + 0x300 +# define CIRRUS_DEFAULT_IRQ CERF_ETH_IRQ +#elif CONFIG_ARCH_CDB89712 +# define CIRRUS_DEFAULT_IO ETHER_BASE + 0x300 +# define CIRRUS_DEFAULT_IRQ IRQ_EINT3 +#else +# define CIRRUS_DEFAULT_IO 0 +# define CIRRUS_DEFAULT_IRQ 0 +#endif /* #ifdef CONFIG_SA1100_CERF */ + +typedef struct { + struct net_device_stats stats; + u16 txlen; +} cirrus_t; + +typedef struct { + u16 io_base; /* I/O Base Address */ + u16 irq; /* Interrupt Number */ + u16 dma; /* DMA Channel Numbers */ + u32 mem_base; /* Memory Base Address */ + u32 rom_base; /* Boot PROM Base Address */ + u32 rom_mask; /* Boot PROM Address Mask */ + u8 mac[6]; /* Individual Address */ +} cirrus_eeprom_t; + +/* + * I/O routines + */ + +static inline u16 cirrus_read (struct net_device *dev,u16 reg) +{ + outw (reg,dev->base_addr + PP_Address); + return (inw (dev->base_addr + PP_Data)); +} + +static inline void cirrus_write (struct net_device *dev,u16 reg,u16 value) +{ + outw (reg,dev->base_addr + PP_Address); + outw (value,dev->base_addr + PP_Data); +} + +static inline void cirrus_set (struct net_device *dev,u16 reg,u16 value) +{ + cirrus_write (dev,reg,cirrus_read (dev,reg) | value); +} + +static inline void cirrus_clear (struct net_device *dev,u16 reg,u16 value) +{ + cirrus_write (dev,reg,cirrus_read (dev,reg) & ~value); +} + +static inline void cirrus_frame_read (struct net_device *dev,struct sk_buff *skb,u16 length) +{ + insw (dev->base_addr,skb_put (skb,length),(length + 1) / 2); +} + +static inline void cirrus_frame_write (struct net_device *dev,struct sk_buff *skb) +{ + outsw (dev->base_addr,skb->data,(skb->len + 1) / 2); +} + +/* + * Debugging functions + */ + +#ifdef DEBUG +static inline int printable (int c) +{ + return ((c >= 32 && c <= 126) || + (c >= 174 && c <= 223) || + (c >= 242 && c <= 243) || + (c >= 252 && c <= 253)); +} + +static void dump16 (struct net_device *dev,const u8 *s,size_t len) +{ + int i; + char str[128]; + + if (!len) return; + + *str = '\0'; + + for (i = 0; i < len; i++) { + if (i && !(i % 4)) strcat (str," "); + sprintf (str,"%s%.2x ",str,s[i]); + } + + for ( ; i < 16; i++) { + if (i && !(i % 4)) strcat (str," "); + strcat (str," "); + } + + strcat (str," "); + for (i = 0; i < len; i++) sprintf (str,"%s%c",str,printable (s[i]) ? s[i] : '.'); + + printk (KERN_DEBUG "%s: %s\n",dev->name,str); +} + +static void hexdump (struct net_device *dev,const void *ptr,size_t size) +{ + const u8 *s = (u8 *) ptr; + int i; + for (i = 0; i < size / 16; i++, s += 16) dump16 (dev,s,16); + dump16 (dev,s,size % 16); +} + +static void dump_packet (struct net_device *dev,struct sk_buff *skb,const char *type) +{ + printk (KERN_INFO "%s: %s %d byte frame %.2x:%.2x:%.2x:%.2x:%.2x:%.2x to %.2x:%.2x:%.2x:%.2x:%.2x:%.2x type %.4x\n", + dev->name, + type, + skb->len, + skb->data[0],skb->data[1],skb->data[2],skb->data[3],skb->data[4],skb->data[5], + skb->data[6],skb->data[7],skb->data[8],skb->data[9],skb->data[10],skb->data[11], + (skb->data[12] << 8) | skb->data[13]); + if (skb->len < 0x100) hexdump (dev,skb->data,skb->len); +} +#endif /* #ifdef DEBUG */ + +/* + * Driver functions + */ + +static void cirrus_receive (struct net_device *dev) +{ + cirrus_t *priv = (cirrus_t *) dev->priv; + struct sk_buff *skb; + u16 status,length; + + status = cirrus_read (dev,PP_RxStatus); + length = cirrus_read (dev,PP_RxLength); + + if (!(status & RxOK)) { + priv->stats.rx_errors++; + if ((status & (Runt | Extradata))) priv->stats.rx_length_errors++; + if ((status & CRCerror)) priv->stats.rx_crc_errors++; + return; + } + + if ((skb = dev_alloc_skb (length + 4)) == NULL) { + priv->stats.rx_dropped++; + return; + } + + skb->dev = dev; + skb_reserve (skb,2); + + cirrus_frame_read (dev,skb,length); + +#ifdef DEBUG + dump_packet (dev,skb,"recv"); +#endif /* #ifdef DEBUG */ + + skb->protocol = eth_type_trans (skb,dev); + + netif_rx (skb); + dev->last_rx = jiffies; + + priv->stats.rx_packets++; + priv->stats.rx_bytes += length; +} + +static int cirrus_send_start (struct sk_buff *skb,struct net_device *dev) +{ + cirrus_t *priv = (cirrus_t *) dev->priv; + u16 status; + + netif_stop_queue (dev); + + cirrus_write (dev,PP_TxCMD,TxStart (After5)); + cirrus_write (dev,PP_TxLength,skb->len); + + status = cirrus_read (dev,PP_BusST); + + if ((status & TxBidErr)) { + printk (KERN_WARNING "%s: Invalid frame size %d!\n",dev->name,skb->len); + priv->stats.tx_errors++; + priv->stats.tx_aborted_errors++; + priv->txlen = 0; + return (1); + } + + if (!(status & Rdy4TxNOW)) { + printk (KERN_WARNING "%s: Transmit buffer not free!\n",dev->name); + priv->stats.tx_errors++; + priv->txlen = 0; + /* FIXME: store skb and send it in interrupt handler */ + return (1); + } + + cirrus_frame_write (dev,skb); + +#ifdef DEBUG + dump_packet (dev,skb,"send"); +#endif /* #ifdef DEBUG */ + + dev->trans_start = jiffies; + + dev_kfree_skb (skb); + + priv->txlen = skb->len; + + return (0); +} + +static void cirrus_interrupt (int irq,void *id,struct pt_regs *regs) +{ + struct net_device *dev = (struct net_device *) id; + cirrus_t *priv; + u16 status; + + if (dev->priv == NULL) { + printk (KERN_WARNING "%s: irq %d for unknown device.\n",dev->name,irq); + return; + } + + priv = (cirrus_t *) dev->priv; + + while ((status = cirrus_read (dev,PP_ISQ))) { + switch (RegNum (status)) { + case RxEvent: + cirrus_receive (dev); + break; + + case TxEvent: + priv->stats.collisions += ColCount (cirrus_read (dev,PP_TxCOL)); + if (!(RegContent (status) & TxOK)) { + priv->stats.tx_errors++; + if ((RegContent (status) & Out_of_window)) priv->stats.tx_window_errors++; + if ((RegContent (status) & Jabber)) priv->stats.tx_aborted_errors++; + break; + } else if (priv->txlen) { + priv->stats.tx_packets++; + priv->stats.tx_bytes += priv->txlen; + } + priv->txlen = 0; + netif_wake_queue (dev); + break; + + case BufEvent: + if ((RegContent (status) & RxMiss)) { + u16 missed = MissCount (cirrus_read (dev,PP_RxMISS)); + priv->stats.rx_errors += missed; + priv->stats.rx_missed_errors += missed; + } + if ((RegContent (status) & TxUnderrun)) { + priv->stats.tx_errors++; + priv->stats.tx_fifo_errors++; + } + /* FIXME: if Rdy4Tx, transmit last sent packet (if any) */ + priv->txlen = 0; + netif_wake_queue (dev); + break; + + case TxCOL: + priv->stats.collisions += ColCount (cirrus_read (dev,PP_TxCOL)); + break; + + case RxMISS: + status = MissCount (cirrus_read (dev,PP_RxMISS)); + priv->stats.rx_errors += status; + priv->stats.rx_missed_errors += status; + break; + } + } +} + +static void cirrus_transmit_timeout (struct net_device *dev) +{ + cirrus_t *priv = (cirrus_t *) dev->priv; + priv->stats.tx_errors++; + priv->stats.tx_heartbeat_errors++; + priv->txlen = 0; + netif_wake_queue (dev); +} + +static int cirrus_start (struct net_device *dev) +{ + int result; + + /* valid ethernet address? */ + if (!is_valid_ether_addr(dev->dev_addr)) { + printk(KERN_ERR "%s: invalid ethernet MAC address\n",dev->name); + return (-EINVAL); + } + + /* install interrupt handler */ + if ((result = request_irq (dev->irq,&cirrus_interrupt,0,dev->name,dev)) < 0) { + printk (KERN_ERR "%s: could not register interrupt %d\n",dev->name,dev->irq); + return (result); + } + + /* enable the ethernet controller */ + cirrus_set (dev,PP_RxCFG,RxOKiE | BufferCRC | CRCerroriE | RuntiE | ExtradataiE); + cirrus_set (dev,PP_RxCTL,RxOKA | IndividualA | BroadcastA); + cirrus_set (dev,PP_TxCFG,TxOKiE | Out_of_windowiE | JabberiE); + cirrus_set (dev,PP_BufCFG,Rdy4TxiE | RxMissiE | TxUnderruniE | TxColOvfiE | MissOvfloiE); + cirrus_set (dev,PP_LineCTL,SerRxON | SerTxON); + cirrus_set (dev,PP_BusCTL,EnableRQ); + +#ifdef FULL_DUPLEX + cirrus_set (dev,PP_TestCTL,FDX); +#endif /* #ifdef FULL_DUPLEX */ + + /* start the queue */ + netif_start_queue (dev); + + MOD_INC_USE_COUNT; + + return (0); +} + +static int cirrus_stop (struct net_device *dev) +{ + /* disable ethernet controller */ + cirrus_write (dev,PP_BusCTL,0); + cirrus_write (dev,PP_TestCTL,0); + cirrus_write (dev,PP_SelfCTL,0); + cirrus_write (dev,PP_LineCTL,0); + cirrus_write (dev,PP_BufCFG,0); + cirrus_write (dev,PP_TxCFG,0); + cirrus_write (dev,PP_RxCTL,0); + cirrus_write (dev,PP_RxCFG,0); + + /* uninstall interrupt handler */ + free_irq (dev->irq,dev); + + /* stop the queue */ + netif_stop_queue (dev); + + MOD_DEC_USE_COUNT; + + return (0); +} + +static int cirrus_set_mac_address (struct net_device *dev, void *p) +{ + struct sockaddr *addr = (struct sockaddr *)p; + int i; + + if (netif_running(dev)) + return -EBUSY; + + memcpy(dev->dev_addr, addr->sa_data, dev->addr_len); + + /* configure MAC address */ + for (i = 0; i < ETH_ALEN; i += 2) + cirrus_write (dev,PP_IA + i,dev->dev_addr[i] | (dev->dev_addr[i + 1] << 8)); + + return 0; +} + +static struct net_device_stats *cirrus_get_stats (struct net_device *dev) +{ + cirrus_t *priv = (cirrus_t *) dev->priv; + return (&priv->stats); +} + +static void cirrus_set_receive_mode (struct net_device *dev) +{ + if ((dev->flags & IFF_PROMISC)) + cirrus_set (dev,PP_RxCTL,PromiscuousA); + else + cirrus_clear (dev,PP_RxCTL,PromiscuousA); + + if ((dev->flags & IFF_ALLMULTI) && dev->mc_list) + cirrus_set (dev,PP_RxCTL,MulticastA); + else + cirrus_clear (dev,PP_RxCTL,MulticastA); +} + +static int cirrus_eeprom_wait (struct net_device *dev) +{ + int i; + + for (i = 0; i < 200; i++) { + if (!(cirrus_read (dev,PP_SelfST) & SIBUSY)) + return (0); + udelay (1); + } + + return (-1); +} + +static int cirrus_eeprom_read (struct net_device *dev,u16 *value,u16 offset) +{ + if (cirrus_eeprom_wait (dev) < 0) + return (-1); + + cirrus_write (dev,PP_EEPROMCommand,offset | EEReadRegister); + + if (cirrus_eeprom_wait (dev) < 0) + return (-1); + + *value = cirrus_read (dev,PP_EEPROMData); + + return (0); +} + +static int cirrus_eeprom (struct net_device *dev,cirrus_eeprom_t *eeprom) +{ + u16 offset,buf[16],*word; + u8 checksum = 0,*byte; + + if (cirrus_eeprom_read (dev,buf,0) < 0) { + read_timed_out: + printk (KERN_DEBUG "%s: EEPROM read timed out\n",dev->name); + return (-ETIMEDOUT); + } + + if ((buf[0] >> 8) != 0xa1) { + printk (KERN_DEBUG "%s: No EEPROM present\n",dev->name); + return (-ENODEV); + } + + if ((buf[0] & 0xff) < sizeof (buf)) { + eeprom_too_small: + printk (KERN_DEBUG "%s: EEPROM too small\n",dev->name); + return (-ENODEV); + } + + for (offset = 1; offset < (buf[0] & 0xff); offset++) { + if (cirrus_eeprom_read (dev,buf + offset,offset) < 0) + goto read_timed_out; + + if (buf[offset] == 0xffff) + goto eeprom_too_small; + } + + if (buf[1] != 0x2020) { + printk (KERN_DEBUG "%s: Group Header #1 mismatch\n",dev->name); + return (-EIO); + } + + if (buf[5] != 0x502c) { + printk (KERN_DEBUG "%s: Group Header #2 mismatch\n",dev->name); + return (-EIO); + } + + if (buf[12] != 0x2158) { + printk (KERN_DEBUG "%s: Group Header #3 mismatch\n",dev->name); + return (-EIO); + } + + eeprom->io_base = buf[2]; + eeprom->irq = buf[3]; + eeprom->dma = buf[4]; + eeprom->mem_base = (buf[7] << 16) | buf[6]; + eeprom->rom_base = (buf[9] << 16) | buf[8]; + eeprom->rom_mask = (buf[11] << 16) | buf[10]; + + word = (u16 *) eeprom->mac; + for (offset = 0; offset < 3; offset++) word[offset] = buf[13 + offset]; + + byte = (u8 *) buf; + for (offset = 0; offset < sizeof (buf); offset++) checksum += byte[offset]; + + if (cirrus_eeprom_read (dev,&offset,0x10) < 0) + goto read_timed_out; + + if ((offset >> 8) != (u8) (0x100 - checksum)) { + printk (KERN_DEBUG "%s: Checksum mismatch (expected 0x%.2x, got 0x%.2x instead\n", + dev->name, + (u8) (0x100 - checksum), + offset >> 8); + return (-EIO); + } + + return (0); +} + +/* + * Architecture dependant code + */ + +#ifdef CONFIG_SA1100_FRODO +static void frodo_reset (struct net_device *dev) +{ + int i; + volatile u16 value; + + /* reset ethernet controller */ + FRODO_CPLD_ETHERNET |= FRODO_ETH_RESET; + mdelay (50); + FRODO_CPLD_ETHERNET &= ~FRODO_ETH_RESET; + mdelay (50); + + /* we tied SBHE to CHIPSEL, so each memory access ensure the chip is in 16-bit mode */ + for (i = 0; i < 3; i++) value = cirrus_read (dev,0); + + /* FIXME: poll status bit */ +} +#endif /* #ifdef CONFIG_SA1100_FRODO */ + +/* + * Driver initialization routines + */ + +static int io = 0; +static int irq = 0; + +int __init cirrus_probe (struct net_device *dev) +{ + static cirrus_t priv; + int i,result; + u16 value; + cirrus_eeprom_t eeprom; + + printk ("Cirrus Logic CS8900A driver for Linux (V0.02)\n"); + + memset (&priv,0,sizeof (cirrus_t)); + + ether_setup (dev); + + dev->open = cirrus_start; + dev->stop = cirrus_stop; + dev->hard_start_xmit = cirrus_send_start; + dev->get_stats = cirrus_get_stats; + dev->set_multicast_list = cirrus_set_receive_mode; + dev->set_mac_address = cirrus_set_mac_address; + dev->tx_timeout = cirrus_transmit_timeout; + dev->watchdog_timeo = HZ; + + dev->dev_addr[0] = 0x00; + dev->dev_addr[1] = 0x00; + dev->dev_addr[2] = 0x00; + dev->dev_addr[3] = 0x00; + dev->dev_addr[4] = 0x00; + dev->dev_addr[5] = 0x00; + + dev->if_port = IF_PORT_10BASET; + dev->priv = (void *) &priv; + + SET_MODULE_OWNER (dev); + + dev->base_addr = CIRRUS_DEFAULT_IO; + dev->irq = CIRRUS_DEFAULT_IRQ; + + /* module parameters override everything */ + if (io > 0) dev->base_addr = io; + if (irq > 0) dev->irq = irq; + + if (!dev->base_addr) { + printk (KERN_ERR + "%s: No default I/O base address defined. Use io=... or\n" + "%s: define CIRRUS_DEFAULT_IO for your platform\n", + dev->name,dev->name); + return (-EINVAL); + } + + if (!dev->irq) { + printk (KERN_ERR + "%s: No default IRQ number defined. Use irq=... or\n" + "%s: define CIRRUS_DEFAULT_IRQ for your platform\n", + dev->name,dev->name); + return (-EINVAL); + } + + if ((result = check_region (dev->base_addr,16))) { + printk (KERN_ERR "%s: can't get I/O port address 0x%lx\n",dev->name,dev->base_addr); + return (result); + } + + if (!request_region (dev->base_addr,16,dev->name)) + return -EBUSY; + +#ifdef CONFIG_SA1100_FRODO + frodo_reset (dev); +#endif /* #ifdef CONFIG_SA1100_FRODO */ + + /* if an EEPROM is present, use it's MAC address */ + if (!cirrus_eeprom (dev,&eeprom)) + for (i = 0; i < 6; i++) + dev->dev_addr[i] = eeprom.mac[i]; + + /* verify EISA registration number for Cirrus Logic */ + if ((value = cirrus_read (dev,PP_ProductID)) != EISA_REG_CODE) { + printk (KERN_ERR "%s: incorrect signature 0x%.4x\n",dev->name,value); + return (-ENXIO); + } + + /* verify chip version */ + value = cirrus_read (dev,PP_ProductID + 2); + if (VERSION (value) != CS8900A) { + printk (KERN_ERR "%s: unknown chip version 0x%.8x\n",dev->name,VERSION (value)); + return (-ENXIO); + } + printk (KERN_INFO "%s: CS8900A rev %c detected\n",dev->name,'B' + REVISION (value) - REV_B); + + /* setup interrupt number */ + cirrus_write (dev,PP_IntNum,0); + + /* configure MAC address */ + for (i = 0; i < ETH_ALEN; i += 2) + cirrus_write (dev,PP_IA + i,dev->dev_addr[i] | (dev->dev_addr[i + 1] << 8)); + + return (0); +} + +EXPORT_NO_SYMBOLS; + +static struct net_device dev; + +static int __init cirrus_init (void) +{ + memset (&dev,0,sizeof (struct net_device)); + dev.init = cirrus_probe; + return (register_netdev (&dev)); +} + +static void __exit cirrus_cleanup (void) +{ + release_region (dev.base_addr,16); + unregister_netdev (&dev); +} + +MODULE_AUTHOR ("Abraham van der Merwe "); +MODULE_DESCRIPTION ("Cirrus Logic CS8900A driver for Linux (V0.02)"); +MODULE_LICENSE ("GPL"); +MODULE_PARM_DESC (io,"I/O Base Address"); +MODULE_PARM (io,"i"); +MODULE_PARM_DESC (irq,"IRQ Number"); +MODULE_PARM (irq,"i"); + +module_init (cirrus_init); +module_exit (cirrus_cleanup); + diff -urN orig/drivers/net/cirrus.h linux/drivers/net/cirrus.h --- orig/drivers/net/cirrus.h Thu Jan 1 01:00:00 1970 +++ linux/drivers/net/cirrus.h Sun Sep 15 20:23:09 2002 @@ -0,0 +1,235 @@ +#ifndef CIRRUS_H +#define CIRRUS_H + +/* + * linux/drivers/net/cirrus.h + * + * Author: Abraham van der Merwe + * + * A Cirrus Logic CS8900A driver for Linux + * based on the cs89x0 driver written by Russell Nelson, + * Donald Becker, and others. + * + * This source code is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * version 2 as published by the Free Software Foundation. + */ + +/* + * Ports + */ + +#define PP_Address 0x0a /* PacketPage Pointer Port (Section 4.10.10) */ +#define PP_Data 0x0c /* PacketPage Data Port (Section 4.10.10) */ + +/* + * Registers + */ + +#define PP_ProductID 0x0000 /* Section 4.3.1 Product Identification Code */ +#define PP_MemBase 0x002c /* Section 4.9.2 Memory Base Address Register */ +#define PP_IntNum 0x0022 /* Section 3.2.3 Interrupt Number */ +#define PP_EEPROMCommand 0x0040 /* Section 4.3.11 EEPROM Command */ +#define PP_EEPROMData 0x0042 /* Section 4.3.12 EEPROM Data */ +#define PP_RxCFG 0x0102 /* Section 4.4.6 Receiver Configuration */ +#define PP_RxCTL 0x0104 /* Section 4.4.8 Receiver Control */ +#define PP_TxCFG 0x0106 /* Section 4.4.9 Transmit Configuration */ +#define PP_BufCFG 0x010a /* Section 4.4.12 Buffer Configuration */ +#define PP_LineCTL 0x0112 /* Section 4.4.16 Line Control */ +#define PP_SelfCTL 0x0114 /* Section 4.4.18 Self Control */ +#define PP_BusCTL 0x0116 /* Section 4.4.20 Bus Control */ +#define PP_TestCTL 0x0118 /* Section 4.4.22 Test Control */ +#define PP_ISQ 0x0120 /* Section 4.4.5 Interrupt Status Queue */ +#define PP_TxEvent 0x0128 /* Section 4.4.10 Transmitter Event */ +#define PP_BufEvent 0x012c /* Section 4.4.13 Buffer Event */ +#define PP_RxMISS 0x0130 /* Section 4.4.14 Receiver Miss Counter */ +#define PP_TxCOL 0x0132 /* Section 4.4.15 Transmit Collision Counter */ +#define PP_SelfST 0x0136 /* Section 4.4.19 Self Status */ +#define PP_BusST 0x0138 /* Section 4.4.21 Bus Status */ +#define PP_TxCMD 0x0144 /* Section 4.4.11 Transmit Command */ +#define PP_TxLength 0x0146 /* Section 4.5.2 Transmit Length */ +#define PP_IA 0x0158 /* Section 4.6.2 Individual Address (IEEE Address) */ +#define PP_RxStatus 0x0400 /* Section 4.7.1 Receive Status */ +#define PP_RxLength 0x0402 /* Section 4.7.1 Receive Length (in bytes) */ +#define PP_RxFrame 0x0404 /* Section 4.7.2 Receive Frame Location */ +#define PP_TxFrame 0x0a00 /* Section 4.7.2 Transmit Frame Location */ + +/* + * Values + */ + +/* PP_IntNum */ +#define INTRQ0 0x0000 +#define INTRQ1 0x0001 +#define INTRQ2 0x0002 +#define INTRQ3 0x0003 + +/* PP_ProductID */ +#define EISA_REG_CODE 0x630e +#define REVISION(x) (((x) & 0x1f00) >> 8) +#define VERSION(x) ((x) & ~0x1f00) + +#define CS8900A 0x0000 +#define REV_B 7 +#define REV_C 8 +#define REV_D 9 + +/* PP_RxCFG */ +#define Skip_1 0x0040 +#define StreamE 0x0080 +#define RxOKiE 0x0100 +#define RxDMAonly 0x0200 +#define AutoRxDMAE 0x0400 +#define BufferCRC 0x0800 +#define CRCerroriE 0x1000 +#define RuntiE 0x2000 +#define ExtradataiE 0x4000 + +/* PP_RxCTL */ +#define IAHashA 0x0040 +#define PromiscuousA 0x0080 +#define RxOKA 0x0100 +#define MulticastA 0x0200 +#define IndividualA 0x0400 +#define BroadcastA 0x0800 +#define CRCerrorA 0x1000 +#define RuntA 0x2000 +#define ExtradataA 0x4000 + +/* PP_TxCFG */ +#define Loss_of_CRSiE 0x0040 +#define SQErroriE 0x0080 +#define TxOKiE 0x0100 +#define Out_of_windowiE 0x0200 +#define JabberiE 0x0400 +#define AnycolliE 0x0800 +#define T16colliE 0x8000 + +/* PP_BufCFG */ +#define SWint_X 0x0040 +#define RxDMAiE 0x0080 +#define Rdy4TxiE 0x0100 +#define TxUnderruniE 0x0200 +#define RxMissiE 0x0400 +#define Rx128iE 0x0800 +#define TxColOvfiE 0x1000 +#define MissOvfloiE 0x2000 +#define RxDestiE 0x8000 + +/* PP_LineCTL */ +#define SerRxON 0x0040 +#define SerTxON 0x0080 +#define AUIonly 0x0100 +#define AutoAUI_10BT 0x0200 +#define ModBackoffE 0x0800 +#define PolarityDis 0x1000 +#define L2_partDefDis 0x2000 +#define LoRxSquelch 0x4000 + +/* PP_SelfCTL */ +#define RESET 0x0040 +#define SWSuspend 0x0100 +#define HWSleepE 0x0200 +#define HWStandbyE 0x0400 +#define HC0E 0x1000 +#define HC1E 0x2000 +#define HCB0 0x4000 +#define HCB1 0x8000 + +/* PP_BusCTL */ +#define ResetRxDMA 0x0040 +#define DMAextend 0x0100 +#define UseSA 0x0200 +#define MemoryE 0x0400 +#define DMABurst 0x0800 +#define IOCHRDYE 0x1000 +#define RxDMAsize 0x2000 +#define EnableRQ 0x8000 + +/* PP_TestCTL */ +#define DisableLT 0x0080 +#define ENDECloop 0x0200 +#define AUIloop 0x0400 +#define DisableBackoff 0x0800 +#define FDX 0x4000 + +/* PP_ISQ */ +#define RegNum(x) ((x) & 0x3f) +#define RegContent(x) ((x) & ~0x3d) + +#define RxEvent 0x0004 +#define TxEvent 0x0008 +#define BufEvent 0x000c +#define RxMISS 0x0010 +#define TxCOL 0x0012 + +/* PP_RxStatus */ +#define IAHash 0x0040 +#define Dribblebits 0x0080 +#define RxOK 0x0100 +#define Hashed 0x0200 +#define IndividualAdr 0x0400 +#define Broadcast 0x0800 +#define CRCerror 0x1000 +#define Runt 0x2000 +#define Extradata 0x4000 + +#define HashTableIndex(x) ((x) >> 0xa) + +/* PP_TxCMD */ +#define After5 0 +#define After381 1 +#define After1021 2 +#define AfterAll 3 +#define TxStart(x) ((x) << 6) + +#define Force 0x0100 +#define Onecoll 0x0200 +#define InhibitCRC 0x1000 +#define TxPadDis 0x2000 + +/* PP_BusST */ +#define TxBidErr 0x0080 +#define Rdy4TxNOW 0x0100 + +/* PP_TxEvent */ +#define Loss_of_CRS 0x0040 +#define SQEerror 0x0080 +#define TxOK 0x0100 +#define Out_of_window 0x0200 +#define Jabber 0x0400 +#define T16coll 0x8000 + +#define TX_collisions(x) (((x) >> 0xb) & ~0x8000) + +/* PP_BufEvent */ +#define SWint 0x0040 +#define RxDMAFrame 0x0080 +#define Rdy4Tx 0x0100 +#define TxUnderrun 0x0200 +#define RxMiss 0x0400 +#define Rx128 0x0800 +#define RxDest 0x8000 + +/* PP_RxMISS */ +#define MissCount(x) ((x) >> 6) + +/* PP_TxCOL */ +#define ColCount(x) ((x) >> 6) + +/* PP_SelfST */ +#define T3VActive 0x0040 +#define INITD 0x0080 +#define SIBUSY 0x0100 +#define EEPROMpresent 0x0200 +#define EEPROMOK 0x0400 +#define ELpresent 0x0800 +#define EEsize 0x1000 + +/* PP_EEPROMCommand */ +#define EEWriteRegister 0x0100 +#define EEReadRegister 0x0200 +#define EEEraseRegister 0x0300 +#define ELSEL 0x0400 + +#endif /* #ifndef CIRRUS_H */ diff -urN orig/drivers/net/cs89x0.c linux/drivers/net/cs89x0.c --- orig/drivers/net/cs89x0.c Mon Aug 5 13:31:04 2002 +++ linux/drivers/net/cs89x0.c Mon Aug 5 14:32:18 2002 @@ -115,6 +115,7 @@ */ +#include #include #include #include @@ -427,18 +428,18 @@ /* if they give us an odd I/O address, then do ONE write to the address port, to get it back to address zero, where we expect to find the EISA signature word. An IO with a base of 0x3 - will skip the test for the ADD_PORT. */ + will skip the test for the ADD_PORT. */ if (ioaddr & 1) { if (net_debug > 1) printk(KERN_INFO "%s: odd ioaddr 0x%x\n", dev->name, ioaddr); - if ((ioaddr & 2) != 2) + if ((ioaddr & 2) != 2) if ((inw((ioaddr & ~3)+ ADD_PORT) & ADD_MASK) != ADD_SIG) { printk(KERN_ERR "%s: bad signature 0x%x\n", dev->name, inw((ioaddr & ~3)+ ADD_PORT)); retval = -ENODEV; goto out2; } - ioaddr &= ~3; + ioaddr &= ~3; outw(PP_ChipID, ioaddr + ADD_PORT); } printk("PP_addr=0x%x\n", inw(ioaddr + ADD_PORT)); @@ -446,7 +447,7 @@ if (inw(ioaddr + DATA_PORT) != CHIP_EISA_ID_SIG) { printk(KERN_ERR "%s: incorrect signature 0x%x\n", dev->name, inw(ioaddr + DATA_PORT)); - retval = -ENODEV; + retval = -ENODEV; goto out2; } @@ -477,7 +478,7 @@ dev->base_addr); reset_chip(dev); - + /* Here we read the current configuration of the chip. If there is no Extended EEPROM then the idea is to not disturb the chip configuration, it should have been correctly setup by automatic diff -urN orig/drivers/net/ether00.c linux/drivers/net/ether00.c --- orig/drivers/net/ether00.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/net/ether00.c Fri Feb 21 15:17:33 2003 @@ -0,0 +1,996 @@ +/* + * drivers/net/ether00.c + * + * Copyright (C) 2001 Altera Corporation + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ + +/* includes */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include + +static int ether00_get_ethernet_address(struct net_device* dev); + +MODULE_AUTHOR("Clive Davies"); +MODULE_DESCRIPTION("Altera Ether00 IP core driver"); +MODULE_LICENSE("GPL"); + +static long base=0x60000000; +static int irq=0x1; +static int phy_irq=0x2; +MODULE_PARM(base,"l"); +MODULE_PARM(irq,"i"); +MODULE_PARM(phy_irq,"i"); + +#define TX_TIMEOUT (400*HZ/1000) +#define PKT_BUF_SZ 1540 /* Size of each rx buffer */ + + +#undef DEBUG +#define DEBUG(x) + +#define __dma_va(x) (unsigned int)((unsigned int)priv->dma_data+(((unsigned int)(x))&(EXC_SPSRAM_BLOCK0_SIZE-1))) +#define __dma_pa(x) (unsigned int)(EXC_SPSRAM_BLOCK0_BASE+(((unsigned int)(x))-(unsigned int)priv->dma_data)) + +#define ETHER00_BASE 0 +#define ETHER00_TYPE +#define ETHER00_NAME "ether00" +#define MAC_REG_SIZE 0x400 /* size of MAC register area */ + + + +/* typedefs */ + +/* The definition of the driver control structure */ + +#define RX_NUM_BUFF 10 +#define RX_NUM_FDESC 10 +#define TX_NUM_FDESC 10 + +struct tx_fda_ent{ + FDA_DESC fd; + BUF_DESC bd; + BUF_DESC pad; +}; +struct rx_fda_ent{ + FDA_DESC fd; + BUF_DESC bd; + BUF_DESC pad; +}; +struct rx_blist_ent{ + FDA_DESC fd; + BUF_DESC bd; + BUF_DESC pad; +}; +struct net_priv +{ + struct net_device_stats stats; + struct sk_buff* skb; + void* dma_data; + struct rx_blist_ent* rx_blist_vp; + struct rx_fda_ent* rx_fda_ptr; + struct tx_fda_ent* tx_fdalist_vp; + struct tq_struct tq_memupdate; + unsigned char rx_disabled; + unsigned char memupdate_scheduled; + unsigned char queue_stopped; + spinlock_t dma_lock; + unsigned int tx_head; + unsigned int tx_tail; +}; + +static const char vendor_id[2]={0x07,0xed}; + +#ifdef ETHER00_DEBUG + +/* Dump (most) registers for debugging puposes */ + +static void dump_regs(struct net_device *dev){ + struct net_priv* priv=dev->priv; + unsigned int* i; + + printk("\n RX free descriptor area:\n"); + + for(i=(unsigned int*)priv->rx_fda_ptr; + i<((unsigned int*)(priv->rx_fda_ptr+RX_NUM_FDESC));){ + printk("%#8x %#8x %#8x %#8x\n",*i,*(i+1),*(i+2),*(i+3)); + i+=4; + } + + printk("\n RX buffer list:\n"); + + for(i=(unsigned int*)priv->rx_blist_vp; + i<((unsigned int*)(priv->rx_blist_vp+RX_NUM_BUFF));){ + printk("%#8x %#8x %#8x %#8x\n",*i,*(i+1),*(i+2),*(i+3)); + i+=4; + } + + printk("\n TX frame descriptor list:\n"); + + for(i=(unsigned int*)priv->tx_fdalist_vp; + i<((unsigned int*)(priv->tx_fdalist_vp+TX_NUM_FDESC));){ + printk("%#8x %#8x %#8x %#8x\n",*i,*(i+1),*(i+2),*(i+3)); + i+=4; + } + + printk("\ndma ctl=%#x\n",readw(ETHER_DMA_CTL(dev->base_addr))); + printk("txfrmptr=%#x\n",readw(ETHER_TXFRMPTR(dev->base_addr))); + printk("txthrsh=%#x\n",readw(ETHER_TXTHRSH(dev->base_addr))); + printk("txpollctr=%#x\n",readw(ETHER_TXPOLLCTR(dev->base_addr))); + printk("blfrmptr=%#x\n",readw(ETHER_BLFRMPTR(dev->base_addr))); + printk("rxfragsize=%#x\n",readw(ETHER_RXFRAGSIZE(dev->base_addr))); + printk("tx_int_en=%#x\n",readw(ETHER_INT_EN(dev->base_addr))); + printk("fda_bas=%#x\n",readw(ETHER_FDA_BAS(dev->base_addr))); + printk("fda_lim=%#x\n",readw(ETHER_FDA_LIM(dev->base_addr))); + printk("int_src=%#x\n",readw(ETHER_INT_SRC(dev->base_addr))); + printk("pausecnt=%#x\n",readw(ETHER_PAUSECNT(dev->base_addr))); + printk("rempaucnt=%#x\n",readw(ETHER_REMPAUCNT(dev->base_addr))); + printk("txconfrmstat=%#x\n",readw(ETHER_TXCONFRMSTAT(dev->base_addr))); + printk("mac_ctl=%#x\n",readw(ETHER_MAC_CTL(dev->base_addr))); + printk("arc_ctl=%#x\n",readw(ETHER_ARC_CTL(dev->base_addr))); + printk("tx_ctl=%#x\n",readw(ETHER_TX_CTL(dev->base_addr))); +} +#endif /* ETHER00_DEBUG */ + + +static int ether00_write_phy(struct net_device *dev, short address, short value) +{ + volatile int count = 1024; + writew(value,ETHER_MD_DATA(dev->base_addr)); + writew( ETHER_MD_CA_BUSY_MSK | + ETHER_MD_CA_WR_MSK | + (address & ETHER_MD_CA_ADDR_MSK), + ETHER_MD_CA(dev->base_addr)); + + /* Wait for the command to complete */ + while((readw(ETHER_MD_CA(dev->base_addr)) & ETHER_MD_CA_BUSY_MSK)&&count){ + count--; + } + if (!count){ + printk("Write to phy failed, addr=%#x, data=%#x\n",address, value); + return -EIO; + } + return 0; +} + +static int ether00_read_phy(struct net_device *dev, short address) +{ + volatile int count = 1024; + writew( ETHER_MD_CA_BUSY_MSK | + (address & ETHER_MD_CA_ADDR_MSK), + ETHER_MD_CA(dev->base_addr)); + + /* Wait for the command to complete */ + while((readw(ETHER_MD_CA(dev->base_addr)) & ETHER_MD_CA_BUSY_MSK)&&count){ + count--; + } + if (!count){ + printk(KERN_WARNING "Read from phy timed out\n"); + return -EIO; + } + return readw(ETHER_MD_DATA(dev->base_addr)); +} + +static void ether00_phy_int(int irq_num, void* dev_id, struct pt_regs* regs) +{ + struct net_device* dev=dev_id; + int irq_status; + + irq_status=ether00_read_phy(dev, PHY_IRQ_CONTROL); + + if(irq_status & PHY_IRQ_CONTROL_ANEG_COMP_INT_MSK){ + /* + * Autonegotiation complete on epxa10db. The mac doesn't + * twig if we're in full duplex so we need to check the + * phy diagnostic register and configure the mac accordingly + */ + if(ether00_read_phy(dev, PHY_DIAGNOSTIC)&PHY_DIAGNOSTIC_DPLX_MSK){ + int tmp; + tmp=readl(ETHER_MAC_CTL(dev->base_addr)); + writel(tmp|ETHER_MAC_CTL_FULLDUP_MSK, + ETHER_MAC_CTL(dev->base_addr)); + } + else + { + int tmp; + tmp=readl(ETHER_MAC_CTL(dev->base_addr)); + writel(tmp&(~ETHER_MAC_CTL_FULLDUP_MSK), + ETHER_MAC_CTL(dev->base_addr)); + + } + } + + if(irq_status&PHY_IRQ_CONTROL_LS_CHG_INT_MSK){ + + if(ether00_read_phy(dev, PHY_STATUS)& PHY_STATUS_LINK_MSK){ + /* Link is up */ + netif_carrier_on(dev); + }else{ + netif_carrier_off(dev); + } + } + +} + +static void setup_blist_entry(struct sk_buff* skb,struct rx_blist_ent* blist_ent_ptr){ + /* Make the buffer consistent with the cache as the mac is going to write + * directly into it*/ + blist_ent_ptr->fd.FDSystem=(unsigned int)skb; + blist_ent_ptr->bd.BuffData=(char*)__pa(skb->data); + consistent_sync(skb->data,PKT_BUF_SZ,PCI_DMA_FROMDEVICE); + /* align IP on 16 Byte (DMA_CTL set to skip 2 bytes) */ + skb_reserve(skb,2); + blist_ent_ptr->bd.BuffLength=PKT_BUF_SZ-2; + blist_ent_ptr->fd.FDLength=1; + blist_ent_ptr->fd.FDCtl=FDCTL_COWNSFD_MSK; + blist_ent_ptr->bd.BDCtl=BDCTL_COWNSBD_MSK; +} + + +static int ether00_mem_init(struct net_device* dev) +{ + struct net_priv* priv=dev->priv; + struct tx_fda_ent *tx_fd_ptr,*tx_end_ptr; + struct rx_blist_ent* blist_ent_ptr; + int i; + + /* + * Grab a block of on chip SRAM to contain the control stuctures for + * the ethernet MAC. This uncached becuase it needs to be accesses by both + * bus masters (cpu + mac). However, it shouldn't matter too much in terms + * of speed as its on chip memory + */ + priv->dma_data=ioremap_nocache(EXC_SPSRAM_BLOCK0_BASE,EXC_SPSRAM_BLOCK0_SIZE ); + if (!priv->dma_data) + return -ENOMEM; + + priv->rx_fda_ptr=(struct rx_fda_ent*)priv->dma_data; + /* + * Now share it out amongst the Frame descriptors and the buffer list + */ + priv->rx_blist_vp=(struct rx_blist_ent*)((unsigned int)priv->dma_data+RX_NUM_FDESC*sizeof(struct rx_fda_ent)); + + /* + *Initalise the FDA list + */ + /* set ownership to the controller */ + memset(priv->rx_fda_ptr,0x80,RX_NUM_FDESC*sizeof(struct rx_fda_ent)); + + /* + *Initialise the buffer list + */ + blist_ent_ptr=priv->rx_blist_vp; + i=0; + while(blist_ent_ptr<(priv->rx_blist_vp+RX_NUM_BUFF)){ + struct sk_buff *skb; + blist_ent_ptr->fd.FDLength=1; + skb=dev_alloc_skb(PKT_BUF_SZ); + if(skb){ + setup_blist_entry(skb,blist_ent_ptr); + blist_ent_ptr->fd.FDNext=(FDA_DESC*)__dma_pa(blist_ent_ptr+1); + blist_ent_ptr->bd.BDStat=i++; + blist_ent_ptr++; + } + else + { + printk("Failed to initalise buffer list\n"); + } + + } + blist_ent_ptr--; + blist_ent_ptr->fd.FDNext=(FDA_DESC*)__dma_pa(priv->rx_blist_vp); + + priv->tx_fdalist_vp=(struct tx_fda_ent*)(priv->rx_blist_vp+RX_NUM_BUFF); + + /* Initialise the buffers to be a circular list. The mac will then go poll + * the list until it finds a frame ready to transmit */ + tx_end_ptr=priv->tx_fdalist_vp+TX_NUM_FDESC; + for(tx_fd_ptr=priv->tx_fdalist_vp;tx_fd_ptrfd.FDNext=(FDA_DESC*)__dma_pa((tx_fd_ptr+1)); + tx_fd_ptr->fd.FDCtl=1; + tx_fd_ptr->fd.FDStat=0; + tx_fd_ptr->fd.FDSystem=0; + tx_fd_ptr->fd.FDLength=1; + + } + /* Change the last FDNext pointer to make a circular list */ + tx_fd_ptr--; + tx_fd_ptr->fd.FDNext=(FDA_DESC*)__dma_pa(priv->tx_fdalist_vp); + + /* Point the device at the chain of Rx and Tx Buffers */ + writel((unsigned int)__dma_pa(priv->rx_fda_ptr),ETHER_FDA_BAS(dev->base_addr)); + writel((RX_NUM_FDESC-1)*sizeof(struct rx_fda_ent),ETHER_FDA_LIM(dev->base_addr)); + writel((unsigned int)__dma_pa(priv->rx_blist_vp),ETHER_BLFRMPTR(dev->base_addr)); + + priv->tx_tail = priv->tx_head = (unsigned int) priv->tx_fdalist_vp; + writel((unsigned int) __dma_pa(priv->tx_head), ETHER_TXFRMPTR(dev->base_addr)); + + return 0; +} + + +void ether00_mem_update(void* dev_id) +{ + struct net_device* dev=dev_id; + struct net_priv* priv=dev->priv; + struct rx_blist_ent* blist_ent_ptr; + unsigned long flags; + int enable_rx = 0; + + priv->tq_memupdate.sync=0; + priv->memupdate_scheduled=0; + + /* Fill in any missing buffers from the received queue */ + blist_ent_ptr=priv->rx_blist_vp; + while(blist_ent_ptr<(priv->rx_blist_vp+RX_NUM_BUFF)){ + spin_lock_irqsave(&priv->dma_lock,flags); + /* fd.FDSystem of 0 indicates we failed to allocate the buffer in the ISR */ + if(!blist_ent_ptr->fd.FDSystem){ + struct sk_buff *skb; + skb=dev_alloc_skb(PKT_BUF_SZ); + blist_ent_ptr->fd.FDSystem=(unsigned int)skb; + if(skb){ + setup_blist_entry(skb,blist_ent_ptr); + enable_rx = 1; + } + else + { + /* + * reschedule the clean up, since we + * didn't patch up all the buffers + */ + + if(!priv->memupdate_scheduled){ + schedule_task(&priv->tq_memupdate); + priv->memupdate_scheduled=1; + } + spin_unlock_irqrestore(&priv->dma_lock,flags); + break; + } + } + spin_unlock_irqrestore(&priv->dma_lock,flags); + blist_ent_ptr++; + } + + if(enable_rx){ + if (!priv->rx_disabled){ + priv->rx_disabled = 0; + writel(ETHER_RX_CTL_RXEN_MSK,ETHER_RX_CTL(dev->base_addr)); + } + } +} + + +static void ether00_int( int irq_num, void* dev_id, struct pt_regs* regs) +{ + struct net_device* dev=dev_id; + struct net_priv* priv=dev->priv; + + unsigned int interruptValue; + + int enable_tx = 0; + struct tx_fda_ent *fda_ptr; + struct sk_buff* skb; + + interruptValue=readl(ETHER_INT_SRC(dev->base_addr)); + + if(!(readl(ETHER_INT_SRC(dev->base_addr)) & ETHER_INT_SRC_IRQ_MSK)) + { + return; /* Interrupt wasn't caused by us!! */ + } + + if(readl(ETHER_INT_SRC(dev->base_addr))& + (ETHER_INT_SRC_INTMACRX_MSK | + ETHER_INT_SRC_FDAEX_MSK | + ETHER_INT_SRC_BLEX_MSK)) { + struct rx_blist_ent* blist_ent_ptr; + struct rx_fda_ent* fda_ent_ptr; + struct sk_buff* skb; + + fda_ent_ptr=priv->rx_fda_ptr; + spin_lock(&priv->dma_lock); + while(fda_ent_ptr<(priv->rx_fda_ptr+RX_NUM_FDESC)){ + int result; + + if(!(fda_ent_ptr->fd.FDCtl&FDCTL_COWNSFD_MSK)) + { + /* This frame is ready for processing */ + /*find the corresponding buffer in the bufferlist */ + blist_ent_ptr=priv->rx_blist_vp+fda_ent_ptr->bd.BDStat; + skb=(struct sk_buff*)blist_ent_ptr->fd.FDSystem; + + /* Pass this skb up the stack */ + skb->dev=dev; + skb_put(skb,fda_ent_ptr->fd.FDLength); + skb->protocol=eth_type_trans(skb,dev); + skb->ip_summed=CHECKSUM_UNNECESSARY; + result=netif_rx(skb); + /* Update statistics */ + priv->stats.rx_packets++; + priv->stats.rx_bytes+=fda_ent_ptr->fd.FDLength; + + /* Free the FDA entry */ + fda_ent_ptr->bd.BDStat=0xff; + fda_ent_ptr->fd.FDCtl=FDCTL_COWNSFD_MSK; + + /* Allocate a new skb and point the bd entry to it */ + blist_ent_ptr->fd.FDSystem=0; + skb=dev_alloc_skb(PKT_BUF_SZ); + if(skb){ + setup_blist_entry(skb,blist_ent_ptr); + + } + else if(!priv->memupdate_scheduled){ + int tmp; + /* There are no buffers at the moment, so schedule */ + /* the background task to sort this out */ + schedule_task(&priv->tq_memupdate); + priv->memupdate_scheduled=1; + printk(KERN_DEBUG "%s:No buffers",dev->name); + /* If this interrupt was due to a lack of buffers then + * we'd better stop the receiver too */ + if(interruptValueÐER_INT_SRC_BLEX_MSK){ + priv->rx_disabled=1; + tmp=readl(ETHER_INT_SRC(dev->base_addr)); + writel(tmp&~ETHER_RX_CTL_RXEN_MSK,ETHER_RX_CTL(dev->base_addr)); + printk(KERN_DEBUG "%s:Halting rx",dev->name); + } + + } + + } + fda_ent_ptr++; + } + spin_unlock(&priv->dma_lock); + + /* Clear the interrupts */ + writel(ETHER_INT_SRC_INTMACRX_MSK | ETHER_INT_SRC_FDAEX_MSK + | ETHER_INT_SRC_BLEX_MSK,ETHER_INT_SRC(dev->base_addr)); + + } + + if(readl(ETHER_INT_SRC(dev->base_addr))ÐER_INT_SRC_INTMACTX_MSK){ + + /* Transmit interrupt */ + + fda_ptr=(struct tx_fda_ent*) priv->tx_tail; + + /* free up all completed frames */ + + while(!(FDCTL_COWNSFD_MSK&fda_ptr->fd.FDCtl) && fda_ptr->fd.FDSystem){ + priv->stats.tx_packets++; + priv->stats.tx_bytes+=fda_ptr->bd.BuffLength; + skb=(struct sk_buff*)fda_ptr->fd.FDSystem; + dev_kfree_skb_irq(skb); + fda_ptr->fd.FDSystem=0; + fda_ptr->fd.FDStat=0; + fda_ptr->fd.FDCtl=0; + fda_ptr = (struct tx_fda_ent *)__dma_va(fda_ptr->fd.FDNext); + enable_tx = 1; + } + priv->tx_tail = (unsigned int) fda_ptr; + + if(priv->queue_stopped && enable_tx){ + priv->queue_stopped=0; + netif_wake_queue(dev); + } + + /* Clear the interrupt */ + writel(ETHER_INT_SRC_INTMACTX_MSK,ETHER_INT_SRC(dev->base_addr)); + } + + if (readl(ETHER_INT_SRC(dev->base_addr)) & (ETHER_INT_SRC_SWINT_MSK| + ETHER_INT_SRC_INTEARNOT_MSK| + ETHER_INT_SRC_INTLINK_MSK| + ETHER_INT_SRC_INTEXBD_MSK| + ETHER_INT_SRC_INTTXCTLCMP_MSK)) + { + /* + * Not using any of these so they shouldn't happen + * + * In the cased of INTEXBD - if you allocate more + * than 28 decsriptors you may need to think about this + */ + printk("Not using this interrupt\n"); + } + + if (readl(ETHER_INT_SRC(dev->base_addr)) & + (ETHER_INT_SRC_INTSBUS_MSK | + ETHER_INT_SRC_INTNRABT_MSK + |ETHER_INT_SRC_DMPARERR_MSK)) + { + /* + * Hardware errors, we can either ignore them and hope they go away + *or reset the device, I'll try the first for now to see if they happen + */ + printk("Hardware error\n"); + } +} + +static void ether00_setup_ethernet_address(struct net_device* dev) +{ + int tmp; + + dev->addr_len=6; + writew(0,ETHER_ARC_ADR(dev->base_addr)); + writel((dev->dev_addr[0]<<24) | + (dev->dev_addr[1]<<16) | + (dev->dev_addr[2]<<8) | + dev->dev_addr[3], + ETHER_ARC_DATA(dev->base_addr)); + + writew(4,ETHER_ARC_ADR(dev->base_addr)); + tmp=readl(ETHER_ARC_DATA(dev->base_addr)); + tmp&=0xffff; + tmp|=(dev->dev_addr[4]<<24) | (dev->dev_addr[5]<<16); + writel(tmp, ETHER_ARC_DATA(dev->base_addr)); + /* Enable this entry in the ARC */ + + writel(1,ETHER_ARC_ENA(dev->base_addr)); + + return; +} + + +static void ether00_reset(struct net_device *dev) +{ + /* reset the controller */ + writew(ETHER_MAC_CTL_RESET_MSK,ETHER_MAC_CTL(dev->base_addr)); + + /* + * Make sure we're not going to send anything + */ + + writew(ETHER_TX_CTL_TXHALT_MSK,ETHER_TX_CTL(dev->base_addr)); + + /* + * Make sure we're not going to receive anything + */ + writew(ETHER_RX_CTL_RXHALT_MSK,ETHER_RX_CTL(dev->base_addr)); + + /* + * Disable Interrupts for now, and set the burst size to 8 bytes + */ + + writel(ETHER_DMA_CTL_INTMASK_MSK | + ((8 << ETHER_DMA_CTL_DMBURST_OFST) & ETHER_DMA_CTL_DMBURST_MSK) + |(2<base_addr)); + + + /* + * Set TxThrsh - start transmitting a packet after 1514 + * bytes or when a packet is complete, whichever comes first + */ + writew(1514,ETHER_TXTHRSH(dev->base_addr)); + + /* + * Set TxPollCtr. Each cycle is + * 61.44 microseconds with a 33 MHz bus + */ + writew(1,ETHER_TXPOLLCTR(dev->base_addr)); + + /* + * Set Rx_Ctl - Turn off reception and let RxData turn it + * on later + */ + writew(ETHER_RX_CTL_RXHALT_MSK,ETHER_RX_CTL(dev->base_addr)); + +} + + +static void ether00_set_multicast(struct net_device* dev) +{ + int count=dev->mc_count; + + /* Set promiscuous mode if it's asked for. */ + + if (dev->flags&IFF_PROMISC){ + + writew( ETHER_ARC_CTL_COMPEN_MSK | + ETHER_ARC_CTL_BROADACC_MSK | + ETHER_ARC_CTL_GROUPACC_MSK | + ETHER_ARC_CTL_STATIONACC_MSK, + ETHER_ARC_CTL(dev->base_addr)); + return; + } + + /* + * Get all multicast packets if required, or if there are too + * many addresses to fit in hardware + */ + if (dev->flags & IFF_ALLMULTI){ + writew( ETHER_ARC_CTL_COMPEN_MSK | + ETHER_ARC_CTL_GROUPACC_MSK | + ETHER_ARC_CTL_BROADACC_MSK, + ETHER_ARC_CTL(dev->base_addr)); + return; + } + if (dev->mc_count > (ETHER_ARC_SIZE - 1)){ + + printk(KERN_WARNING "Too many multicast addresses for hardware to filter - receiving all multicast packets\n"); + writew( ETHER_ARC_CTL_COMPEN_MSK | + ETHER_ARC_CTL_GROUPACC_MSK | + ETHER_ARC_CTL_BROADACC_MSK, + ETHER_ARC_CTL(dev->base_addr)); + return; + } + + if(dev->mc_count){ + struct dev_mc_list *mc_list_ent=dev->mc_list; + unsigned int temp,i; + DEBUG(printk("mc_count=%d mc_list=%#x\n",dev-> mc_count, dev->mc_list)); + DEBUG(printk("mc addr=%02#x%02x%02x%02x%02x%02x\n", + mc_list_ent->dmi_addr[5], + mc_list_ent->dmi_addr[4], + mc_list_ent->dmi_addr[3], + mc_list_ent->dmi_addr[2], + mc_list_ent->dmi_addr[1], + mc_list_ent->dmi_addr[0]);) + + /* + * The first 6 bytes are the MAC address, so + * don't change them! + */ + writew(4,ETHER_ARC_ADR(dev->base_addr)); + temp=readl(ETHER_ARC_DATA(dev->base_addr)); + temp&=0xffff0000; + + /* Disable the current multicast stuff */ + writel(1,ETHER_ARC_ENA(dev->base_addr)); + + for(;;){ + temp|=mc_list_ent->dmi_addr[1] | + mc_list_ent->dmi_addr[0]<<8; + writel(temp,ETHER_ARC_DATA(dev->base_addr)); + + i=readl(ETHER_ARC_ADR(dev->base_addr)); + writew(i+4,ETHER_ARC_ADR(dev->base_addr)); + + temp=mc_list_ent->dmi_addr[5]| + mc_list_ent->dmi_addr[4]<<8 | + mc_list_ent->dmi_addr[3]<<16 | + mc_list_ent->dmi_addr[2]<<24; + writel(temp,ETHER_ARC_DATA(dev->base_addr)); + + count--; + if(!mc_list_ent->next || !count){ + break; + } + DEBUG(printk("mc_list_next=%#x\n",mc_list_ent->next);) + mc_list_ent=mc_list_ent->next; + + + i=readl(ETHER_ARC_ADR(dev->base_addr)); + writel(i+4,ETHER_ARC_ADR(dev->base_addr)); + + temp=mc_list_ent->dmi_addr[3]| + mc_list_ent->dmi_addr[2]<<8 | + mc_list_ent->dmi_addr[1]<<16 | + mc_list_ent->dmi_addr[0]<<24; + writel(temp,ETHER_ARC_DATA(dev->base_addr)); + + i=readl(ETHER_ARC_ADR(dev->base_addr)); + writel(i+4,ETHER_ARC_ADR(dev->base_addr)); + + temp=mc_list_ent->dmi_addr[4]<<16 | + mc_list_ent->dmi_addr[5]<<24; + + writel(temp,ETHER_ARC_DATA(dev->base_addr)); + + count--; + if(!mc_list_ent->next || !count){ + break; + } + mc_list_ent=mc_list_ent->next; + } + + + if(count) + printk(KERN_WARNING "Multicast list size error\n"); + + + writew( ETHER_ARC_CTL_BROADACC_MSK| + ETHER_ARC_CTL_COMPEN_MSK, + ETHER_ARC_CTL(dev->base_addr)); + + } + + /* enable the active ARC enties */ + writew((1<<(dev->mc_count+2))-1,ETHER_ARC_ENA(dev->base_addr)); +} + + +static int ether00_open(struct net_device* dev) +{ + int result,tmp; + struct net_priv* priv; + + if (!ether00_get_ethernet_address(dev)){ + printk("%s: Invalid ethernet MAC address. Please set using " + "ifconfig\n", dev->name); + return -EINVAL; + } + + dev->base_addr=(unsigned int)ioremap_nocache(base,SZ_4K); + + dev->irq=irq; + + /* Allocate private memory */ + dev->priv=kmalloc(sizeof(struct net_priv),GFP_KERNEL); + if(!dev->priv) + return -ENOMEM; + memset(dev->priv,0,sizeof(struct net_priv)); + priv=(struct net_priv*)dev->priv; + priv->tq_memupdate.routine=ether00_mem_update; + priv->tq_memupdate.data=(void*) dev; + spin_lock_init(&priv->dma_lock); + + /* Install interrupt handlers */ + result=request_irq(dev->irq,ether00_int,0,"ether00",dev); + if(result) + goto open_err1; + + result=request_irq(phy_irq,ether00_phy_int,0,"ether00_phy",dev); + if(result) + goto open_err2; + + ether00_reset(dev); + result=ether00_mem_init(dev); + if(result) + goto open_err3; + + + ether00_setup_ethernet_address(dev); + + ether00_set_multicast(dev); + + result=ether00_write_phy(dev,PHY_CONTROL, PHY_CONTROL_ANEGEN_MSK | PHY_CONTROL_RANEG_MSK); + if(result) + goto open_err4; + result=ether00_write_phy(dev,PHY_IRQ_CONTROL, PHY_IRQ_CONTROL_LS_CHG_IE_MSK | + PHY_IRQ_CONTROL_ANEG_COMP_IE_MSK); + if(result) + goto open_err4; + + /* Start the device enable interrupts */ + writew(ETHER_RX_CTL_RXEN_MSK +// | ETHER_RX_CTL_STRIPCRC_MSK + | ETHER_RX_CTL_ENGOOD_MSK + | ETHER_RX_CTL_ENRXPAR_MSK| ETHER_RX_CTL_ENLONGERR_MSK + | ETHER_RX_CTL_ENOVER_MSK| ETHER_RX_CTL_ENCRCERR_MSK, + ETHER_RX_CTL(dev->base_addr)); + + writew(ETHER_TX_CTL_TXEN_MSK| + ETHER_TX_CTL_ENEXDEFER_MSK| + ETHER_TX_CTL_ENLCARR_MSK| + ETHER_TX_CTL_ENEXCOLL_MSK| + ETHER_TX_CTL_ENLATECOLL_MSK| + ETHER_TX_CTL_ENTXPAR_MSK| + ETHER_TX_CTL_ENCOMP_MSK, + ETHER_TX_CTL(dev->base_addr)); + + tmp=readl(ETHER_DMA_CTL(dev->base_addr)); + writel(tmp&~ETHER_DMA_CTL_INTMASK_MSK,ETHER_DMA_CTL(dev->base_addr)); + + return 0; + + open_err4: + ether00_reset(dev); + open_err3: + free_irq(2,dev); + open_err2: + free_irq(dev->irq,dev); + open_err1: + iounmap((void*)dev->base_addr); + kfree(dev->priv); + return result; + +} + + +static int ether00_tx(struct sk_buff* skb, struct net_device* dev) +{ + struct net_priv *priv=dev->priv; + struct tx_fda_ent *fda_ptr; + unsigned long flags; + int retcode = 0; + + /* + * Find an empty slot in which to stick the frame + */ + + spin_lock_irqsave(&priv->dma_lock,flags); + + fda_ptr=(struct tx_fda_ent*) priv->tx_head; + + priv->tx_head =(unsigned int) __dma_va(fda_ptr->fd.FDNext); + + /* Write the skb data from the cache*/ + consistent_sync(skb->data,skb->len,PCI_DMA_TODEVICE); + fda_ptr->bd.BuffData=(char*)__pa(skb->data); + fda_ptr->bd.BuffLength=(unsigned short)skb->len; + /* Save the pointer to the skb for freeing later */ + fda_ptr->fd.FDSystem=(unsigned int)skb; + fda_ptr->fd.FDStat=0; + /* Pass ownership of the buffers to the controller */ + fda_ptr->fd.FDCtl=1; + fda_ptr->fd.FDCtl|=FDCTL_COWNSFD_MSK; + + fda_ptr=(struct tx_fda_ent*) priv->tx_head; + + /* If the next buffer in the list is full, stop the queue */ + if (fda_ptr->fd.FDSystem){ + netif_stop_queue(dev); + priv->queue_stopped=1; + } + + spin_unlock_irqrestore(&priv->dma_lock,flags); + return retcode; +} + +static void ether00_tx_timeout(struct net_device* dev) +{ + /* + * Something really bad has happened here. This + * SHOULD never happen. Given that, it's difficult + * to know what to do to recover. For now we'll just + * count the error, and restart the queue. + */ + + struct net_priv *priv=dev->priv; + + priv->stats.tx_errors++; + + priv->queue_stopped=0; + netif_wake_queue(dev); + +} + +static struct net_device_stats *ether00_stats(struct net_device* dev) +{ + struct net_priv *priv=dev->priv; + return &priv->stats; +} + + +static int ether00_stop(struct net_device* dev) +{ + struct net_priv *priv=dev->priv; + int tmp; + + /* Stop/disable the device. */ + tmp=readw(ETHER_RX_CTL(dev->base_addr)); + tmp&=~(ETHER_RX_CTL_RXEN_MSK | ETHER_RX_CTL_ENGOOD_MSK); + tmp|=ETHER_RX_CTL_RXHALT_MSK; + writew(tmp,ETHER_RX_CTL(dev->base_addr)); + + tmp=readl(ETHER_TX_CTL(dev->base_addr)); + tmp&=~ETHER_TX_CTL_TXEN_MSK; + tmp|=ETHER_TX_CTL_TXHALT_MSK; + writel(tmp,ETHER_TX_CTL(dev->base_addr)); + + /* Free up system resources */ + free_irq(dev->irq,dev); + free_irq(2,dev); + iounmap(priv->dma_data); + iounmap((void*)dev->base_addr); + kfree(priv); + + return 0; +} + + +static int ether00_get_ethernet_address(struct net_device* dev) +{ + struct mtd_info *mymtd=NULL; + size_t retlen; + + /* + * For the Epxa10 dev board (camelot), the ethernet MAC + * address is of the form 00:aa:aa:00:xx:xx where + * 00:aa:aa is the Altera vendor ID and xx:xx is the + * last 2 bytes of the board serial number, as programmed + * into the OTP area of the flash device on EBI1. If this + * isn't an expa10 dev board, or there's no mtd support to + * read the serial number from flash then we'll force the + * use to set their own mac address using ifconfig. + */ + +#ifdef CONFIG_ARCH_CAMELOT +#ifdef CONFIG_MTD + /* get the mtd_info structure for the first mtd device*/ + mymtd=get_mtd_device(NULL,0); + + if(!mymtd || !mymtd->read_user_prot_reg){ + printk(KERN_WARNING "%s: Failed to read MAC address from flash\n",dev->name); + }else{ + mymtd->read_user_prot_reg(mymtd,2,1,&retlen,&dev->dev_addr[5]); + mymtd->read_user_prot_reg(mymtd,3,1,&retlen,&dev->dev_addr[4]); + dev->dev_addr[3]=0; + dev->dev_addr[2]=vendor_id[1]; + dev->dev_addr[1]=vendor_id[0]; + dev->dev_addr[0]=0; + } +#else + printk(KERN_WARNING "%s: MTD support required to read MAC address from EPXA10 dev board\n", dev->name); +#endif +#endif + + return (is_valid_ether_addr(dev->dev_addr)); + +} + +static int ether00_init(struct net_device* dev) +{ + + ether_setup(dev); + + dev->open=ether00_open; + dev->stop=ether00_stop; + dev->set_multicast_list=ether00_set_multicast; + dev->hard_start_xmit=ether00_tx; + dev->get_stats=ether00_stats; + dev->tx_timeout=ether00_tx_timeout; + dev->watchdog_timeo=TX_TIMEOUT; + + SET_MODULE_OWNER(dev); + return 0; +} + + +struct net_device ether00_dev={ + init:ether00_init, + name:"eth%d", +}; + + +static void __exit ether00_cleanup_module(void) +{ + unregister_netdev(ðer00_dev); +} +module_exit(ether00_cleanup_module); + + +static int __init ether00_mod_init(void) +{ + int result; + + result=register_netdev(ðer00_dev); + if(result) + printk("Ether00: Error %i registering driver\n",result); + return result; +} + +module_init(ether00_mod_init); + + diff -urN orig/drivers/net/irda/Config.in linux/drivers/net/irda/Config.in --- orig/drivers/net/irda/Config.in Mon Aug 5 13:31:06 2002 +++ linux/drivers/net/irda/Config.in Mon Aug 5 13:44:21 2002 @@ -33,7 +33,7 @@ dep_tristate 'VLSI 82C147 SIR/MIR/FIR (Experimental)' CONFIG_VLSI_FIR $CONFIG_IRDA fi if [ "$CONFIG_ARCH_SA1100" = "y" ]; then - dep_tristate 'SA1100 Internal IR' CONFIG_SA1100_FIR $CONFIG_IRDA + dep_tristate 'SA1100 Internal IR' CONFIG_SA1100_FIR $CONFIG_IRDA $CONFIG_EXPERIMENTAL fi endmenu diff -urN orig/drivers/net/irda/sa1100_ir.c linux/drivers/net/irda/sa1100_ir.c --- orig/drivers/net/irda/sa1100_ir.c Mon Aug 5 13:31:06 2002 +++ linux/drivers/net/irda/sa1100_ir.c Mon Aug 5 13:44:22 2002 @@ -38,11 +38,7 @@ #include -#ifndef CONFIG_SA1100_H3600 -#define clr_h3600_egpio(x) do { } while (0) -#define set_h3600_egpio(x) do { } while (0) -#endif - +/* Yopy wants fixing */ #ifndef GPIO_IRDA_FIR #define GPIO_IRDA_FIR (0) #endif @@ -174,8 +170,8 @@ if (machine_is_assabet()) ASSABET_BCR_clear(ASSABET_BCR_IRDA_FSEL); - if (machine_is_h3600()) - clr_h3600_egpio(EGPIO_H3600_IR_FSEL); + if (machine_is_h3xxx()) + clr_h3600_egpio(IPAQ_EGPIO_IR_FSEL); if (machine_is_yopy()) PPSR &= ~GPIO_IRDA_FIR; @@ -199,8 +195,8 @@ if (machine_is_assabet()) ASSABET_BCR_set(ASSABET_BCR_IRDA_FSEL); - if (machine_is_h3600()) - set_h3600_egpio(EGPIO_H3600_IR_FSEL); + if (machine_is_h3xxx()) + set_h3600_egpio(IPAQ_EGPIO_IR_FSEL); if (machine_is_yopy()) PPSR |= GPIO_IRDA_FIR; @@ -246,10 +242,7 @@ static inline int sa1100_irda_set_power_h3600(struct sa1100_irda *si, unsigned int state) { - if (state) - set_h3600_egpio(EGPIO_H3600_IR_ON); - else - clr_h3600_egpio(EGPIO_H3600_IR_ON); + assign_h3600_egpio( IPAQ_EGPIO_IR_ON, state ); return 0; } @@ -283,7 +276,7 @@ if (machine_is_assabet()) ret = sa1100_irda_set_power_assabet(si, state); - if (machine_is_h3600()) + if (machine_is_h3xxx()) ret = sa1100_irda_set_power_h3600(si, state); if (machine_is_yopy()) ret = sa1100_irda_set_power_yopy(si, state); @@ -727,11 +720,6 @@ netif_wake_queue(dev); } -/* - * Note that we will never build up a backlog of frames; the protocol is a - * half duplex protocol which basically means we transmit a frame, we - * receive a frame, we transmit the next frame etc. - */ static int sa1100_irda_hard_xmit(struct sk_buff *skb, struct net_device *dev) { struct sa1100_irda *si = dev->priv; @@ -758,6 +746,8 @@ } if (!IS_FIR(si)) { + netif_stop_queue(dev); + si->tx_buff.data = si->tx_buff.head; si->tx_buff.len = async_wrap_skb(skb, si->tx_buff.data, si->tx_buff.truesize); diff -urN orig/drivers/net/irda/w83977af_ir.c linux/drivers/net/irda/w83977af_ir.c --- orig/drivers/net/irda/w83977af_ir.c Wed Feb 27 14:25:06 2002 +++ linux/drivers/net/irda/w83977af_ir.c Tue Feb 26 17:39:05 2002 @@ -205,7 +205,7 @@ /* FIXME: The HP HDLS-1100 does not support 1152000! */ self->qos.baud_rate.bits = IR_9600|IR_19200|IR_38400|IR_57600| - IR_115200|IR_576000|IR_1152000|(IR_4000000 << 8); + IR_115200/*|IR_576000|IR_1152000|(IR_4000000 << 8)*/; /* The HP HDLS-1100 needs 1 ms according to the specs */ self->qos.min_turn_time.bits = qos_mtt_bits; @@ -1344,7 +1344,7 @@ case SIOCSBANDWIDTH: /* Set bandwidth */ if (!capable(CAP_NET_ADMIN)) { ret = -EPERM; - goto out; + break; } w83977af_change_speed(self, irq->ifr_baudrate); break; diff -urN orig/drivers/net/smc9194.c linux/drivers/net/smc9194.c --- orig/drivers/net/smc9194.c Mon Aug 5 13:31:09 2002 +++ linux/drivers/net/smc9194.c Mon Aug 5 16:02:09 2002 @@ -12,8 +12,8 @@ . AUI/TP selection ( mine has 10Base2/10BaseT select ) . . Arguments: - . io = for the base address - . irq = for the IRQ + . io = for the base address + . irq = for the IRQ . ifport = 0 for autodetect, 1 for TP, 2 for AUI ( or 10base2 ) . . author: @@ -51,12 +51,21 @@ . allocation . 08/20/00 Arnaldo Melo fix kfree(skb) in smc_hardware_send_packet . 12/15/00 Christian Jullien fix "Warning: kfree_skb on hard IRQ" + . 06/23/01 Russell King Separate out IO functions for different bus + . types. + . Use dev->name instead of CARDNAME for printk + . Add ethtool support, full duplex support + . Add LAN91C96 support. . 11/08/01 Matt Domsch Use common crc32 function ----------------------------------------------------------------------------*/ +#define DRV_NAME "smc9194" +#define DRV_VERSION "0.15" + static const char version[] = - "smc9194.c:v0.14 12/15/00 by Erik Stahlman (erik@vt.edu)\n"; + DRV_NAME ".c:v" DRV_VERSION " 12/15/00 by Erik Stahlman (erik@vt.edu)\n"; +#include #include #include #include @@ -69,16 +78,26 @@ #include #include #include +#include #include #include -#include -#include #include +#include #include #include #include +#include +#include +#include +#include + +#ifdef CONFIG_ARCH_SA1100 +#include +#include +#endif + #include "smc9194.h" /*------------------------------------------------------------------------ . @@ -152,29 +171,27 @@ -------------------------------------------------------------------------*/ #define CARDNAME "SMC9194" +static const char *chip_ids[15] = { + NULL, + NULL, + NULL, + "SMC91C90/91C92", /* 3 */ + "SMC91C94/91C96", /* 4 */ + "SMC91C95", /* 5 */ + NULL, + "SMC91C100", /* 7 */ + "SMC91C100FD", /* 8 */ + NULL, + NULL, + NULL, + NULL, + NULL, + NULL +}; -/* store this information for the driver.. */ -struct smc_local { - /* - these are things that the kernel wants me to keep, so users - can find out semi-useless statistics of how well the card is - performing - */ - struct net_device_stats stats; - - /* - If I have to wait until memory is available to send - a packet, I will store the skbuff here, until I get the - desired memory. Then, I'll send it out and free it. - */ - struct sk_buff * saved_skb; - - /* - . This keeps track of how many packets that I have - . sent out. When an TX_EMPTY interrupt comes, I know - . that all of these have been sent. - */ - int packets_waiting; +static const char * interfaces[2] = { + "TP", + "AUI" }; @@ -202,6 +219,11 @@ static int smc_open(struct net_device *dev); /* + . This handles the ethtool interface +*/ +static int smc_ioctl(struct net_device *dev, struct ifreq *rq, int cmd); + +/* . Our watchdog timed out. Called by the networking layer */ static void smc_timeout(struct net_device *dev); @@ -217,11 +239,11 @@ . This routine allows the proc file system to query the driver's . statistics. */ -static struct net_device_stats * smc_query_statistics( struct net_device *dev); +static struct net_device_stats * smc_query_statistics(struct net_device *dev); /* - . Finally, a call to set promiscuous mode ( for TCPDUMP and related - . programs ) and multicast modes. + . Finally, a call to set promiscuous mode (for TCPDUMP and related + . programs) and multicast modes. */ static void smc_set_multicast_list(struct net_device *dev); @@ -240,12 +262,12 @@ . This is a separate procedure to handle the receipt of a packet, to . leave the interrupt code looking slightly cleaner */ -static inline void smc_rcv( struct net_device *dev ); +static inline void smc_rcv(struct net_device *dev); /* . This handles a TX interrupt, which is only called when an error . relating to a packet is sent. */ -static inline void smc_tx( struct net_device * dev ); +static inline void smc_tx(struct net_device * dev); /* ------------------------------------------------------------ @@ -261,39 +283,287 @@ */ static int smc_probe(struct net_device *dev, int ioaddr); -/* - . A rather simple routine to print out a packet for debugging purposes. -*/ -#if SMC_DEBUG > 2 -static void print_packet( byte *, int ); -#endif - -#define tx_done(dev) 1 - /* this is called to actually send the packet to the chip */ -static void smc_hardware_send_packet( struct net_device * dev ); +static void smc_hardware_send_packet(struct net_device * dev); /* Since I am not sure if I will have enough room in the chip's ram . to store the packet, I call this routine, which either sends it . now, or generates an interrupt when the card is ready for the . packet */ -static int smc_wait_to_send_packet( struct sk_buff * skb, struct net_device *dev ); +static int smc_wait_to_send_packet(struct sk_buff * skb, struct net_device *dev); /* this does a soft reset on the device */ -static void smc_reset( int ioaddr ); +static void smc_reset(struct net_device *dev); /* Enable Interrupts, Receive, and Transmit */ -static void smc_enable( int ioaddr ); +static void smc_enable(struct net_device *dev); /* this puts the device in an inactive state */ -static void smc_shutdown( int ioaddr ); +static void smc_shutdown(struct net_device *dev); /* This routine will find the IRQ of the driver if one is not . specified in the input to the device. */ -static int smc_findirq( int ioaddr ); +static int smc_findirq(struct net_device *dev); + +#ifndef CONFIG_ASSABET_NEPONSET +/* + * These functions allow us to handle IO addressing as we wish - this + * ethernet controller can be connected to a variety of busses. Some + * busses do not support 16 bit or 32 bit transfers. --rmk + */ +static inline u8 smc_inb(u_int base, u_int reg) +{ + return inb(base + reg); +} + +static inline u16 smc_inw(u_int base, u_int reg) +{ + return inw(base + reg); +} + +static inline void smc_ins(u_int base, u_int reg, u8 *data, u_int len) +{ + u_int port = base + reg; +#ifdef USE_32_BIT + /* QUESTION: Like in the TX routine, do I want + to send the DWORDs or the bytes first, or some + mixture. A mixture might improve already slow PIO + performance */ + PRINTK3((" Reading %d dwords (and %d bytes) \n", + len >> 2, len & 3)); + insl(port, data, len >> 2); + /* read the left over bytes */ + insb(port, data + (len & ~3), len & 3); +#else + PRINTK3((" Reading %d words and %d byte(s) \n", + len >> 1, len & 1)); + insw(port, data, len >> 1); + if (len & 1) { + data += len & ~1; + *data = inb(port); + } +#endif +} + +static inline void smc_outb(u8 val, u_int base, u_int reg) +{ + outb(val, base + reg); +} + +static inline void smc_outw(u16 val, u_int base, u_int reg) +{ + outw(val, base + reg); +} + +static inline void smc_outl(u32 val, u_int base, u_int reg) +{ + u_int port = base + reg; +#ifdef USE_32_BIT + outl(val, port); +#else + outw(val, port); + outw(val >> 16, port); +#endif +} + +static inline void smc_outs(u_int base, u_int reg, u8 *data, u_int len) +{ + u_int port = base + reg; +#ifdef USE_32_BIT + if (len & 2) { + outsl(port, data, len >> 2); + outw(*((word *)(data + (len & ~3))), port); + } + else + outsl(port, data, len >> 2); +#else + outsw(port, data, len >> 1); +#endif +} + + +/*------------------------------------------------------------------------- + . I define some macros to make it easier to do somewhat common + . or slightly complicated, repeated tasks. + --------------------------------------------------------------------------*/ + +/* select a register bank, 0 to 3 */ + +#define SMC_SELECT_BANK(x) \ + { \ + smc_outw(x, ioaddr, BANK_SELECT); \ + } + +/* define a small delay for the reset */ +#define SMC_DELAY() \ + { \ + smc_inw(ioaddr, RCR); \ + smc_inw(ioaddr, RCR); \ + smc_inw(ioaddr, RCR); \ + } + +/* this enables an interrupt in the interrupt mask register */ +#define SMC_ENABLE_INT(x) \ + { \ + byte mask; \ + mask = smc_inb(ioaddr, INT_MASK); \ + mask |= (x); \ + smc_outb(mask, ioaddr, INT_MASK); \ + } + +/* this sets the absolutel interrupt mask */ +#define SMC_SET_INT(x) \ + { \ + smc_outw((x), INT_MASK); \ + } + +#else + +#undef SMC_IO_EXTENT +#define SMC_IO_EXTENT (16 << 2) + +/* + * These functions allow us to handle IO addressing as we wish - this + * ethernet controller can be connected to a variety of busses. Some + * busses do not support 16 bit or 32 bit transfers. --rmk + */ +static inline u8 smc_inb(u_int base, u_int reg) +{ + u_int port = base + reg * 4; + + return readb(port); +} + +static inline u16 smc_inw(u_int base, u_int reg) +{ + u_int port = base + reg * 4; + + return readb(port) | readb(port + 4) << 8; +} + +static inline void smc_ins(u_int base, u_int reg, u8 *data, u_int len) +{ + u_int port = base + reg * 4; + + insb(port, data, len); +} + +static inline void smc_outb(u8 val, u_int base, u_int reg) +{ + u_int port = base + reg * 4; + + writeb(val, port); +} + +static inline void smc_outw(u16 val, u_int base, u_int reg) +{ + u_int port = base + reg * 4; + + writeb(val, port); + writeb(val >> 8, port + 4); +} + +static inline void smc_outl(u32 val, u_int base, u_int reg) +{ + u_int port = base + reg * 4; + + writeb(val, port); + writeb(val >> 8, port + 4); + writeb(val >> 16, port + 8); + writeb(val >> 24, port + 12); +} + +static inline void smc_outs(u_int base, u_int reg, u8 *data, u_int len) +{ + u_int port = base + reg * 4; + + outsb(port, data, len & ~1); +} + +/*------------------------------------------------------------------------- + . I define some macros to make it easier to do somewhat common + . or slightly complicated, repeated tasks. + --------------------------------------------------------------------------*/ + +/* select a register bank, 0 to 3 */ + +#define SMC_SELECT_BANK(x) \ + { \ + smc_outb(x, ioaddr, BANK_SELECT); \ + } + +/* define a small delay for the reset */ +#define SMC_DELAY() \ + { \ + smc_inb(ioaddr, RCR); \ + smc_inb(ioaddr, RCR); \ + smc_inb(ioaddr, RCR); \ + } + +/* this enables an interrupt in the interrupt mask register */ +#define SMC_ENABLE_INT(x) \ + { \ + byte mask; \ + mask = smc_inb(ioaddr, INT_MASK); \ + mask |= (x); \ + smc_outb(mask, ioaddr, INT_MASK); \ + } + +/* this sets the absolutel interrupt mask */ +#define SMC_SET_INT(x) \ + { \ + smc_outb((x), ioaddr, INT_MASK); \ + } + +#endif /* - . Function: smc_reset( int ioaddr ) + . A rather simple routine to print out a packet for debugging purposes. +*/ +#if SMC_DEBUG > 2 +static void print_packet(byte * buf, int length) +{ + int i; + int remainder; + int lines; + + printk("Packet of length %d \n", length); + lines = length / 16; + remainder = length % 16; + + for (i = 0; i < lines ; i ++) { + int cur; + + for (cur = 0; cur < 8; cur ++) { + byte a, b; + + a = *(buf ++); + b = *(buf ++); + printk("%02x%02x ", a, b); + } + printk("\n"); + } + for (i = 0; i < remainder/2 ; i++) { + byte a, b; + + a = *(buf ++); + b = *(buf ++); + printk("%02x%02x ", a, b); + } + if (remainder & 1) { + byte a; + + a = *buf++; + printk("%02x", a); + } + printk("\n"); +} +#else +#define print_packet(buf,len) do { } while (0) +#endif + +/* + . Function: smc_reset(struct net_device *dev) . Purpose: . This sets the SMC91xx chip to its normal state, hopefully from whatever . mess that any other DOS driver has put it in. @@ -309,36 +579,37 @@ . 5. clear all interrupts . */ -static void smc_reset( int ioaddr ) +static void smc_reset(struct net_device *dev) { + u_int ioaddr = dev->base_addr; + /* This resets the registers mostly to defaults, but doesn't affect EEPROM. That seems unnecessary */ - SMC_SELECT_BANK( 0 ); - outw( RCR_SOFTRESET, ioaddr + RCR ); + SMC_SELECT_BANK(0); + smc_outw(RCR_SOFTRESET, ioaddr, RCR); /* this should pause enough for the chip to be happy */ - SMC_DELAY( ); + SMC_DELAY(); /* Set the transmit and receive configuration registers to default values */ - outw( RCR_CLEAR, ioaddr + RCR ); - outw( TCR_CLEAR, ioaddr + TCR ); + smc_outw(RCR_CLEAR, ioaddr, RCR); + smc_outw(TCR_CLEAR, ioaddr, TCR); /* set the control register to automatically release successfully transmitted packets, to make the best use out of our limited memory */ - SMC_SELECT_BANK( 1 ); - outw( inw( ioaddr + CONTROL ) | CTL_AUTO_RELEASE , ioaddr + CONTROL ); + SMC_SELECT_BANK(1); + smc_outw(smc_inw(ioaddr, CONTROL) | CTL_AUTO_RELEASE, ioaddr, CONTROL); /* Reset the MMU */ - SMC_SELECT_BANK( 2 ); - outw( MC_RESET, ioaddr + MMU_CMD ); + SMC_SELECT_BANK(2); + smc_outw(MC_RESET, ioaddr, MMU_CMD); /* Note: It doesn't seem that waiting for the MMU busy is needed here, but this is a place where future chipsets _COULD_ break. Be wary of issuing another MMU command right after this */ - - outb( 0, ioaddr + INT_MASK ); + SMC_SET_INT(0); } /* @@ -349,20 +620,21 @@ . 2. Enable the receiver . 3. Enable interrupts */ -static void smc_enable( int ioaddr ) +static void smc_enable(struct net_device *dev) { - SMC_SELECT_BANK( 0 ); + u_int ioaddr = dev->base_addr; + SMC_SELECT_BANK(0); /* see the header file for options in TCR/RCR NORMAL*/ - outw( TCR_NORMAL, ioaddr + TCR ); - outw( RCR_NORMAL, ioaddr + RCR ); + smc_outw(TCR_NORMAL, ioaddr, TCR); + smc_outw(RCR_NORMAL, ioaddr, RCR); /* now, enable interrupts */ - SMC_SELECT_BANK( 2 ); - outb( SMC_INTERRUPT_MASK, ioaddr + INT_MASK ); + SMC_SELECT_BANK(2); + SMC_SET_INT(SMC_INTERRUPT_MASK); } /* - . Function: smc_shutdown + . Function: smc_shutdown(struct net_device *dev) . Purpose: closes down the SMC91xxx chip. . Method: . 1. zero the interrupt mask @@ -375,26 +647,28 @@ . the manual says that it will wake up in response to any I/O requests . in the register space. Empirical results do not show this working. */ -static void smc_shutdown( int ioaddr ) +static void smc_shutdown(struct net_device *dev) { + u_int ioaddr = dev->base_addr; + /* no more interrupts for me */ - SMC_SELECT_BANK( 2 ); - outb( 0, ioaddr + INT_MASK ); + SMC_SELECT_BANK(2); + SMC_SET_INT(0); /* and tell the card to stay away from that nasty outside world */ - SMC_SELECT_BANK( 0 ); - outb( RCR_CLEAR, ioaddr + RCR ); - outb( TCR_CLEAR, ioaddr + TCR ); + SMC_SELECT_BANK(0); + smc_outb(RCR_CLEAR, ioaddr, RCR); + smc_outb(TCR_CLEAR, ioaddr, TCR); #if 0 /* finally, shut the chip down */ - SMC_SELECT_BANK( 1 ); - outw( inw( ioaddr + CONTROL ), CTL_POWERDOWN, ioaddr + CONTROL ); + SMC_SELECT_BANK(1); + smc_outw(smc_inw(ioaddr, CONTROL), CTL_POWERDOWN, ioaddr, CONTROL); #endif } /* - . Function: smc_setmulticast( int ioaddr, int count, dev_mc_list * adds ) + . Function: smc_setmulticast(int ioaddr, int count, dev_mc_list * adds) . Purpose: . This sets the internal hardware table to filter out unwanted multicast . packets before they take up memory. @@ -411,26 +685,28 @@ */ -static void smc_setmulticast( int ioaddr, int count, struct dev_mc_list * addrs ) { +static void smc_setmulticast(struct net_device *dev, int count, struct dev_mc_list * addrs) +{ + u_int ioaddr = dev->base_addr; int i; - unsigned char multicast_table[ 8 ]; + unsigned char multicast_table[8]; struct dev_mc_list * cur_addr; /* table for flipping the order of 3 bits */ unsigned char invert3[] = { 0, 4, 2, 6, 1, 5, 3, 7 }; /* start with a table of all zeros: reject all */ - memset( multicast_table, 0, sizeof( multicast_table ) ); + memset(multicast_table, 0, sizeof(multicast_table)); cur_addr = addrs; - for ( i = 0; i < count ; i ++, cur_addr = cur_addr->next ) { + for (i = 0; i < count ; i ++, cur_addr = cur_addr->next) { int position; /* do we have a pointer here? */ - if ( !cur_addr ) + if (!cur_addr) break; /* make sure this is a multicast address - shouldn't this be a given if we have it here ? */ - if ( !( *cur_addr->dmi_addr & 1 ) ) + if (!(*cur_addr->dmi_addr & 1)) continue; /* only use the low order bits */ @@ -442,15 +718,15 @@ } /* now, the table can be loaded into the chipset */ - SMC_SELECT_BANK( 3 ); + SMC_SELECT_BANK(3); - for ( i = 0; i < 8 ; i++ ) { - outb( multicast_table[i], ioaddr + MULTICAST1 + i ); + for (i = 0; i < 8 ; i++) { + smc_outb(multicast_table[i], ioaddr, MULTICAST1 + i); } } /* - . Function: smc_wait_to_send_packet( struct sk_buff * skb, struct net_device * ) + . Function: smc_wait_to_send_packet(struct sk_buff * skb, struct net_device *) . Purpose: . Attempt to allocate memory for a packet, if chip-memory is not . available, then tell the card to generate an interrupt when it @@ -465,10 +741,10 @@ . o (NO): Enable interrupts and let the interrupt handler deal with it. . o (YES):Send it now. */ -static int smc_wait_to_send_packet( struct sk_buff * skb, struct net_device * dev ) +static int smc_wait_to_send_packet(struct sk_buff * skb, struct net_device * dev) { struct smc_local *lp = (struct smc_local *)dev->priv; - unsigned short ioaddr = dev->base_addr; + u_int ioaddr = dev->base_addr; word length; unsigned short numPages; word time_out; @@ -477,28 +753,28 @@ /* Well, I want to send the packet.. but I don't know if I can send it right now... */ - if ( lp->saved_skb) { + if (lp->saved_skb) { /* THIS SHOULD NEVER HAPPEN. */ lp->stats.tx_aborted_errors++; - printk(CARDNAME": Bad Craziness - sent packet while busy.\n" ); + printk("%s: Bad Craziness - sent packet while busy.\n", + dev->name); return 1; } lp->saved_skb = skb; length = ETH_ZLEN < skb->len ? skb->len : ETH_ZLEN; - /* ** The MMU wants the number of pages to be the number of 256 bytes - ** 'pages', minus 1 ( since a packet can't ever have 0 pages :) ) + ** 'pages', minus 1 (since a packet can't ever have 0 pages :)) ** ** Pkt size for allocating is data length +6 (for additional status words, ** length and ctl!) If odd size last byte is included in this header. */ - numPages = ((length & 0xfffe) + 6) / 256; + numPages = ((length & 0xfffe) + 6) / 256; - if (numPages > 7 ) { - printk(CARDNAME": Far too big packet error. \n"); + if (numPages > 7) { + printk("%s: Far too big packet error.\n", dev->name); /* freeing the packet is a good thing here... but should . any packets of this size get down here? */ dev_kfree_skb (skb); @@ -507,12 +783,13 @@ netif_wake_queue(dev); return 0; } + /* either way, a packet is waiting now */ lp->packets_waiting++; /* now, try to allocate the memory */ - SMC_SELECT_BANK( 2 ); - outw( MC_ALLOC | numPages, ioaddr + MMU_CMD ); + SMC_SELECT_BANK(2); + smc_outw(MC_ALLOC | numPages, ioaddr, MMU_CMD); /* . Performance Hack . @@ -529,21 +806,21 @@ do { word status; - status = inb( ioaddr + INTERRUPT ); - if ( status & IM_ALLOC_INT ) { + status = smc_inb(ioaddr, INTERRUPT); + if (status & IM_ALLOC_INT) { /* acknowledge the interrupt */ - outb( IM_ALLOC_INT, ioaddr + INTERRUPT ); - break; + smc_outb(IM_ALLOC_INT, ioaddr, INTERRUPT); + break; } - } while ( -- time_out ); + } while (-- time_out); - if ( !time_out ) { + if (!time_out) { /* oh well, wait until the chip finds memory later */ - SMC_ENABLE_INT( IM_ALLOC_INT ); - PRINTK2((CARDNAME": memory allocation deferred. \n")); + SMC_ENABLE_INT(IM_ALLOC_INT); + PRINTK2(("%s: memory allocation deferred.\n", dev->name)); /* it's deferred, but I'll handle it later */ - return 0; - } + return 0; + } /* or YES! I can send the packet now.. */ smc_hardware_send_packet(dev); netif_wake_queue(dev); @@ -551,46 +828,46 @@ } /* - . Function: smc_hardware_send_packet(struct net_device * ) + . Function: smc_hardware_send_packet(struct net_device *) . Purpose: . This sends the actual packet to the SMC9xxx chip. . . Algorithm: . First, see if a saved_skb is available. - . ( this should NOT be called if there is no 'saved_skb' + . (this should NOT be called if there is no 'saved_skb' . Now, find the packet number that the chip allocated . Point the data pointers at it in memory . Set the length word in the chip's memory . Dump the packet to chip memory - . Check if a last byte is needed ( odd length packet ) + . Check if a last byte is needed (odd length packet) . if so, set the control flag right . Tell the card to send it . Enable the transmit interrupt, so I know if it failed . Free the kernel data if I actually sent it. */ -static void smc_hardware_send_packet( struct net_device * dev ) +static void smc_hardware_send_packet(struct net_device *dev) { struct smc_local *lp = (struct smc_local *)dev->priv; - byte packet_no; - struct sk_buff * skb = lp->saved_skb; - word length; - unsigned short ioaddr; - byte * buf; - - ioaddr = dev->base_addr; + struct sk_buff *skb = lp->saved_skb; + word length, lastword; + u_int ioaddr = dev->base_addr; + byte packet_no; + byte *buf; - if ( !skb ) { - PRINTK((CARDNAME": In XMIT with no packet to send \n")); + if (!skb) { + PRINTK(("%s: In XMIT with no packet to send\n", dev->name)); return; } + length = ETH_ZLEN < skb->len ? skb->len : ETH_ZLEN; buf = skb->data; /* If I get here, I _know_ there is a packet slot waiting for me */ - packet_no = inb( ioaddr + PNR_ARR + 1 ); - if ( packet_no & 0x80 ) { + packet_no = smc_inb(ioaddr, PNR_ARR + 1); + if (packet_no & 0x80) { /* or isn't there? BAD CHIP! */ - printk(KERN_DEBUG CARDNAME": Memory allocation failed. \n"); + printk(KERN_DEBUG "%s: Memory allocation failed.\n", + dev->name); dev_kfree_skb_any(skb); lp->saved_skb = NULL; netif_wake_queue(dev); @@ -598,26 +875,19 @@ } /* we have a packet address, so tell the card to use it */ - outb( packet_no, ioaddr + PNR_ARR ); + smc_outb(packet_no, ioaddr, PNR_ARR); /* point to the beginning of the packet */ - outw( PTR_AUTOINC , ioaddr + POINTER ); + smc_outw(PTR_AUTOINC, ioaddr, POINTER); - PRINTK3((CARDNAME": Trying to xmit packet of length %x\n", length )); -#if SMC_DEBUG > 2 - print_packet( buf, length ); -#endif + PRINTK3(("%s: Trying to xmit packet of length %x\n", + dev->name, length)); - /* send the packet length ( +6 for status, length and ctl byte ) - and the status word ( set to zeros ) */ -#ifdef USE_32_BIT - outl( (length +6 ) << 16 , ioaddr + DATA_1 ); -#else - outw( 0, ioaddr + DATA_1 ); - /* send the packet length ( +6 for status words, length, and ctl*/ - outb( (length+6) & 0xFF,ioaddr + DATA_1 ); - outb( (length+6) >> 8 , ioaddr + DATA_1 ); -#endif + print_packet(buf, length); + + /* send the packet length (+6 for status, length and ctl byte) + and the status word (set to zeros) */ + smc_outl((length + 6) << 16, ioaddr, DATA_1); /* send the actual data . I _think_ it's faster to send the longs first, and then @@ -626,32 +896,22 @@ . a good idea to check which is optimal? But that could take . almost as much time as is saved? */ -#ifdef USE_32_BIT - if ( length & 0x2 ) { - outsl(ioaddr + DATA_1, buf, length >> 2 ); - outw( *((word *)(buf + (length & 0xFFFFFFFC))),ioaddr +DATA_1); - } - else - outsl(ioaddr + DATA_1, buf, length >> 2 ); -#else - outsw(ioaddr + DATA_1 , buf, (length ) >> 1); -#endif - /* Send the last byte, if there is one. */ + smc_outs(ioaddr, DATA_1, buf, length); - if ( (length & 1) == 0 ) { - outw( 0, ioaddr + DATA_1 ); - } else { - outb( buf[length -1 ], ioaddr + DATA_1 ); - outb( 0x20, ioaddr + DATA_1); - } + /* Send the last byte, if there is one. */ + if ((length & 1) == 0) + lastword = 0; + else + lastword = 0x2000 | buf[length - 1]; + smc_outw(lastword, ioaddr, DATA_1); /* enable the interrupts */ - SMC_ENABLE_INT( (IM_TX_INT | IM_TX_EMPTY_INT) ); + SMC_ENABLE_INT(IM_TX_INT | IM_TX_EMPTY_INT); /* and let the chipset deal with it */ - outw( MC_ENQUEUE , ioaddr + MMU_CMD ); + smc_outw(MC_ENQUEUE, ioaddr, MMU_CMD); - PRINTK2((CARDNAME": Sent packet of length %d \n",length)); + PRINTK2(("%s: Sent packet of length %d\n", dev->name, length)); lp->saved_skb = NULL; dev_kfree_skb_any (skb); @@ -666,7 +926,7 @@ /*------------------------------------------------------------------------- | - | smc_init( struct net_device * dev ) + | smc_init(struct net_device * dev) | Input parameters: | dev->base_addr == 0, try to find all possible locations | dev->base_addr == 1, return failure code @@ -681,6 +941,65 @@ */ int __init smc_init(struct net_device *dev) { + int ret = -ENODEV; +#if defined(CONFIG_ASSABET_NEPONSET) + if (machine_is_assabet() && machine_has_neponset()) { + unsigned int *addr; + unsigned char ecor; + unsigned long flags; + + NCR_0 |= NCR_ENET_OSC_EN; + dev->irq = IRQ_NEPONSET_SMC9196; + + /* + * Map the attribute space. This is overkill, but clean. + */ + addr = ioremap(0x18000000 + (1 << 25), 64 * 1024 * 4); + if (!addr) + return -ENOMEM; + + /* + * Reset the device. We must disable IRQs around this. + */ + local_irq_save(flags); + ecor = readl(addr + ECOR) & ~ECOR_RESET; + writel(ecor | ECOR_RESET, addr + ECOR); + udelay(100); + + /* + * The device will ignore all writes to the enable bit while + * reset is asserted, even if the reset bit is cleared in the + * same write. Must clear reset first, then enable the device. + */ + writel(ecor, addr + ECOR); + writel(ecor | ECOR_ENABLE, addr + ECOR); + + /* + * Force byte mode. + */ + writel(readl(addr + ECSR) | ECSR_IOIS8, addr + ECSR); + local_irq_restore(flags); + + iounmap(addr); + + /* + * Wait for the chip to wake up. + */ + mdelay(1); + + /* + * Map the real registers. + */ + addr = ioremap(0x18000000, 8 * 1024); + if (!addr) + return -ENOMEM; + + ret = smc_probe(dev, (int)addr); + if (ret) + iounmap(addr); + } + +#elif defined(CONFIG_ISA) int i; int base_addr = dev->base_addr; @@ -698,7 +1017,8 @@ return 0; /* couldn't find anything */ - return -ENODEV; +#endif + return ret; } /*---------------------------------------------------------------------- @@ -708,10 +1028,11 @@ . interrupt, so an auto-detect routine can detect it, and find the IRQ, ------------------------------------------------------------------------ */ -int __init smc_findirq( int ioaddr ) +int __init smc_findirq(struct net_device *dev) { int timeout = 20; unsigned long cookie; + u_int ioaddr = dev->base_addr; /* I have to do a STI() here, because this is called from @@ -727,26 +1048,25 @@ * when done. */ - + /* enable ALLOCation interrupts ONLY. */ SMC_SELECT_BANK(2); - /* enable ALLOCation interrupts ONLY */ - outb( IM_ALLOC_INT, ioaddr + INT_MASK ); + SMC_SET_INT(IM_ALLOC_INT); /* . Allocate 512 bytes of memory. Note that the chip was just . reset so all the memory is available */ - outw( MC_ALLOC | 1, ioaddr + MMU_CMD ); + smc_outw(MC_ALLOC | 1, ioaddr, MMU_CMD); /* . Wait until positive that the interrupt has been generated */ - while ( timeout ) { + while (timeout) { byte int_status; - int_status = inb( ioaddr + INTERRUPT ); + int_status = smc_inb(ioaddr, INTERRUPT); - if ( int_status & IM_ALLOC_INT ) + if (int_status & IM_ALLOC_INT) break; /* got the interrupt */ timeout--; } @@ -765,7 +1085,7 @@ SMC_DELAY(); /* and disable all interrupts again */ - outb( 0, ioaddr + INT_MASK ); + SMC_SET_INT(0); /* clear hardware interrupts again, because that's how it was when I was called... */ @@ -775,8 +1095,87 @@ return probe_irq_off(cookie); } +static int __init smc_probe_chip(struct net_device *dev, int ioaddr) +{ + unsigned int temp; + + /* First, see if the high byte is 0x33 */ + temp = smc_inw(ioaddr, BANK_SELECT); + if ((temp & 0xFF00) != 0x3300) + return -ENODEV; + + /* The above MIGHT indicate a device, but I need to write to further + test this. */ + smc_outw(0, ioaddr, BANK_SELECT); + temp = smc_inw(ioaddr, BANK_SELECT); + if ((temp & 0xFF00) != 0x3300) + return -ENODEV; + +#ifndef CONFIG_ASSABET_NEPONSET + /* well, we've already written once, so hopefully another time won't + hurt. This time, I need to switch the bank register to bank 1, + so I can access the base address register */ + SMC_SELECT_BANK(1); + temp = smc_inw(ioaddr, BASE); + if (ioaddr != (temp >> 3 & 0x3E0)) { + printk("%s: IOADDR %x doesn't match configuration (%x)." + "Probably not a SMC chip\n", dev->name, + ioaddr, (base_address_register >> 3) & 0x3E0); + /* well, the base address register didn't match. Must not have + been a SMC chip after all. */ + return -ENODEV; + } +#endif + + return 0; +} + +/* + . If dev->irq is 0, then the device has to be banged on to see + . what the IRQ is. + . + . This banging doesn't always detect the IRQ, for unknown reasons. + . a workaround is to reset the chip and try again. + . + . Interestingly, the DOS packet driver *SETS* the IRQ on the card to + . be what is requested on the command line. I don't do that, mostly + . because the card that I have uses a non-standard method of accessing + . the IRQs, and because this _should_ work in most configurations. + . + . Specifying an IRQ is done with the assumption that the user knows + . what (s)he is doing. No checking is done!!!! + . +*/ +static int __init smc_probe_irq(struct net_device *dev) +{ + if (dev->irq < 2) { + int trials; + + trials = 3; + while (trials--) { + dev->irq = smc_findirq(dev); + if (dev->irq) + break; + /* kick the card and try again */ + smc_reset(dev); + } + } + if (dev->irq == 0) { + printk("%s: Couldn't autodetect your IRQ. Use irq=xx.\n", + dev->name); + return -ENODEV; + } + + /* + * Some machines (eg, PCs) need to cannonicalize their IRQs. + */ + dev->irq = irq_cannonicalize(dev->irq); + + return 0; +} + /*---------------------------------------------------------------------- - . Function: smc_probe( int ioaddr ) + . Function: smc_probe(struct net_device *dev, int ioaddr) . . Purpose: . Tests to see if a given ioaddr points to an SMC9xxx chip. @@ -806,16 +1205,14 @@ */ static int __init smc_probe(struct net_device *dev, int ioaddr) { + struct smc_local *smc; int i, memory, retval; static unsigned version_printed; - unsigned int bank; const char *version_string; - const char *if_string; /* registers */ word revision_register; - word base_address_register; word configuration_register; word memory_info_register; word memory_cfg_register; @@ -824,44 +1221,24 @@ if (!request_region(ioaddr, SMC_IO_EXTENT, dev->name)) return -EBUSY; - /* First, see if the high byte is 0x33 */ - bank = inw( ioaddr + BANK_SELECT ); - if ( (bank & 0xFF00) != 0x3300 ) { - retval = -ENODEV; - goto err_out; - } - /* The above MIGHT indicate a device, but I need to write to further - test this. */ - outw( 0x0, ioaddr + BANK_SELECT ); - bank = inw( ioaddr + BANK_SELECT ); - if ( (bank & 0xFF00 ) != 0x3300 ) { - retval = -ENODEV; - goto err_out; - } - /* well, we've already written once, so hopefully another time won't - hurt. This time, I need to switch the bank register to bank 1, - so I can access the base address register */ - SMC_SELECT_BANK(1); - base_address_register = inw( ioaddr + BASE ); - if ( ioaddr != ( base_address_register >> 3 & 0x3E0 ) ) { - printk(CARDNAME ": IOADDR %x doesn't match configuration (%x)." - "Probably not a SMC chip\n", - ioaddr, base_address_register >> 3 & 0x3E0 ); - /* well, the base address register didn't match. Must not have - been a SMC chip after all. */ - retval = -ENODEV; + /* + * Do the basic probes. + */ + retval = smc_probe_chip(dev, ioaddr); + if (retval) goto err_out; - } /* check if the revision register is something that I recognize. These might need to be added to later, as future revisions could be added. */ SMC_SELECT_BANK(3); - revision_register = inw( ioaddr + REVISION ); - if ( !chip_ids[ ( revision_register >> 4 ) & 0xF ] ) { + revision_register = smc_inw(ioaddr, REVISION); + version_string = chip_ids[(revision_register >> 4) & 15]; + if (!version_string) { /* I don't recognize this chip, so... */ - printk(CARDNAME ": IO %x: Unrecognized revision register:" - " %x, Contact author. \n", ioaddr, revision_register ); + printk("%s: IO %x: unrecognized revision register: %x, " + "contact author.\n", dev->name, ioaddr, + revision_register); retval = -ENODEV; goto err_out; @@ -872,138 +1249,122 @@ against the hardware address, or do some other tests. */ if (version_printed++ == 0) - printk("%s", version); + printk(KERN_INFO "%s", version); /* fill in some of the fields */ dev->base_addr = ioaddr; /* - . Get the MAC address ( bank 1, regs 4 - 9 ) + . Get the MAC address (bank 1, regs 4 - 9) */ - SMC_SELECT_BANK( 1 ); - for ( i = 0; i < 6; i += 2 ) { + SMC_SELECT_BANK(1); + for (i = 0; i < 6; i += 2) { word address; - address = inw( ioaddr + ADDR0 + i ); - dev->dev_addr[ i + 1] = address >> 8; - dev->dev_addr[ i ] = address & 0xFF; + address = smc_inw(ioaddr, ADDR0 + i); + dev->dev_addr[i + 1] = address >> 8; + dev->dev_addr[i] = address & 0xFF; } + if (!is_valid_ether_addr(dev->dev_addr)) + printk("%s: Invalid ethernet MAC address. Please set using " + "ifconfig\n", dev->name); + /* get the memory information */ - SMC_SELECT_BANK( 0 ); - memory_info_register = inw( ioaddr + MIR ); - memory_cfg_register = inw( ioaddr + MCR ); - memory = ( memory_cfg_register >> 9 ) & 0x7; /* multiplier */ - memory *= 256 * ( memory_info_register & 0xFF ); + SMC_SELECT_BANK(0); + memory_info_register = smc_inw(ioaddr, MIR); + memory_cfg_register = smc_inw(ioaddr, MCR); + memory = (memory_cfg_register >> 9) & 0x7; /* multiplier */ + memory *= 256 * (memory_info_register & 0xFF); + + /* now, reset the chip, and put it into a known state */ + smc_reset(dev); /* - Now, I want to find out more about the chip. This is sort of - redundant, but it's cleaner to have it in both, rather than having - one VERY long probe procedure. - */ - SMC_SELECT_BANK(3); - revision_register = inw( ioaddr + REVISION ); - version_string = chip_ids[ ( revision_register >> 4 ) & 0xF ]; - if ( !version_string ) { - /* I shouldn't get here because this call was done before.... */ - retval = -ENODEV; + * Ok, now that we have everything in a + * sane state, probe for the interrupt. + */ + retval = smc_probe_irq(dev); + if (retval) goto err_out; - } - /* is it using AUI or 10BaseT ? */ - if ( dev->if_port == 0 ) { - SMC_SELECT_BANK(1); - configuration_register = inw( ioaddr + CONFIG ); - if ( configuration_register & CFG_AUI_SELECT ) - dev->if_port = 2; - else - dev->if_port = 1; + /* Initialize the private structure. */ + if (dev->priv == NULL) { + dev->priv = kmalloc(sizeof(struct smc_local), GFP_KERNEL); + if (dev->priv == NULL) { + retval = -ENOMEM; + goto err_out; + } } - if_string = interfaces[ dev->if_port - 1 ]; - /* now, reset the chip, and put it into a known state */ - smc_reset( ioaddr ); + smc = dev->priv; + + /* set the private data to zero by default */ + memset(smc, 0, sizeof(struct smc_local)); /* - . If dev->irq is 0, then the device has to be banged on to see - . what the IRQ is. - . - . This banging doesn't always detect the IRQ, for unknown reasons. - . a workaround is to reset the chip and try again. - . - . Interestingly, the DOS packet driver *SETS* the IRQ on the card to - . be what is requested on the command line. I don't do that, mostly - . because the card that I have uses a non-standard method of accessing - . the IRQs, and because this _should_ work in most configurations. - . - . Specifying an IRQ is done with the assumption that the user knows - . what (s)he is doing. No checking is done!!!! - . - */ - if ( dev->irq < 2 ) { - int trials; + * Get the interface characteristics. + * is it using AUI or 10BaseT ? + */ + switch (dev->if_port) { + case IF_PORT_10BASET: + smc->port = PORT_TP; + break; + + case IF_PORT_AUI: + smc->port = PORT_AUI; + break; - trials = 3; - while ( trials-- ) { - dev->irq = smc_findirq( ioaddr ); - if ( dev->irq ) - break; - /* kick the card and try again */ - smc_reset( ioaddr ); + default: + SMC_SELECT_BANK(1); + configuration_register = smc_inw(ioaddr, CONFIG); + if (configuration_register & CFG_AUI_SELECT) { + dev->if_port = IF_PORT_AUI; + smc->port = PORT_AUI; + } else { + dev->if_port = IF_PORT_10BASET; + smc->port = PORT_TP; } - } - if (dev->irq == 0 ) { - printk(CARDNAME": Couldn't autodetect your IRQ. Use irq=xx.\n"); - retval = -ENODEV; - goto err_out; + break; } - /* now, print out the card info, in a short format.. */ + /* all interfaces are half-duplex by default */ + smc->duplex = DUPLEX_HALF; - printk("%s: %s(r:%d) at %#3x IRQ:%d INTF:%s MEM:%db ", dev->name, - version_string, revision_register & 0xF, ioaddr, dev->irq, - if_string, memory ); + /* now, print out the card info, in a short format.. */ + printk("%s: %s (rev %d) at %#3x IRQ:%d INTF:%s MEM:%db ", dev->name, + version_string, revision_register & 15, ioaddr, dev->irq, + interfaces[smc->port], memory); /* . Print the Ethernet address */ printk("ADDR: "); for (i = 0; i < 5; i++) - printk("%2.2x:", dev->dev_addr[i] ); - printk("%2.2x \n", dev->dev_addr[5] ); - - - /* Initialize the private structure. */ - if (dev->priv == NULL) { - dev->priv = kmalloc(sizeof(struct smc_local), GFP_KERNEL); - if (dev->priv == NULL) { - retval = -ENOMEM; - goto err_out; - } - } - /* set the private data to zero by default */ - memset(dev->priv, 0, sizeof(struct smc_local)); + printk("%2.2x:", dev->dev_addr[i]); + printk("%2.2x\n", dev->dev_addr[5]); /* Fill in the fields of the device structure with ethernet values. */ ether_setup(dev); /* Grab the IRQ */ - retval = request_irq(dev->irq, &smc_interrupt, 0, dev->name, dev); - if (retval) { + retval = request_irq(dev->irq, &smc_interrupt, 0, dev->name, dev); + if (retval) { printk("%s: unable to get IRQ %d (irqval=%d).\n", dev->name, dev->irq, retval); kfree(dev->priv); dev->priv = NULL; - goto err_out; - } + goto err_out; + } - dev->open = smc_open; - dev->stop = smc_close; - dev->hard_start_xmit = smc_wait_to_send_packet; - dev->tx_timeout = smc_timeout; - dev->watchdog_timeo = HZ/20; - dev->get_stats = smc_query_statistics; - dev->set_multicast_list = smc_set_multicast_list; + dev->open = smc_open; + dev->stop = smc_close; + dev->hard_start_xmit = smc_wait_to_send_packet; + dev->tx_timeout = smc_timeout; + dev->watchdog_timeo = HZ/20; + dev->get_stats = smc_query_statistics; + dev->set_multicast_list = smc_set_multicast_list; + dev->do_ioctl = smc_ioctl; return 0; @@ -1012,42 +1373,43 @@ return retval; } -#if SMC_DEBUG > 2 -static void print_packet( byte * buf, int length ) +/* + * This is responsible for setting the chip appropriately + * for the interface type. This should only be called while + * the interface is up and running. + */ +static void smc_set_port(struct net_device *dev) { -#if 0 - int i; - int remainder; - int lines; - - printk("Packet of length %d \n", length ); - lines = length / 16; - remainder = length % 16; + struct smc_local *smc = dev->priv; + u_int ioaddr = dev->base_addr; + u_int val; - for ( i = 0; i < lines ; i ++ ) { - int cur; - - for ( cur = 0; cur < 8; cur ++ ) { - byte a, b; - - a = *(buf ++ ); - b = *(buf ++ ); - printk("%02x%02x ", a, b ); - } - printk("\n"); + SMC_SELECT_BANK(1); + val = smc_inw(ioaddr, CONFIG); + switch (smc->port) { + case PORT_TP: + val &= ~CFG_AUI_SELECT; + break; + + case PORT_AUI: + val |= CFG_AUI_SELECT; + break; } - for ( i = 0; i < remainder/2 ; i++ ) { - byte a, b; + smc_outw(val, ioaddr, CONFIG); - a = *(buf ++ ); - b = *(buf ++ ); - printk("%02x%02x ", a, b ); + SMC_SELECT_BANK(0); + val = smc_inw(ioaddr, TCR); + switch (smc->duplex) { + case DUPLEX_HALF: + val &= ~TCR_FDSE; + break; + + case DUPLEX_FULL: + val |= TCR_FDSE; + break; } - printk("\n"); -#endif + smc_outw(val, ioaddr, TCR); } -#endif - /* * Open and Initialize the board @@ -1057,48 +1419,141 @@ */ static int smc_open(struct net_device *dev) { - int ioaddr = dev->base_addr; + struct smc_local *smc = dev->priv; + u_int ioaddr = dev->base_addr; + int i; - int i; /* used to set hw ethernet address */ + /* + * Check that the address is valid. If its not, refuse + * to bring the device up. The user must specify an + * address using ifconfig eth0 hw ether xx:xx:xx:xx:xx:xx + */ + if (!is_valid_ether_addr(dev->dev_addr)) + return -EINVAL; /* clear out all the junk that was put here before... */ - memset(dev->priv, 0, sizeof(struct smc_local)); + smc->saved_skb = NULL; + smc->packets_waiting = 0; /* reset the hardware */ - - smc_reset( ioaddr ); - smc_enable( ioaddr ); + smc_reset(dev); + smc_enable(dev); /* Select which interface to use */ - - SMC_SELECT_BANK( 1 ); - if ( dev->if_port == 1 ) { - outw( inw( ioaddr + CONFIG ) & ~CFG_AUI_SELECT, - ioaddr + CONFIG ); - } - else if ( dev->if_port == 2 ) { - outw( inw( ioaddr + CONFIG ) | CFG_AUI_SELECT, - ioaddr + CONFIG ); - } + smc_set_port(dev); /* - According to Becker, I have to set the hardware address + According to Becker, I have to set the hardware address at this point, because the (l)user can set it with an ioctl. Easily done... */ - SMC_SELECT_BANK( 1 ); - for ( i = 0; i < 6; i += 2 ) { + SMC_SELECT_BANK(1); + for (i = 0; i < 6; i += 2) { word address; - address = dev->dev_addr[ i + 1 ] << 8 ; - address |= dev->dev_addr[ i ]; - outw( address, ioaddr + ADDR0 + i ); + address = dev->dev_addr[i + 1] << 8 ; + address |= dev->dev_addr[i]; + smc_outw(address, ioaddr, ADDR0 + i); } netif_start_queue(dev); return 0; } +/* + * This is our template. Fill the rest in at run-time + */ +static const struct ethtool_cmd ecmd_template = { + supported: SUPPORTED_10baseT_Half | + SUPPORTED_10baseT_Full | + SUPPORTED_TP | + SUPPORTED_AUI, + speed: SPEED_10, + autoneg: AUTONEG_DISABLE, + maxtxpkt: 1, + maxrxpkt: 1, + transceiver: XCVR_INTERNAL, +}; + +static int smc_ioctl(struct net_device *dev, struct ifreq *rq, int cmd) +{ + struct smc_local *smc = dev->priv; + u32 etcmd; + int ret = -EINVAL; + + if (cmd != SIOCETHTOOL) + return -EOPNOTSUPP; + + if (get_user(etcmd, (u32 *)rq->ifr_data)) + return -EFAULT; + + switch (etcmd) { + case ETHTOOL_GSET: { + struct ethtool_cmd ecmd = ecmd_template; + + ecmd.cmd = etcmd; + ecmd.port = smc->port; + ecmd.duplex = smc->duplex; + + ret = copy_to_user(rq->ifr_data, &ecmd, sizeof(ecmd)) + ? -EFAULT : 0; + break; + } + + case ETHTOOL_SSET: { + struct ethtool_cmd ecmd; + + ret = -EPERM; + if (!capable(CAP_NET_ADMIN)) + break; + + ret = -EFAULT; + if (copy_from_user(&ecmd, rq->ifr_data, sizeof(ecmd))) + break; + + /* + * Sanity-check the arguments. + */ + ret = -EINVAL; + if (ecmd.autoneg != AUTONEG_DISABLE) + break; + if (ecmd.speed != SPEED_10) + break; + if (ecmd.duplex != DUPLEX_HALF && ecmd.duplex != DUPLEX_FULL) + break; + if (ecmd.port != PORT_TP && ecmd.port != PORT_AUI) + break; + + smc->port = ecmd.port; + smc->duplex = ecmd.duplex; + + if (netif_running(dev)) + smc_set_port(dev); + + ret = 0; + break; + } + + case ETHTOOL_GDRVINFO: { + struct ethtool_drvinfo edrv; + + memset(&edrv, 0, sizeof(edrv)); + + edrv.cmd = etcmd; + strcpy(edrv.driver, DRV_NAME); + strcpy(edrv.version, DRV_VERSION); + sprintf(edrv.bus_info, "ISA:%8.8lx:%d", + dev->base_addr, dev->irq); + + ret = copy_to_user(rq->ifr_data, &edrv, sizeof(edrv)) + ? -EFAULT : 0; + break; + } + } + + return ret; +} + /*-------------------------------------------------------- . Called by the kernel to send a packet out into the void . of the net. This routine is largely based on @@ -1110,12 +1565,10 @@ { /* If we get here, some higher level has decided we are broken. There should really be a "kick me" function call instead. */ - printk(KERN_WARNING CARDNAME": transmit timed out, %s?\n", - tx_done(dev) ? "IRQ conflict" : - "network cable problem"); + printk(KERN_WARNING "%s: transmit timed out\n", dev->name); /* "kick" the adaptor */ - smc_reset( dev->base_addr ); - smc_enable( dev->base_addr ); + smc_reset(dev); + smc_enable(dev); dev->trans_start = jiffies; /* clear anything saved */ ((struct smc_local *)dev->priv)->saved_skb = NULL; @@ -1135,10 +1588,10 @@ . ---------------------------------------------------------------------*/ -static void smc_interrupt(int irq, void * dev_id, struct pt_regs * regs) +static void smc_interrupt(int irq, void * dev_id, struct pt_regs * regs) { struct net_device *dev = dev_id; - int ioaddr = dev->base_addr; + u_int ioaddr = dev->base_addr; struct smc_local *lp = (struct smc_local *)dev->priv; byte status; @@ -1151,45 +1604,45 @@ - PRINTK3((CARDNAME": SMC interrupt started \n")); + PRINTK3(("%s: SMC interrupt started\n", dev->name)); - saved_bank = inw( ioaddr + BANK_SELECT ); + saved_bank = smc_inw(ioaddr, BANK_SELECT); SMC_SELECT_BANK(2); - saved_pointer = inw( ioaddr + POINTER ); + saved_pointer = smc_inw(ioaddr, POINTER); - mask = inb( ioaddr + INT_MASK ); + mask = smc_inb(ioaddr, INT_MASK); /* clear all interrupts */ - outb( 0, ioaddr + INT_MASK ); + SMC_SET_INT(0); /* set a timeout value, so I don't stay here forever */ timeout = 4; - PRINTK2((KERN_WARNING CARDNAME ": MASK IS %x \n", mask )); + PRINTK2((KERN_WARNING "%s: MASK IS %x\n", dev->name, mask)); do { /* read the status flag, and mask it */ - status = inb( ioaddr + INTERRUPT ) & mask; - if (!status ) + status = smc_inb(ioaddr, INTERRUPT) & mask; + if (!status) break; - PRINTK3((KERN_WARNING CARDNAME - ": Handling interrupt status %x \n", status )); + PRINTK3((KERN_WARNING "%s: handling interrupt status %x\n", + dev->name, status)); if (status & IM_RCV_INT) { /* Got a packet(s). */ - PRINTK2((KERN_WARNING CARDNAME - ": Receive Interrupt\n")); + PRINTK2((KERN_WARNING "%s: receive interrupt\n", + dev->name)); smc_rcv(dev); - } else if (status & IM_TX_INT ) { - PRINTK2((KERN_WARNING CARDNAME - ": TX ERROR handled\n")); + } else if (status & IM_TX_INT) { + PRINTK2((KERN_WARNING "%s: TX ERROR handled\n", + dev->name)); smc_tx(dev); - outb(IM_TX_INT, ioaddr + INTERRUPT ); - } else if (status & IM_TX_EMPTY_INT ) { + smc_outb(IM_TX_INT, ioaddr, INTERRUPT); + } else if (status & IM_TX_EMPTY_INT) { /* update stats */ - SMC_SELECT_BANK( 0 ); - card_stats = inw( ioaddr + COUNTER ); + SMC_SELECT_BANK(0); + card_stats = smc_inw(ioaddr, COUNTER); /* single collisions */ lp->stats.collisions += card_stats & 0xF; card_stats >>= 4; @@ -1198,60 +1651,63 @@ /* these are for when linux supports these statistics */ - SMC_SELECT_BANK( 2 ); - PRINTK2((KERN_WARNING CARDNAME - ": TX_BUFFER_EMPTY handled\n")); - outb( IM_TX_EMPTY_INT, ioaddr + INTERRUPT ); + SMC_SELECT_BANK(2); + PRINTK2((KERN_WARNING "%s: TX_BUFFER_EMPTY handled\n", + dev->name)); + smc_outb(IM_TX_EMPTY_INT, ioaddr, INTERRUPT); mask &= ~IM_TX_EMPTY_INT; lp->stats.tx_packets += lp->packets_waiting; lp->packets_waiting = 0; - } else if (status & IM_ALLOC_INT ) { - PRINTK2((KERN_DEBUG CARDNAME - ": Allocation interrupt \n")); + } else if (status & IM_ALLOC_INT) { + PRINTK2((KERN_DEBUG "%s: Allocation interrupt\n", + dev->name)); /* clear this interrupt so it doesn't happen again */ mask &= ~IM_ALLOC_INT; - smc_hardware_send_packet( dev ); + smc_hardware_send_packet(dev); /* enable xmit interrupts based on this */ - mask |= ( IM_TX_EMPTY_INT | IM_TX_INT ); + mask |= (IM_TX_EMPTY_INT | IM_TX_INT); /* and let the card send more packets to me */ netif_wake_queue(dev); - PRINTK2((CARDNAME": Handoff done successfully.\n")); - } else if (status & IM_RX_OVRN_INT ) { + PRINTK2(("%s: Handoff done successfully.\n", + dev->name)); + } else if (status & IM_RX_OVRN_INT) { lp->stats.rx_errors++; lp->stats.rx_fifo_errors++; - outb( IM_RX_OVRN_INT, ioaddr + INTERRUPT ); - } else if (status & IM_EPH_INT ) { - PRINTK((CARDNAME ": UNSUPPORTED: EPH INTERRUPT \n")); - } else if (status & IM_ERCV_INT ) { - PRINTK((CARDNAME ": UNSUPPORTED: ERCV INTERRUPT \n")); - outb( IM_ERCV_INT, ioaddr + INTERRUPT ); + smc_outb(IM_RX_OVRN_INT, ioaddr, INTERRUPT); + } else if (status & IM_EPH_INT) { + PRINTK(("%s: UNSUPPORTED: EPH INTERRUPT\n", + dev->name)); + } else if (status & IM_ERCV_INT) { + PRINTK(("%s: UNSUPPORTED: ERCV INTERRUPT\n", + dev->name)); + smc_outb(IM_ERCV_INT, ioaddr, INTERRUPT); } - } while ( timeout -- ); + } while (timeout --); /* restore state register */ - SMC_SELECT_BANK( 2 ); - outb( mask, ioaddr + INT_MASK ); + SMC_SELECT_BANK(2); + SMC_SET_INT(mask); - PRINTK3(( KERN_WARNING CARDNAME ": MASK is now %x \n", mask )); - outw( saved_pointer, ioaddr + POINTER ); + PRINTK3((KERN_WARNING "%s: MASK is now %x\n", dev->name, mask)); + smc_outw(saved_pointer, ioaddr, POINTER); - SMC_SELECT_BANK( saved_bank ); + SMC_SELECT_BANK(saved_bank); - PRINTK3((CARDNAME ": Interrupt done\n")); + PRINTK3(("%s: Interrupt done\n", dev->name)); return; } /*------------------------------------------------------------- . - . smc_rcv - receive a packet from the card + . smc_rcv - receive a packet from the card . - . There is ( at least ) a packet waiting to be read from + . There is (at least) a packet waiting to be read from . chip-memory. . . o Read the status @@ -1262,55 +1718,57 @@ static void smc_rcv(struct net_device *dev) { struct smc_local *lp = (struct smc_local *)dev->priv; - int ioaddr = dev->base_addr; + u_int ioaddr = dev->base_addr; int packet_number; word status; word packet_length; /* assume bank 2 */ - packet_number = inw( ioaddr + FIFO_PORTS ); + packet_number = smc_inw(ioaddr, FIFO_PORTS); - if ( packet_number & FP_RXEMPTY ) { + if (packet_number & FP_RXEMPTY) { /* we got called , but nothing was on the FIFO */ - PRINTK((CARDNAME ": WARNING: smc_rcv with nothing on FIFO. \n")); + PRINTK(("%s: WARNING: smc_rcv with nothing on FIFO.\n", + dev->name)); /* don't need to restore anything */ return; } /* start reading from the start of the packet */ - outw( PTR_READ | PTR_RCV | PTR_AUTOINC, ioaddr + POINTER ); + smc_outw(PTR_READ | PTR_RCV | PTR_AUTOINC, ioaddr, POINTER); /* First two words are status and packet_length */ - status = inw( ioaddr + DATA_1 ); - packet_length = inw( ioaddr + DATA_1 ); + status = smc_inw(ioaddr, DATA_1); + packet_length = smc_inw(ioaddr, DATA_1); packet_length &= 0x07ff; /* mask off top bits */ - PRINTK2(("RCV: STATUS %4x LENGTH %4x\n", status, packet_length )); + PRINTK2(("RCV: STATUS %4x LENGTH %4x\n", status, packet_length)); /* . the packet length contains 3 extra words : . status, length, and an extra word with an odd byte . */ packet_length -= 6; - if ( !(status & RS_ERRORS ) ){ + if (!(status & RS_ERRORS)){ /* do stuff to make a new packet */ struct sk_buff * skb; byte * data; /* read one extra byte */ - if ( status & RS_ODDFRAME ) + if (status & RS_ODDFRAME) packet_length++; /* set multicast stats */ - if ( status & RS_MULTICAST ) + if (status & RS_MULTICAST) lp->stats.multicast++; - skb = dev_alloc_skb( packet_length + 5); + skb = dev_alloc_skb(packet_length + 5); - if ( skb == NULL ) { - printk(KERN_NOTICE CARDNAME ": Low memory, packet dropped.\n"); + if (skb == NULL) { + printk(KERN_NOTICE "%s: Low memory, packet dropped.\n", + dev->name); lp->stats.rx_dropped++; goto done; } @@ -1320,36 +1778,15 @@ ! in the worse case */ - skb_reserve( skb, 2 ); /* 16 bit alignment */ + skb_reserve(skb, 2); /* 16 bit alignment */ skb->dev = dev; - data = skb_put( skb, packet_length); + data = skb_put(skb, packet_length); -#ifdef USE_32_BIT - /* QUESTION: Like in the TX routine, do I want - to send the DWORDs or the bytes first, or some - mixture. A mixture might improve already slow PIO - performance */ - PRINTK3((" Reading %d dwords (and %d bytes) \n", - packet_length >> 2, packet_length & 3 )); - insl(ioaddr + DATA_1 , data, packet_length >> 2 ); - /* read the left over bytes */ - insb( ioaddr + DATA_1, data + (packet_length & 0xFFFFFC), - packet_length & 0x3 ); -#else - PRINTK3((" Reading %d words and %d byte(s) \n", - (packet_length >> 1 ), packet_length & 1 )); - insw(ioaddr + DATA_1 , data, packet_length >> 1); - if ( packet_length & 1 ) { - data += packet_length & ~1; - *(data++) = inb( ioaddr + DATA_1 ); - } -#endif -#if SMC_DEBUG > 2 - print_packet( data, packet_length ); -#endif + smc_ins(ioaddr, DATA_1, data, packet_length); + print_packet(data, packet_length); - skb->protocol = eth_type_trans(skb, dev ); + skb->protocol = eth_type_trans(skb, dev); netif_rx(skb); dev->last_rx = jiffies; lp->stats.rx_packets++; @@ -1358,15 +1795,17 @@ /* error ... */ lp->stats.rx_errors++; - if ( status & RS_ALGNERR ) lp->stats.rx_frame_errors++; - if ( status & (RS_TOOSHORT | RS_TOOLONG ) ) + if (status & RS_ALGNERR) + lp->stats.rx_frame_errors++; + if (status & (RS_TOOSHORT | RS_TOOLONG)) lp->stats.rx_length_errors++; - if ( status & RS_BADCRC) lp->stats.rx_crc_errors++; + if (status & RS_BADCRC) + lp->stats.rx_crc_errors++; } done: /* error or good, tell the card to get rid of this packet */ - outw( MC_RELEASE, ioaddr + MMU_CMD ); + smc_outw(MC_RELEASE, ioaddr, MMU_CMD); } @@ -1379,62 +1818,64 @@ . Algorithm: . Save pointer and packet no . Get the packet no from the top of the queue - . check if it's valid ( if not, is this an error??? ) + . check if it's valid (if not, is this an error???) . read the status word . record the error - . ( resend? Not really, since we don't want old packets around ) + . (resend? Not really, since we don't want old packets around) . Restore saved values ************************************************************************/ -static void smc_tx( struct net_device * dev ) +static void smc_tx(struct net_device * dev) { - int ioaddr = dev->base_addr; + u_int ioaddr = dev->base_addr; struct smc_local *lp = (struct smc_local *)dev->priv; byte saved_packet; byte packet_no; word tx_status; - /* assume bank 2 */ + /* assume bank 2 */ - saved_packet = inb( ioaddr + PNR_ARR ); - packet_no = inw( ioaddr + FIFO_PORTS ); + saved_packet = smc_inb(ioaddr, PNR_ARR); + packet_no = smc_inw(ioaddr, FIFO_PORTS); packet_no &= 0x7F; /* select this as the packet to read from */ - outb( packet_no, ioaddr + PNR_ARR ); + smc_outb(packet_no, ioaddr, PNR_ARR); /* read the first word from this packet */ - outw( PTR_AUTOINC | PTR_READ, ioaddr + POINTER ); + smc_outw(PTR_AUTOINC | PTR_READ, ioaddr, POINTER); - tx_status = inw( ioaddr + DATA_1 ); - PRINTK3((CARDNAME": TX DONE STATUS: %4x \n", tx_status )); + tx_status = smc_inw(ioaddr, DATA_1); + PRINTK3(("%s: TX DONE STATUS: %4x\n", dev->name, tx_status)); lp->stats.tx_errors++; - if ( tx_status & TS_LOSTCAR ) lp->stats.tx_carrier_errors++; - if ( tx_status & TS_LATCOL ) { - printk(KERN_DEBUG CARDNAME - ": Late collision occurred on last xmit.\n"); + if (tx_status & TS_LOSTCAR) + lp->stats.tx_carrier_errors++; + if (tx_status & TS_LATCOL) { + printk(KERN_DEBUG "%s: Late collision occurred on " + "last xmit.\n", dev->name); lp->stats.tx_window_errors++; } #if 0 - if ( tx_status & TS_16COL ) { ... } + if (tx_status & TS_16COL) { ... } #endif - if ( tx_status & TS_SUCCESS ) { - printk(CARDNAME": Successful packet caused interrupt \n"); + if (tx_status & TS_SUCCESS) { + printk("%s: Successful packet caused interrupt\n", + dev->name); } /* re-enable transmit */ - SMC_SELECT_BANK( 0 ); - outw( inw( ioaddr + TCR ) | TCR_ENABLE, ioaddr + TCR ); + SMC_SELECT_BANK(0); + smc_outw(smc_inw(ioaddr, TCR) | TCR_ENABLE, ioaddr, TCR); /* kill the packet */ - SMC_SELECT_BANK( 2 ); - outw( MC_FREEPKT, ioaddr + MMU_CMD ); + SMC_SELECT_BANK(2); + smc_outw(MC_FREEPKT, ioaddr, MMU_CMD); /* one less packet waiting for me */ lp->packets_waiting--; - outb( saved_packet, ioaddr + PNR_ARR ); + smc_outb(saved_packet, ioaddr, PNR_ARR); return; } @@ -1450,7 +1891,7 @@ { netif_stop_queue(dev); /* clear everything */ - smc_shutdown( dev->base_addr ); + smc_shutdown(dev); /* Update the statistics here. */ return 0; @@ -1471,16 +1912,16 @@ . . This routine will, depending on the values passed to it, . either make it accept multicast packets, go into - . promiscuous mode ( for TCPDUMP and cousins ) or accept + . promiscuous mode (for TCPDUMP and cousins) or accept . a select set of multicast packets */ static void smc_set_multicast_list(struct net_device *dev) { - short ioaddr = dev->base_addr; + u_int ioaddr = dev->base_addr; SMC_SELECT_BANK(0); - if ( dev->flags & IFF_PROMISC ) - outw( inw(ioaddr + RCR ) | RCR_PROMISC, ioaddr + RCR ); + if (dev->flags & IFF_PROMISC) + smc_outw(smc_inw(ioaddr, RCR) | RCR_PROMISC, ioaddr, RCR); /* BUG? I never disable promiscuous mode if multicasting was turned on. Now, I turn off promiscuous mode, but I don't do anything to multicasting @@ -1492,34 +1933,34 @@ checked before the table is */ else if (dev->flags & IFF_ALLMULTI) - outw( inw(ioaddr + RCR ) | RCR_ALMUL, ioaddr + RCR ); + smc_outw(smc_inw(ioaddr, RCR) | RCR_ALMUL, ioaddr, RCR); /* We just get all multicast packets even if we only want them . from one source. This will be changed at some future . point. */ - else if (dev->mc_count ) { + else if (dev->mc_count) { /* support hardware multicasting */ /* be sure I get rid of flags I might have set */ - outw( inw( ioaddr + RCR ) & ~(RCR_PROMISC | RCR_ALMUL), - ioaddr + RCR ); + smc_outw(smc_inw(ioaddr, RCR) & ~(RCR_PROMISC | RCR_ALMUL), + ioaddr, RCR); /* NOTE: this has to set the bank, so make sure it is the last thing called. The bank is set to zero at the top */ - smc_setmulticast( ioaddr, dev->mc_count, dev->mc_list ); + smc_setmulticast(dev, dev->mc_count, dev->mc_list); } - else { - outw( inw( ioaddr + RCR ) & ~(RCR_PROMISC | RCR_ALMUL), - ioaddr + RCR ); + else { + smc_outw(smc_inw(ioaddr, RCR) & ~(RCR_PROMISC | RCR_ALMUL), + ioaddr, RCR); /* since I'm disabling all multicast entirely, I need to clear the multicast list */ - SMC_SELECT_BANK( 3 ); - outw( 0, ioaddr + MULTICAST1 ); - outw( 0, ioaddr + MULTICAST2 ); - outw( 0, ioaddr + MULTICAST3 ); - outw( 0, ioaddr + MULTICAST4 ); + SMC_SELECT_BANK(3); + smc_outw(0, ioaddr, MULTICAST1); + smc_outw(0, ioaddr, MULTICAST2); + smc_outw(0, ioaddr, MULTICAST3); + smc_outw(0, ioaddr, MULTICAST4); } } @@ -1540,21 +1981,26 @@ int init_module(void) { - int result; - if (io == 0) - printk(KERN_WARNING - CARDNAME": You shouldn't use auto-probing with insmod!\n" ); + printk(KERN_WARNING CARDNAME + ": You shouldn't use auto-probing with insmod!\n"); + + /* + * Note: dev->if_port has changed to be 2.4 compliant. + * We keep the ifport insmod parameter the same though. + */ + switch (ifport) { + case 1: devSMC9194.if_port = IF_PORT_10BASET; break; + case 2: devSMC9194.if_port = IF_PORT_AUI; break; + default: devSMC9194.if_port = 0; break; + } /* copy the parameters from insmod into the device structure */ devSMC9194.base_addr = io; devSMC9194.irq = irq; - devSMC9194.if_port = ifport; - devSMC9194.init = smc_init; - if ((result = register_netdev(&devSMC9194)) != 0) - return result; + devSMC9194.init = smc_init; - return 0; + return register_netdev(&devSMC9194); } void cleanup_module(void) diff -urN orig/drivers/net/smc9194.h linux/drivers/net/smc9194.h --- orig/drivers/net/smc9194.h Mon Oct 1 23:10:57 2001 +++ linux/drivers/net/smc9194.h Fri Sep 28 20:42:28 2001 @@ -63,10 +63,11 @@ #define TCR 0 /* transmit control register */ #define TCR_ENABLE 0x0001 /* if this is 1, we can transmit */ +#define TCR_PAD_ENABLE 0x0080 /* pads short packets to 64 bytes */ +#define TCR_MON_CNS 0x0400 /* monitors the carrier status */ #define TCR_FDUPLX 0x0800 /* receive packets sent out */ #define TCR_STP_SQET 0x1000 /* stop transmitting if Signal quality error */ -#define TCR_MON_CNS 0x0400 /* monitors the carrier status */ -#define TCR_PAD_ENABLE 0x0080 /* pads short packets to 64 bytes */ +#define TCR_FDSE 0x8000 /* full duplex, switched ethernet */ #define TCR_CLEAR 0 /* do NOTHING */ /* the normal settings for the TCR register : */ @@ -107,7 +108,10 @@ #define CTL_CR_ENABLE 0x40 #define CTL_TE_ENABLE 0x0020 #define CTL_AUTO_RELEASE 0x0800 -#define CTL_EPROM_ACCESS 0x0003 /* high if Eprom is being read */ +#define CTL_EPROM_SELECT 0x0004 +#define CTL_EPROM_RELOAD 0x0002 +#define CTL_EPROM_STORE 0x0001 +#define CTL_EPROM_ACCESS (CTL_EPROM_RELOAD | CTL_EPROM_STORE) /* high if Eprom is being read */ /* BANK 2 */ #define MMU_CMD 0 @@ -130,7 +134,6 @@ #define PTR_READ 0x2000 #define PTR_RCV 0x8000 #define PTR_AUTOINC 0x4000 -#define PTR_AUTO_INC 0x0040 #define DATA_1 8 #define DATA_2 10 @@ -162,17 +165,6 @@ #define CHIP_9195 5 #define CHIP_91100 7 -static const char * chip_ids[ 15 ] = { - NULL, NULL, NULL, - /* 3 */ "SMC91C90/91C92", - /* 4 */ "SMC91C94", - /* 5 */ "SMC91C95", - NULL, - /* 7 */ "SMC91C100", - /* 8 */ "SMC91C100FD", - NULL, NULL, NULL, - NULL, NULL, NULL}; - /* . Transmit status bits */ @@ -192,40 +184,20 @@ #define RS_MULTICAST 0x0001 #define RS_ERRORS (RS_ALGNERR | RS_BADCRC | RS_TOOLONG | RS_TOOSHORT) -static const char * interfaces[ 2 ] = { "TP", "AUI" }; - -/*------------------------------------------------------------------------- - . I define some macros to make it easier to do somewhat common - . or slightly complicated, repeated tasks. - --------------------------------------------------------------------------*/ - -/* select a register bank, 0 to 3 */ - -#define SMC_SELECT_BANK(x) { outw( x, ioaddr + BANK_SELECT ); } - -/* define a small delay for the reset */ -#define SMC_DELAY() { inw( ioaddr + RCR );\ - inw( ioaddr + RCR );\ - inw( ioaddr + RCR ); } - -/* this enables an interrupt in the interrupt mask register */ -#define SMC_ENABLE_INT(x) {\ - unsigned char mask;\ - SMC_SELECT_BANK(2);\ - mask = inb( ioaddr + INT_MASK );\ - mask |= (x);\ - outb( mask, ioaddr + INT_MASK ); \ -} - -/* this disables an interrupt from the interrupt mask register */ - -#define SMC_DISABLE_INT(x) {\ - unsigned char mask;\ - SMC_SELECT_BANK(2);\ - mask = inb( ioaddr + INT_MASK );\ - mask &= ~(x);\ - outb( mask, ioaddr + INT_MASK ); \ -} +/* + * SMC91C96 ethernet config and status registers. + * These are in the "attribute" space. + */ +#define ECOR 0x8000 +#define ECOR_RESET 0x80 +#define ECOR_LEVEL_IRQ 0x40 +#define ECOR_WR_ATTRIB 0x04 +#define ECOR_ENABLE 0x01 + +#define ECSR 0x8002 +#define ECSR_IOIS8 0x20 +#define ECSR_PWRDWN 0x04 +#define ECSR_INT 0x02 /*---------------------------------------------------------------------- . Define the interrupts that I want to receive from the card @@ -236,6 +208,37 @@ . IM_RX_OVRN_INT, because I have to kick the receiver --------------------------------------------------------------------------*/ #define SMC_INTERRUPT_MASK (IM_EPH_INT | IM_RX_OVRN_INT | IM_RCV_INT) + +/* store this information for the driver.. */ +struct smc_local { + /* + these are things that the kernel wants me to keep, so users + can find out semi-useless statistics of how well the card is + performing + */ + struct net_device_stats stats; + + /* + If I have to wait until memory is available to send + a packet, I will store the skbuff here, until I get the + desired memory. Then, I'll send it out and free it. + */ + struct sk_buff * saved_skb; + + /* + . This keeps track of how many packets that I have + . sent out. When an TX_EMPTY interrupt comes, I know + . that all of these have been sent. + */ + int packets_waiting; + + /* + . Interface status. These correspond to the parameters + . in the ethtool_cmd structure. + */ + u8 duplex; + u8 port; +}; #endif /* _SMC_9194_H_ */ diff -urN orig/drivers/parport/Config.in linux/drivers/parport/Config.in --- orig/drivers/parport/Config.in Thu Dec 20 11:04:05 2001 +++ linux/drivers/parport/Config.in Mon Aug 5 23:45:16 2002 @@ -33,7 +33,8 @@ fi fi if [ "$CONFIG_ARM" = "y" ]; then - dep_tristate ' Archimedes hardware' CONFIG_PARPORT_ARC $CONFIG_PARPORT + dep_tristate ' Archimedes hardware' CONFIG_PARPORT_ARC $CONFIG_PARPORT $CONFIG_ARCH_ARC + dep_tristate ' Accelent SA1110 IDP' CONFIG_PARPORT_IDP $CONFIG_PARPORT $CONFIG_SA1100_ACCELENT fi if [ "$CONFIG_AMIGA" = "y" ]; then dep_tristate ' Amiga builtin port' CONFIG_PARPORT_AMIGA $CONFIG_PARPORT diff -urN orig/drivers/parport/Makefile linux/drivers/parport/Makefile --- orig/drivers/parport/Makefile Mon Oct 1 23:11:00 2001 +++ linux/drivers/parport/Makefile Mon Aug 5 23:44:34 2002 @@ -29,6 +29,7 @@ obj-$(CONFIG_PARPORT_ATARI) += parport_atari.o obj-$(CONFIG_PARPORT_SUNBPP) += parport_sunbpp.o obj-$(CONFIG_PARPORT_GSC) += parport_gsc.o +obj-$(CONFIG_PARPORT_IDP) += parport_idp.o include $(TOPDIR)/Rules.make diff -urN orig/drivers/parport/init.c linux/drivers/parport/init.c --- orig/drivers/parport/init.c Sat Jul 21 10:46:59 2001 +++ linux/drivers/parport/init.c Mon Aug 5 23:44:34 2002 @@ -163,6 +163,9 @@ #ifdef CONFIG_PARPORT_SUNBPP parport_sunbpp_init(); #endif +#ifdef CONFIG_PARPORT_IDP + parport_idp_init(); +#endif return 0; } diff -urN orig/drivers/parport/parport_idp.c linux/drivers/parport/parport_idp.c --- orig/drivers/parport/parport_idp.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/parport/parport_idp.c Mon Aug 5 23:44:34 2002 @@ -0,0 +1,247 @@ +/* Low-level polled-mode parallel port routines for the Accelent IDP + * + * Author: Rich Dulabahn + * + * Inspiration taken from parport_amiga.c and parport_atari.c. + * + * To use, under menuconfig: + * 1) Turn on <*> Accelent IDP under Parallel port setup + * 2) Turn on <*> Parallel printer support under Character devices + * + * This will give you parport0 configured as /dev/lp0 + * + * To make the correct /dev/lp* entries, enter /dev and type this: + * + * mknod lp0 c 6 0 + * mknod lp1 c 6 1 + * mknod lp2 c 6 2 + * + */ + +#include +#include +#include +#include + +/* + * Parallel data port is port H, data + * Parallel data direction is port H, direction + * Control port is port I, data, lowest 4 bits + * Status port is port G, data, upper 5 bits + */ + +#define INPUTPOWERHANDLER 0 +/* masks */ +#define CONTROL_MASK 0x0f +#define STATUS_MASK 0xf8 + +#undef DEBUG + +#ifdef DEBUG +#define DPRINTK printk +#else +#define DPRINTK(stuff...) +#endif + +static struct parport *this_port = NULL; + +static unsigned char +parport_idp_read_data(struct parport *p) +{ + unsigned char c; + + c = IDP_FPGA_PORTH_DATA; + DPRINTK("read_data:0x%x\n",c); + return c; +} + +static void +parport_idp_write_data(struct parport *p, unsigned char data) +{ + IDP_FPGA_PORTH_DATA = data; + DPRINTK("write_data:0x%x\n",data); +} + +static unsigned char +parport_idp_read_control(struct parport *p) +{ + unsigned char c; + + c = IDP_FPGA_PORTI_DATA & CONTROL_MASK; + DPRINTK("read_control:0x%x\n",c); + return c; +} + +static void +parport_idp_write_control(struct parport *p, unsigned char control) +{ + unsigned int temp; + + temp = IDP_FPGA_PORTH_DATA; + temp &= ~CONTROL_MASK; + IDP_FPGA_PORTI_DATA = (temp | (control & CONTROL_MASK)); +DPRINTK("write_control:0x%x\n",control); +} + +static unsigned char +parport_idp_frob_control(struct parport *p, unsigned char mask, + unsigned char val) +{ + unsigned char c; + +/* From the parport-lowlevel.txt file...*/ +/* This is equivalent to reading from the control register, masking out +the bits in mask, exclusive-or'ing with the bits in val, and writing +the result to the control register. */ + +/* Easy enough, right? */ + + c = parport_idp_read_control(p); + parport_idp_write_control(p, (c & ~mask) ^ val); + DPRINTK("frob_control:0x%x\n",c); + return c; +} + +static unsigned char +parport_idp_read_status(struct parport *p) +{ + unsigned char c; + + c = IDP_FPGA_PORTG_DATA & STATUS_MASK; + c ^= 0x80; /* toggle S7 bit, active low */ + DPRINTK("read_status:0x%x\n",c); + return c; +} + +static void +parport_idp_init_state(struct pardevice *d, struct parport_state *s) +{ +} + +static void +parport_idp_save_state(struct parport *p, struct parport_state *s) +{ +} + +static void +parport_idp_restore_state(struct parport *p, struct parport_state *s) +{ +} + +static void +parport_idp_interrupt(int irq, void *dev_id, struct pt_regs *regs) +{ +} + +static void +parport_idp_enable_irq(struct parport *p) +{ +} + +static void +parport_idp_disable_irq(struct parport *p) +{ +} + +static void +parport_idp_data_forward(struct parport *p) +{ + IDP_FPGA_PORTH_DIR = 0x00; /* 0 sets to output */ + DPRINTK("data_forward:0x%x\n",0); +} + +static void +parport_idp_data_reverse(struct parport *p) +{ + IDP_FPGA_PORTH_DIR = 0xff; /* and 1 sets to input */ + DPRINTK("data_reverse:0x%x\n",0xff); +} + +static void +parport_idp_inc_use_count(void) +{ + MOD_INC_USE_COUNT; +} + +static void +parport_idp_dec_use_count(void) +{ + MOD_DEC_USE_COUNT; +} + +static struct parport_operations parport_idp_ops = { + parport_idp_write_data, + parport_idp_read_data, + + parport_idp_write_control, + parport_idp_read_control, + parport_idp_frob_control, + + parport_idp_read_status, + + parport_idp_enable_irq, + parport_idp_disable_irq, + + parport_idp_data_forward, + parport_idp_data_reverse, + + parport_idp_init_state, + parport_idp_save_state, + parport_idp_restore_state, + + parport_idp_inc_use_count, + parport_idp_dec_use_count, + + parport_ieee1284_epp_write_data, + parport_ieee1284_epp_read_data, + parport_ieee1284_epp_write_addr, + parport_ieee1284_epp_read_addr, + + parport_ieee1284_ecp_write_data, + parport_ieee1284_ecp_read_data, + parport_ieee1284_ecp_write_addr, + + parport_ieee1284_write_compat, + parport_ieee1284_read_nibble, + parport_ieee1284_read_byte, +}; + + +int __init +parport_idp_init(void) +{ + struct parport *p; + + p = parport_register_port((unsigned long)0,PARPORT_IRQ_NONE,PARPORT_DMA_NONE,&parport_idp_ops); + + if (!p) return 0; /* return 0 on failure */ + + this_port=p; + printk("%s: Accelent IDP parallel port registered.\n", p->name); + parport_proc_register(p); + parport_announce_port(p); + + return 1; +} + +#ifdef MODULE + +MODULE_AUTHOR("Rich Dulabahn"); +MODULE_DESCRIPTION("Parport Driver for Accelent IDP"); +MODULE_SUPPORTED_DEVICE("Accelent IDP builtin Parallel Port"); +MODULE_LICENSE("GPL"); + +int +init_module(void) +{ + return parport_idp_init() ? 0 : -ENODEV; +} + +void +cleanup_module(void) +{ + parport_proc_unregister(this_port); + parport_unregister_port(this_port); +} +#endif + diff -urN orig/drivers/pci/Makefile linux/drivers/pci/Makefile --- orig/drivers/pci/Makefile Mon Aug 5 13:31:15 2002 +++ linux/drivers/pci/Makefile Mon Aug 5 13:44:41 2002 @@ -13,7 +13,7 @@ export-objs := pci.o -obj-$(CONFIG_PCI) += pci.o quirks.o compat.o names.o +obj-$(CONFIG_PCI) += pci.o quirks.o compat.o names.o bridge.o obj-$(CONFIG_PROC_FS) += proc.o ifndef CONFIG_SPARC64 diff -urN orig/drivers/pci/bridge.c linux/drivers/pci/bridge.c --- orig/drivers/pci/bridge.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/pci/bridge.c Mon Nov 5 21:42:26 2001 @@ -0,0 +1,149 @@ + +/* + * Copyright (c) 2001 Red Hat, Inc. All rights reserved. + * + * This software may be freely redistributed under the terms + * of the GNU public license. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. + * + * Author: Arjan van de Ven + * + */ + + +/* + * Generic PCI driver for PCI bridges for powermanagement purposes + * + */ + +#include +#include +#include +#include +#include + +static struct pci_device_id bridge_pci_table[] __devinitdata = { + {/* handle all PCI bridges */ + class: ((PCI_CLASS_BRIDGE_PCI << 8) | 0x00), + class_mask: ~0, + vendor: PCI_ANY_ID, + device: PCI_ANY_ID, + subvendor: PCI_ANY_ID, + subdevice: PCI_ANY_ID, + }, + {0,}, +}; + +static int bridge_probe(struct pci_dev *pdev, const struct pci_device_id *id); +static int pci_bridge_save_state_bus(struct pci_bus *bus, int force); +int pci_generic_resume_compare(struct pci_dev *pdev); + +int pci_bridge_force_restore = 0; + + + + +static int __init bridge_setup(char *str) +{ + if (!strcmp(str,"force")) + pci_bridge_force_restore = 1; + else if (!strcmp(str,"noforce")) + pci_bridge_force_restore = 0; + return 0; +} + +__setup("resume=",bridge_setup); + + +static int pci_bridge_save_state_bus(struct pci_bus *bus, int force) +{ + struct list_head *list; + int error = 0; + + list_for_each(list, &bus->children) { + error = pci_bridge_save_state_bus(pci_bus_b(list),force); + if (error) return error; + } + list_for_each(list, &bus->devices) { + pci_generic_suspend_save(pci_dev_b(list),0); + } + return 0; +} + + +static int pci_bridge_restore_state_bus(struct pci_bus *bus, int force) +{ + struct list_head *list; + int error = 0; + static int printed_warning=0; + + list_for_each(list, &bus->children) { + error = pci_bridge_restore_state_bus(pci_bus_b(list),force); + if (error) return error; + } + list_for_each(list, &bus->devices) { + if (force) + pci_generic_resume_restore(pci_dev_b(list)); + else { + error = pci_generic_resume_compare(pci_dev_b(list)); + if (error && !printed_warning++) { + printk(KERN_WARNING "resume warning: bios doesn't restore PCI state properly\n"); + printk(KERN_WARNING "resume warning: if resume failed, try booting with resume=force\n"); + } + if (error) + return error; + } + } + return 0; +} + +static int bridge_suspend(struct pci_dev *dev, u32 force) +{ + pci_generic_suspend_save(dev,force); + if (dev->subordinate) + pci_bridge_save_state_bus(dev->subordinate,force); + return 0; +} + +static int bridge_resume(struct pci_dev *dev) +{ + + pci_generic_resume_restore(dev); + if (dev->subordinate) + pci_bridge_restore_state_bus(dev->subordinate,pci_bridge_force_restore); + return 0; +} + + +MODULE_DEVICE_TABLE(pci, bridge_pci_table); +static struct pci_driver bridge_ops = { + name: "PCI Bridge", + id_table: bridge_pci_table, + probe: bridge_probe, + suspend: bridge_suspend, + resume: bridge_resume +}; + +static int __devinit bridge_probe(struct pci_dev *pdev, const struct pci_device_id *id) +{ + return 0; +} + +static int __init bridge_init(void) +{ + pci_register_driver(&bridge_ops); + return 0; +} + +static void __exit bridge_exit(void) +{ + pci_unregister_driver(&bridge_ops); +} + + +module_init(bridge_init) +module_exit(bridge_exit) + diff -urN orig/drivers/pci/pci.c linux/drivers/pci/pci.c --- orig/drivers/pci/pci.c Mon Aug 5 13:31:15 2002 +++ linux/drivers/pci/pci.c Tue Oct 22 23:24:41 2002 @@ -357,6 +357,48 @@ return 0; } +int +pci_compare_state(struct pci_dev *dev, u32 *buffer) +{ + int i; + unsigned int temp; + + if (buffer) { + for (i = 0; i < 16; i++) { + pci_read_config_dword(dev,i*4,&temp); + if (temp!=buffer[i]) + return 1; + } + } + return 0; +} + +int pci_generic_suspend_save(struct pci_dev *pdev, u32 state) +{ + if (pdev) + pci_save_state(pdev,pdev->saved_state); + return 0; +} + +int pci_generic_resume_restore(struct pci_dev *pdev) +{ + if (pdev) + pci_restore_state(pdev,pdev->saved_state); + return 0; +} + +int pci_generic_resume_compare(struct pci_dev *pdev) +{ + int retval=0; + if (pdev) + retval = pci_compare_state(pdev,pdev->saved_state); + return retval; +} + +EXPORT_SYMBOL(pci_generic_suspend_save); +EXPORT_SYMBOL(pci_generic_resume_restore); +EXPORT_SYMBOL(pci_generic_resume_compare); + /** * pci_enable_device - Initialize device before it's used by a driver. * @dev: PCI device to be initialized diff -urN orig/drivers/pci/setup-bus.c linux/drivers/pci/setup-bus.c --- orig/drivers/pci/setup-bus.c Mon Aug 5 13:31:16 2002 +++ linux/drivers/pci/setup-bus.c Mon Aug 5 15:01:03 2002 @@ -12,6 +12,8 @@ /* * Nov 2000, Ivan Kokshaysky * PCI-PCI bridges cleanup, sorted resource allocation + * May 2001, Russell King + * Allocate prefetchable memory regions where available. */ #include @@ -32,6 +34,26 @@ #define ROUND_UP(x, a) (((x) + (a) - 1) & ~((a) - 1)) +static inline void +pdev_adjust_mem_ranges(struct resource *res, struct pbus_set_ranges_data *ranges) +{ + unsigned long *end = &ranges->mem_end; + + /* + * We can't use this resources prefetch flag to determine which + * region it belongs to - we may have allocated it in the non- + * prefetchable region. Use the parent resource instead. + */ + if (res->parent->flags & IORESOURCE_PREFETCH) { + end = &ranges->prefetch_end; + if (ranges->prefetch_valid == 0) + BUG(); + } + + if (*end < res->end) + *end = res->end; +} + static int __init pbus_assign_resources_sorted(struct pci_bus *bus, struct pbus_set_ranges_data *ranges) @@ -89,9 +111,8 @@ for (list = head_mem.next; list;) { res = list->res; idx = res - &list->dev->resource[0]; - if (pci_assign_resource(list->dev, idx) == 0 - && ranges->mem_end < res->end) - ranges->mem_end = res->end; + if (pci_assign_resource(list->dev, idx) == 0) + pdev_adjust_mem_ranges(res, ranges); tmp = list; list = list->next; kfree(tmp); @@ -113,9 +134,12 @@ ranges->io_end += 1; if (ranges->mem_end == ranges->mem_start) ranges->mem_end += 1; + if (ranges->prefetch_end == ranges->prefetch_start) + ranges->prefetch_end += 1; #endif ranges->io_end = ROUND_UP(ranges->io_end, 4*1024); ranges->mem_end = ROUND_UP(ranges->mem_end, 1024*1024); + ranges->prefetch_end = ROUND_UP(ranges->prefetch_end, 1024*1024); return found_vga; } @@ -130,15 +154,24 @@ if (!bridge || (bridge->class >> 8) != PCI_CLASS_BRIDGE_PCI) return; + ranges.io_start = bus->resource[0]->start; ranges.io_end = bus->resource[0]->end; ranges.mem_start = bus->resource[1]->start; ranges.mem_end = bus->resource[1]->end; + ranges.prefetch_start = bus->resource[2]->start; + ranges.prefetch_end = bus->resource[2]->end; + pcibios_fixup_pbus_ranges(bus, &ranges); - DBGC((KERN_ERR "PCI: Bus %d, bridge: %s\n", bus->number, bridge->name)); - DBGC((KERN_ERR " IO window: %04lx-%04lx\n", ranges.io_start, ranges.io_end)); - DBGC((KERN_ERR " MEM window: %08lx-%08lx\n", ranges.mem_start, ranges.mem_end)); + DBGC((KERN_ERR "PCI: Bus %d, bridge: %s\n", + bus->number, bridge->name)); + DBGC((KERN_ERR " IO window: %04lx-%04lx\n", + ranges.io_start, ranges.io_end)); + DBGC((KERN_ERR " MEM window: %08lx-%08lx\n", + ranges.mem_start, ranges.mem_end)); + DBGC((KERN_ERR " PREFETCH window: %08lx-%08lx\n", + ranges.prefetch_start, ranges.prefetch_end)); /* Set up the top and bottom of the PCI I/O segment for this bus. */ pci_read_config_dword(bridge, PCI_IO_BASE, &l); @@ -161,13 +194,15 @@ pci_write_config_dword(bridge, PCI_MEMORY_BASE, l); /* Set up PREF base/limit. */ - l = (bus->resource[2]->start >> 16) & 0xfff0; - l |= bus->resource[2]->end & 0xfff00000; + l = (ranges.prefetch_start >> 16) & 0xfff0; + l |= ranges.prefetch_end & 0xfff00000; pci_write_config_dword(bridge, PCI_PREF_MEMORY_BASE, l); /* Check if we have VGA behind the bridge. Enable ISA in either case. */ - l = (bus->resource[0]->flags & IORESOURCE_BUS_HAS_VGA) ? 0x0c : 0x04; + l = (bus->resource[0]->flags & IORESOURCE_BUS_HAS_VGA) ? + PCI_BRIDGE_CTL_VGA | PCI_BRIDGE_CTL_NO_ISA : + PCI_BRIDGE_CTL_NO_ISA; pci_write_config_word(bridge, PCI_BRIDGE_CONTROL, l); } @@ -195,11 +230,16 @@ b->resource[0]->start = ranges->io_start = ranges->io_end; b->resource[1]->start = ranges->mem_start = ranges->mem_end; + if (ranges->prefetch_valid) + b->resource[2]->start = ranges->prefetch_start = + ranges->prefetch_end; pbus_assign_resources(b, ranges); b->resource[0]->end = ranges->io_end - 1; b->resource[1]->end = ranges->mem_end - 1; + if (ranges->prefetch_valid) + b->resource[2]->end = ranges->prefetch_end - 1; /* Add bridge resources to the resource tree. */ if (b->resource[0]->end > b->resource[0]->start && @@ -210,6 +250,10 @@ request_resource(bus->resource[1], b->resource[1]) < 0) printk(KERN_ERR "PCI: failed to reserve MEM " "for bus %d\n", b->number); + if (ranges->prefetch_valid && + request_resource(bus->resource[2], b->resource[2]) < 0) + printk(KERN_ERR "PCI: failed to reserve PREF-MEM " + "for bus %d\n", b->number); pci_setup_bridge(b); } @@ -229,6 +273,11 @@ ranges.mem_start = b->resource[1]->start + PCIBIOS_MIN_MEM; ranges.io_end = ranges.io_start; ranges.mem_end = ranges.mem_start; + ranges.prefetch_valid = b->resource[2] != NULL; + if (ranges.prefetch_valid) { + ranges.prefetch_start = b->resource[2]->start + PCIBIOS_MIN_MEM; + ranges.prefetch_end = ranges.prefetch_start; + } ranges.found_vga = 0; pbus_assign_resources(b, &ranges); } diff -urN orig/drivers/pcmcia/Config.in linux/drivers/pcmcia/Config.in --- orig/drivers/pcmcia/Config.in Mon Aug 5 13:31:16 2002 +++ linux/drivers/pcmcia/Config.in Mon Aug 5 13:44:43 2002 @@ -14,20 +14,23 @@ tristate 'PCMCIA/CardBus support' CONFIG_PCMCIA if [ "$CONFIG_PCMCIA" != "n" ]; then + # yes, I really mean the following... + if [ "$CONFIG_ISA" = "y" -o "$CONFIG_ARCH_SA1100" = "y" ]; then + define_bool CONFIG_PCMCIA_PROBE y + fi if [ "$CONFIG_PCI" != "n" ]; then bool ' CardBus support' CONFIG_CARDBUS fi + dep_bool ' i82092 compatible bridge support' CONFIG_I82092 $CONFIG_PCI + bool ' i82365 compatible bridge support' CONFIG_I82365 bool ' Databook TCIC host bridge support' CONFIG_TCIC if [ "$CONFIG_HD64465" = "y" ]; then dep_tristate ' HD64465 host bridge support' CONFIG_HD64465_PCMCIA $CONFIG_PCMCIA fi - dep_bool ' i82092 compatible bridge support' CONFIG_I82092 $CONFIG_PCI - bool ' i82365 compatible bridge support' CONFIG_I82365 - if [ "$CONFIG_ARCH_SA1100" = "y" ]; then - dep_tristate ' SA1100 support' CONFIG_PCMCIA_SA1100 $CONFIG_PCMCIA - fi - if [ "$CONFIG_8xx" = "y" ]; then - dep_tristate ' M8xx support' CONFIG_PCMCIA_M8XX $CONFIG_PCMCIA - fi fi +if [ "$CONFIG_ARM" = "y" ]; then + dep_tristate ' CLPS6700 support' CONFIG_PCMCIA_CLPS6700 $CONFIG_ARCH_CLPS711X $CONFIG_PCMCIA + dep_tristate ' SA1100 support' CONFIG_PCMCIA_SA1100 $CONFIG_ARCH_SA1100 $CONFIG_PCMCIA +fi + endmenu diff -urN orig/drivers/pcmcia/Makefile linux/drivers/pcmcia/Makefile --- orig/drivers/pcmcia/Makefile Mon Aug 5 13:31:16 2002 +++ linux/drivers/pcmcia/Makefile Fri Feb 21 15:23:13 2003 @@ -59,14 +59,17 @@ endif endif +obj-$(CONFIG_PCMCIA_CLPS6700) += clps6700.o obj-$(CONFIG_PCMCIA_SA1100) += sa1100_cs.o -obj-$(CONFIG_PCMCIA_M8XX) += m8xx_pcmcia.o sa1100_cs-objs-y := sa1100_generic.o +sa1100_cs-objs-$(CONFIG_SA1100_ADSAGC) += sa1100_graphicsmaster.o sa1111_generic.o sa1100_cs-objs-$(CONFIG_SA1100_ADSBITSY) += sa1100_adsbitsy.o sa1111_generic.o +sa1100_cs-objs-$(CONFIG_SA1100_ADSBITSYPLUS) += sa1100_adsbitsyplus.o sa1111_generic.o sa1100_cs-objs-$(CONFIG_SA1100_ASSABET) += sa1100_assabet.o sa1100_cs-objs-$(CONFIG_ASSABET_NEPONSET) += sa1100_neponset.o sa1111_generic.o sa1100_cs-objs-$(CONFIG_SA1100_BADGE4) += sa1100_badge4.o sa1111_generic.o +sa1100_cs-objs-$(CONFIG_SA1100_CONSUS) += sa1100_neponset.o sa1111_generic.o sa1100_cs-objs-$(CONFIG_SA1100_CERF) += sa1100_cerf.o sa1100_cs-objs-$(CONFIG_SA1100_FLEXANET) += sa1100_flexanet.o sa1100_cs-objs-$(CONFIG_SA1100_FREEBIRD) += sa1100_freebird.o diff -urN orig/drivers/pcmcia/cistpl.c linux/drivers/pcmcia/cistpl.c --- orig/drivers/pcmcia/cistpl.c Thu Dec 20 11:04:06 2001 +++ linux/drivers/pcmcia/cistpl.c Tue Jan 8 17:12:12 2002 @@ -285,7 +285,7 @@ s->cis_mem.flags &= ~MAP_ACTIVE; s->ss_entry->set_mem_map(s->sock, &s->cis_mem); if (!(s->cap.features & SS_CAP_STATIC_MAP)) - release_mem_region(s->cis_mem.sys_start, s->cap.map_size); + release_mem_resource(s->cis_mem.sys_start, s->cap.map_size); bus_iounmap(s->cap.bus, s->cis_virt); s->cis_mem.sys_start = 0; s->cis_virt = NULL; diff -urN orig/drivers/pcmcia/clps6700.c linux/drivers/pcmcia/clps6700.c --- orig/drivers/pcmcia/clps6700.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/pcmcia/clps6700.c Wed Jan 23 15:28:55 2002 @@ -0,0 +1,498 @@ +/* + * linux/drivers/pcmcia/clps6700.c + * + * Copyright (C) 2000 Deep Blue Solutions Ltd + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include + +#include +#include + +#include +#include +#include + +#include "clps6700.h" + +#define DEBUG + +MODULE_AUTHOR("Russell King"); +MODULE_DESCRIPTION("CL-PS6700 PCMCIA socket driver"); + +#define NR_CLPS6700 2 + +struct clps6700_skt { + u_int nr; + u_int physbase; + u_int regbase; + u_int pmr; + u_int cpcr; + u_int cpcr_3v3; + u_int cpcr_5v0; + u_int cur_pmr; + u_int cur_cicr; + u_int cur_pcimr; + u_int cur_cpcr; + void (*handler)(void *, u_int); + void *handler_info; + + u_int ev_pending; + spinlock_t ev_lock; +}; + +static struct clps6700_skt *skts[NR_CLPS6700]; + +static int clps6700_sock_init(u_int sock) +{ + struct clps6700_skt *skt = skts[sock]; + + skt->cur_cicr = 0; + skt->cur_pmr = skt->pmr; + skt->cur_pcimr = 0; + skt->cur_cpcr = skt->cpcr; + +#ifdef DEBUG + printk("skt%d: sock_init()\n", sock); +#endif + + __raw_writel(skt->cur_pmr, skt->regbase + PMR); + __raw_writel(skt->cur_cpcr, skt->regbase + CPCR); + __raw_writel(0x01f8, skt->regbase + SICR); + __raw_writel(0x0000, skt->regbase + DMACR); + __raw_writel(skt->cur_cicr, skt->regbase + CICR); + __raw_writel(0x1f00, skt->regbase + CITR0A); + __raw_writel(0x0000, skt->regbase + CITR0B); + __raw_writel(0x1f00, skt->regbase + CITR1A); + __raw_writel(0x0000, skt->regbase + CITR1B); + __raw_writel(skt->cur_pcimr, skt->regbase + PCIMR); + + /* + * Enable Auto Idle Mode in PM register + */ + __raw_writel(-1, skt->regbase + PCIRR1); + __raw_writel(-1, skt->regbase + PCIRR2); + __raw_writel(-1, skt->regbase + PCIRR3); + + return 0; +} + +static int clps6700_suspend(u_int sock) +{ + return 0; +} + +static int clps6700_register_callback(u_int sock, void (*handler)(void *, u_int), void *info) +{ + struct clps6700_skt *skt = skts[sock]; + +#ifdef DEBUG + printk("skt%d: register_callback: %p (%p)\n", sock, handler, info); +#endif + + skt->handler_info = info; + skt->handler = handler; + + return 0; +} + +static int clps6700_inquire_socket(u_int sock, socket_cap_t *cap) +{ + cap->features = SS_CAP_PCCARD | SS_CAP_STATIC_MAP | SS_CAP_MEM_ALIGN; + cap->irq_mask = 0; /* available IRQs for this socket */ + cap->map_size = PAGE_SIZE; /* minimum mapping size */ + cap->pci_irq = 0; /* PCI interrupt number */ + cap->cb_dev = NULL; + cap->bus = NULL; + return 0; +} + +static int __clps6700_get_status(struct clps6700_skt *skt) +{ + unsigned int v, val; + + v = __raw_readl(skt->regbase + PCIILR); + val = 0; + if ((v & (PCM_CD1 | PCM_CD2)) == 0) + val |= SS_DETECT; + if ((v & (PCM_BVD2 | PCM_BVD1)) == PCM_BVD1) + val |= SS_BATWARN; + if ((v & PCM_BVD2) == 0) + val |= SS_BATDEAD; + + if (v & PCM_RDYL) + val |= SS_READY; + if (v & PCM_VS1) + val |= SS_3VCARD; + if (v & PCM_VS2) + val |= SS_XVCARD; + +#ifdef DEBUG + printk("skt%d: PCIILR: %08x -> (%s %s %s %s %s %s)\n", + skt->nr, v, + val & SS_READY ? "rdy" : "---", + val & SS_DETECT ? "det" : "---", + val & SS_BATWARN ? "bw" : "--", + val & SS_BATDEAD ? "bd" : "--", + val & SS_3VCARD ? "3v" : "--", + val & SS_XVCARD ? "xv" : "--"); +#endif + return val; +} + +static int clps6700_get_status(u_int sock, u_int *valp) +{ + struct clps6700_skt *skt = skts[sock]; + + *valp = __clps6700_get_status(skt); + + return 0; /* not used! */ +} + +static int clps6700_get_socket(u_int sock, socket_state_t *state) +{ + return -EINVAL; +} + +static int clps6700_set_socket(u_int sock, socket_state_t *state) +{ + struct clps6700_skt *skt = skts[sock]; + unsigned long flags; + u_int cpcr = skt->cur_cpcr, pmr = skt->cur_pmr, cicr = skt->cur_cicr; + u_int pcimr = 0; + + cicr &= ~(CICR_ENABLE | CICR_RESET | CICR_IOMODE); + + if (state->flags & SS_PWR_AUTO) + pmr |= PMR_DCAR | PMR_PDCR; + + /* + * Note! We must NOT assert the Card Enable bit until reset has + * been de-asserted. Some cards indicate not ready, which then + * hangs our next access. (Bug in CLPS6700?) + */ + if (state->flags & SS_RESET) + cicr |= CICR_RESET | CICR_RESETOE; + else if (state->flags & SS_OUTPUT_ENA) + cicr |= CICR_ENABLE; + + if (state->flags & SS_IOCARD) { + cicr |= CICR_IOMODE; + +/* if (state->csc_mask & SS_STSCHG)*/ + } else { + if (state->csc_mask & SS_BATDEAD) + pcimr |= PCM_BVD2; + if (state->csc_mask & SS_BATWARN) + pcimr |= PCM_BVD1; + if (state->csc_mask & SS_READY) + pcimr |= PCM_RDYL; + } + + if (state->csc_mask & SS_DETECT) + pcimr |= PCM_CD1 | PCM_CD2; + + switch (state->Vcc) { + case 0: break; + case 33: cpcr |= skt->cpcr_3v3; pmr |= PMR_CPE; break; + case 50: cpcr |= skt->cpcr_5v0; pmr |= PMR_CPE; break; + default: return -EINVAL; + } + +#ifdef DEBUG + printk("skt%d: PMR: %04x, CPCR: %04x, CICR: %04x PCIMR: %04x " + "(Vcc = %d, flags = %c%c%c%c, csc = %c%c%c%c%c)\n", + sock, pmr, cpcr, cicr, pcimr, state->Vcc, + state->flags & SS_RESET ? 'r' : '-', + state->flags & SS_PWR_AUTO ? 'p' : '-', + state->flags & SS_IOCARD ? 'i' : '-', + state->flags & SS_OUTPUT_ENA ? 'o' : '-', + state->csc_mask & SS_STSCHG ? 's' : '-', + state->csc_mask & SS_BATDEAD ? 'd' : '-', + state->csc_mask & SS_BATWARN ? 'w' : '-', + state->csc_mask & SS_READY ? 'r' : '-', + state->csc_mask & SS_DETECT ? 'c' : '-'); +#endif + + save_flags_cli(flags); + + if (skt->cur_cpcr != cpcr) { + skt->cur_cpcr = cpcr; + __raw_writel(skt->cur_cpcr, skt->regbase + CPCR); + } + + if (skt->cur_pmr != pmr) { + skt->cur_pmr = pmr; + __raw_writel(skt->cur_pmr, skt->regbase + PMR); + } + if (skt->cur_pcimr != pcimr) { + skt->cur_pcimr = pcimr; + __raw_writel(skt->cur_pcimr, skt->regbase + PCIMR); + } + if (skt->cur_cicr != cicr) { + skt->cur_cicr = cicr; + __raw_writel(skt->cur_cicr, skt->regbase + CICR); + } + + restore_flags(flags); + + return 0; +} + +static int clps6700_get_io_map(u_int sock, struct pccard_io_map *io) +{ + return -EINVAL; +} + +static int clps6700_set_io_map(u_int sock, struct pccard_io_map *io) +{ + printk("skt%d: iomap: %d: speed %d, flags %X start %X stop %X\n", + sock, io->map, io->speed, io->flags, io->start, io->stop); + return 0; +} + +static int clps6700_get_mem_map(u_int sock, struct pccard_mem_map *mem) +{ + return -EINVAL; +} + +/* + * Set the memory map attributes for this socket. (ie, mem->speed) + * Note that since we have SS_CAP_STATIC_MAP set, we don't need to do + * any mapping here at all; we just need to return the address (suitable + * for ioremap) to map the requested space in mem->sys_start. + * + * flags & MAP_ATTRIB indicates whether we want attribute space. + */ +static int clps6700_set_mem_map(u_int sock, struct pccard_mem_map *mem) +{ + struct clps6700_skt *skt = skts[sock]; + u_int off; + + printk("skt%d: memmap: %d: speed %d, flags %X start %lX stop %lX card %X\n", + sock, mem->map, mem->speed, mem->flags, mem->sys_start, + mem->sys_stop, mem->card_start); + + if (mem->flags & MAP_ATTRIB) + off = CLPS6700_ATTRIB_BASE; + else + off = CLPS6700_MEM_BASE; + + mem->sys_start = skt->physbase + off; + mem->sys_start += mem->card_start; + + return 0; +} + +static void clps6700_proc_setup(u_int sock, struct proc_dir_entry *base) +{ +} + +static struct pccard_operations clps6700_operations = { + init: clps6700_sock_init, + suspend: clps6700_suspend, + register_callback: clps6700_register_callback, + inquire_socket: clps6700_inquire_socket, + get_status: clps6700_get_status, + get_socket: clps6700_get_socket, + set_socket: clps6700_set_socket, + get_io_map: clps6700_get_io_map, + set_io_map: clps6700_set_io_map, + get_mem_map: clps6700_get_mem_map, + set_mem_map: clps6700_set_mem_map, + proc_setup: clps6700_proc_setup +}; + +static void clps6700_bh(void *dummy) +{ + int i; + + for (i = 0; i < NR_CLPS6700; i++) { + struct clps6700_skt *skt = skts[i]; + unsigned long flags; + u_int events; + + if (!skt) + continue; + + /* + * Note! We must read the pending event state + * with interrupts disabled, otherwise we race + * with our own interrupt routine! + */ + spin_lock_irqsave(&skt->ev_lock, flags); + events = skt->ev_pending; + skt->ev_pending = 0; + spin_unlock_irqrestore(&skt->ev_lock, flags); + + if (skt->handler && events) + skt->handler(skt->handler_info, events); + } +} + +static struct tq_struct clps6700_task = { + routine: clps6700_bh +}; + +static void clps6700_interrupt(int irq, void *dev_id, struct pt_regs *regs) +{ + struct clps6700_skt *skt = dev_id; + u_int val, events; + + val = __raw_readl(skt->regbase + PCISR); + if (!val) + return; + + __raw_writel(val, skt->regbase + PCICR); + + events = 0; + if (val & (PCM_CD1 | PCM_CD2)) + events |= SS_DETECT; + if (val & PCM_BVD1) + events |= SS_BATWARN; + if (val & PCM_BVD2) + events |= SS_BATDEAD; + if (val & PCM_RDYL) + events |= SS_READY; + + spin_lock(&skt->ev_lock); + skt->ev_pending |= events; + spin_unlock(&skt->ev_lock); + schedule_task(&clps6700_task); +} + +static int __init clps6700_init_skt(int nr) +{ + struct clps6700_skt *skt; + int ret; + + skt = kmalloc(sizeof(struct clps6700_skt), GFP_KERNEL); + if (!skt) + return -ENOMEM; + + memset(skt, 0, sizeof(struct clps6700_skt)); + + spin_lock_init(&skt->ev_lock); + + skt->nr = nr; + skt->physbase = nr ? CS5_PHYS_BASE : CS4_PHYS_BASE; + skt->pmr = PMR_AUTOIDLE | PMR_MCPE | PMR_CDWEAK; + skt->cpcr = CPCR_PDIR(PCTL1|PCTL0); + skt->cpcr_3v3 = CPCR_PON(PCTL0); + skt->cpcr_5v0 = CPCR_PON(PCTL0); /* we only do 3v3 */ + + skt->cur_pmr = skt->pmr; + + skt->regbase = (u_int)ioremap(skt->physbase + CLPS6700_REG_BASE, + CLPS6700_REG_SIZE); + ret = -ENOMEM; + if (!skt->regbase) + goto err_free; + + skts[nr] = skt; + + ret = request_irq(IRQ_EINT3, clps6700_interrupt, + SA_SHIRQ, "pcmcia", skt); + + if (ret) { + printk(KERN_ERR "clps6700: unable to grab irq%d (%d)\n", + IRQ_EINT3, ret); + goto err_unmap; + } + return 0; + +err_unmap: + iounmap((void *)skt->regbase); +err_free: + kfree(skt); + skts[nr] = NULL; + return ret; +} + +static void clps6700_free_resources(void) +{ + int i; + + for (i = NR_CLPS6700; i >= 0; i--) { + struct clps6700_skt *skt = skts[i]; + + skts[i] = NULL; + if (skt == NULL) + continue; + + free_irq(IRQ_EINT3, skt); + if (skt->regbase) { + __raw_writel(skt->pmr, skt->regbase + PMR); + __raw_writel(skt->cpcr, skt->regbase + CPCR); + __raw_writel(0, skt->regbase + CICR); + __raw_writel(0, skt->regbase + PCIMR); + } + iounmap((void *)skt->regbase); + kfree(skt); + } +} + +static int __init clps6700_init(void) +{ + unsigned int v; + int err, nr; + + PLD_CF = 0; + v = clps_readl(SYSCON2) | SYSCON2_PCCARD1 | SYSCON2_PCCARD2; + clps_writel(v, SYSCON2); + v = clps_readl(SYSCON1) | SYSCON1_EXCKEN; + clps_writel(v, SYSCON1); + + for (nr = 0; nr < NR_CLPS6700; nr++) { + err = clps6700_init_skt(nr); + if (err) + goto free; + } + + err = register_ss_entry(nr, &clps6700_operations); + if (err) + goto free; + + return 0; + +free: + clps6700_free_resources(); + /* + * An error occurred. Unmap and free all CLPS6700 + */ + return err; +} + +static void __exit clps6700_exit(void) +{ + unregister_ss_entry(&clps6700_operations); + clps6700_free_resources(); +} + +module_init(clps6700_init); +module_exit(clps6700_exit); diff -urN orig/drivers/pcmcia/clps6700.h linux/drivers/pcmcia/clps6700.h --- orig/drivers/pcmcia/clps6700.h Thu Jan 1 01:00:00 1970 +++ linux/drivers/pcmcia/clps6700.h Thu Nov 2 16:58:29 2000 @@ -0,0 +1,85 @@ +#define PCISR 0x0000 /* PC Card Interrupt Status Register */ +#define PCIMR 0x0400 /* PC Card Interrupt Mask Register */ +#define PCICR 0x0800 /* PC Card Interrupt Clear Register */ +#define PCIOSR 0x0c00 /* PC Card Interrupt Output Select Regsiter */ +#define PCIRR1 0x1000 /* PC Card Interrupt Reserved Register 1 */ +#define PCIRR2 0x1400 /* PC Card Interrupt Reserved Register 2 */ +#define PCIRR3 0x1800 /* PC Card Interrupt Reserved Register 3 */ +#define PCIILR 0x1c00 /* PC Card Interrupt Input Level Register */ +#define SICR 0x2000 /* System Interface Configuration Register */ +#define CICR 0x2400 /* Card Interface Configuration Register */ +#define PMR 0x2800 /* Power Management Register */ +#define CPCR 0x2c00 /* Card Power Control Register */ +#define CITR0A 0x3000 /* Card Interface Timing Register 0A */ +#define CITR0B 0x3400 /* Card Interface Timing Register 0B */ +#define CITR1A 0x3800 /* Card Interface Timing Register 1A */ +#define CITR1B 0x3c00 /* Card Interface Timing Register 1B */ +#define DMACR 0x4000 /* DMA Control Register */ +#define DIR 0x4400 /* Device Information Register */ + +#define CLPS6700_ATTRIB_BASE 0x00000000 +#define CLPS6700_IO_BASE 0x04000000 +#define CLPS6700_MEM_BASE 0x08000000 +#define CLPS6700_REG_BASE 0x0c000000 +#define CLPS6700_REG_SIZE 0x00005000 + + +#define PMR_AUTOIDLE (1 << 0) /* auto idle mode */ +#define PMR_FORCEIDLE (1 << 1) /* force idle mode */ +#define PMR_PDCS (1 << 2) /* Power down card on standby */ +#define PMR_PDCR (1 << 3) /* Power down card on removal */ +#define PMR_DCAR (1 << 4) /* Disable card access on removal */ +#define PMR_CPE (1 << 5) /* Card power enable */ +#define PMR_MCPE (1 << 6) /* Monitor card power enable */ +#define PMR_PDREQLSEL (1 << 7) /* If set, PDREQL is a GPIO pin */ +#define PMR_DISSTBY (1 << 8) /* Disable standby */ +#define PMR_ACCSTBY (1 << 9) /* Complete card accesses before standby*/ +#define PMR_CDUNPROT (0 << 10) /* Card detect inputs unprotected */ +#define PMR_CDPROT (1 << 10) /* Card detect inputs protected */ +#define PMR_CDWEAK (2 << 10) /* Weak pullup except in standby */ +#define PMR_CDWEAKAL (3 << 10) /* Weak pullup */ + +#define CPCR_PON(x) ((x)&7) /* PCTL[2:0] value when PMR_CPE = 1 */ +#define CPCR_POFF(x) (((x)&7)<<3) /* PCTL[2:0] value when PMR_CPE = 0 */ +#define CPCR_PDIR(x) (((x)&7)<<6) /* PCTL[2:0] direction */ +#define CPCR_CON(x) (((x)&1)<<9) /* GPIO value when PMR_CPE = 1 */ +#define CPCR_COFF(x) (((x)&1)<<10) /* GPIO value when PMR_CPE = 0 */ +#define CPCR_CDIR(x) (((x)&1)<<11) /* GPIO direction (PMR_PDREQLSEL = 1) */ +#define CPCR_VS(x) (((x)&3)<<12) /* VS[2:1] output value */ +#define CPCR_VSDIR(x) (((x)&3)<<14) /* VS[2:1] direction */ + +#define PCTL0 (1 << 0) +#define PCTL1 (1 << 1) +#define PCTL2 (1 << 2) + +#define CICR_ASRTMR1 (1 << 0) /* Timer 1 select for attribute read */ +#define CICR_ASWTMR1 (1 << 1) /* Timer 1 select for attribute write */ +#define CICR_IOSRTMR1 (1 << 2) /* Timer 1 select for IO read */ +#define CICR_IOSWTMR1 (1 << 3) /* Timer 1 select for IO write */ +#define CICR_MEMSRTMR1 (1 << 4) /* Timer 1 select for memory read */ +#define CICR_MEMSWTMR1 (1 << 5) /* Timer 1 select for memory write */ +#define CICR_AUTOIOSZ (1 << 6) /* Auto size I/O accesses */ +#define CICR_CAW (1 << 7) /* Card access width */ +#define CICR_IOMODE (1 << 8) /* IO mode select */ +#define CICR_ENABLE (1 << 10) /* Card enable */ +#define CICR_RESETOE (1 << 11) /* Card reset output enable */ +#define CICR_RESET (1 << 12) /* Card reset */ + + +#define RD_FAIL (1 << 14) +#define WR_FAIL (1 << 13) +#define IDLE (1 << 12) + +#define FFOTHLD (1 << 11) +#define PCM_RDYL (1 << 10) +#define PCM_WP (1 << 9) +#define PCTL (1 << 8) + +#define PDREQ_L (1 << 6) +#define PCM_VS2 (1 << 5) +#define PCM_VS1 (1 << 4) + +#define PCM_CD2 (1 << 3) +#define PCM_CD1 (1 << 2) +#define PCM_BVD2 (1 << 1) +#define PCM_BVD1 (1 << 0) diff -urN orig/drivers/pcmcia/cs.c linux/drivers/pcmcia/cs.c --- orig/drivers/pcmcia/cs.c Mon Aug 5 13:31:16 2002 +++ linux/drivers/pcmcia/cs.c Mon Aug 5 13:44:44 2002 @@ -675,10 +675,17 @@ static void parse_events(void *info, u_int events) { socket_info_t *s = info; + if (events & SS_DETECT) { int status; get_socket_status(s, &status); + + /* + * If our socket state indicates that a card is present and + * either the socket has not been suspended (for some reason) + * or the card has been removed, shut down the socket first. + */ if ((s->state & SOCKET_PRESENT) && (!(s->state & SOCKET_SUSPEND) || !(status & SS_DETECT))) @@ -1609,7 +1616,7 @@ bus_free_irq(s->cap.bus, req->AssignedIRQ, req->Instance); } -#ifdef CONFIG_ISA +#ifdef CONFIG_PCMCIA_PROBE if (req->AssignedIRQ != s->cap.pci_irq) undo_irq(req->Attributes, req->AssignedIRQ); #endif @@ -1636,7 +1643,7 @@ /* Release system memory */ if(!(s->cap.features & SS_CAP_STATIC_MAP)) - release_mem_region(win->base, win->size); + release_mem_resource(win->base, win->size); win->handle->state &= ~CLIENT_WIN_REQ(win->index); win->magic = 0; @@ -1875,7 +1882,7 @@ if (!s->cap.irq_mask) { irq = s->cap.pci_irq; ret = (irq) ? 0 : CS_IN_USE; -#ifdef CONFIG_ISA +#ifdef CONFIG_PCMCIA_PROBE } else if (s->irq.AssignedIRQ != 0) { /* If the interrupt is already assigned, it must match */ irq = s->irq.AssignedIRQ; diff -urN orig/drivers/pcmcia/ds.c linux/drivers/pcmcia/ds.c --- orig/drivers/pcmcia/ds.c Fri Nov 23 10:12:15 2001 +++ linux/drivers/pcmcia/ds.c Tue Nov 13 14:55:30 2001 @@ -55,6 +55,7 @@ #include #include #include +#include /*====================================================================*/ @@ -880,6 +881,8 @@ EXPORT_SYMBOL(register_pccard_driver); EXPORT_SYMBOL(unregister_pccard_driver); +static devfs_handle_t devfs_handle; + /*====================================================================*/ int __init init_pcmcia_ds(void) @@ -957,8 +960,13 @@ if (i == -EBUSY) printk(KERN_NOTICE "unable to find a free device # for " "Driver Services\n"); - else + else { major_dev = i; + devfs_handle = devfs_register(NULL, "pcmcia", DEVFS_FL_DEFAULT, + major_dev, 0, + S_IFCHR | S_IRUSR | S_IWUSR, + &ds_fops, NULL); + } #ifdef CONFIG_PROC_FS if (proc_pccard) @@ -983,7 +991,9 @@ remove_proc_entry("drivers", proc_pccard); #endif if (major_dev != -1) - unregister_chrdev(major_dev, "pcmcia"); + devfs_unregister_chrdev(major_dev, "pcmcia"); + devfs_unregister(devfs_handle); + for (i = 0; i < sockets; i++) pcmcia_deregister_client(socket_table[i].handle); sockets = 0; diff -urN orig/drivers/pcmcia/i82365.c linux/drivers/pcmcia/i82365.c --- orig/drivers/pcmcia/i82365.c Fri Nov 23 10:12:15 2001 +++ linux/drivers/pcmcia/i82365.c Wed Feb 6 19:20:41 2002 @@ -66,6 +66,15 @@ #include "ricoh.h" #include "o2micro.h" +#ifdef CONFIG_ARCH_EBSA110 +#define I365_MASK (1 << 6) +#define SOCKIRQ2REG(sock,irq) ((irq) ? ((sock) ? 3 : 4) : 0) +#define REG2SOCKIRQ(sock,reg) (6) +#else +#define SOCKIRQ2REG(sock,irq) (irq) +#define REG2SOCKIRQ(sock,reg) (reg) +#endif + #ifdef PCMCIA_DEBUG static int pc_debug = PCMCIA_DEBUG; MODULE_PARM(pc_debug, "i"); @@ -180,13 +189,15 @@ } socket_info_t; /* Where we keep track of our sockets... */ -static int sockets = 0; -static socket_info_t socket[8] = { - { 0, }, /* ... */ -}; +static int sockets /* = 0 */; +static socket_info_t socket[8] /* = { + { 0, }, +} */; /* Default ISA interrupt mask */ +#ifndef I365_MASK #define I365_MASK 0xdeb8 /* irq 15,14,12,11,10,9,7,5,4,3 */ +#endif #ifdef CONFIG_ISA static int grab_irq; @@ -555,7 +566,7 @@ } /* Generate one interrupt */ - i365_set(sock, I365_CSCINT, I365_CSC_DETECT | (irq << 4)); + i365_set(sock, I365_CSCINT, I365_CSC_DETECT | (SOCKIRQ2REG(sock, irq) << 4)); i365_bset(sock, I365_GENCTL, I365_CTL_SW_IRQ); udelay(1000); @@ -1119,7 +1130,7 @@ reg = i365_get(sock, I365_INTCTL); state->flags |= (reg & I365_PC_RESET) ? 0 : SS_RESET; if (reg & I365_PC_IOCARD) state->flags |= SS_IOCARD; - state->io_irq = reg & I365_IRQ_MASK; + state->io_irq = REG2SOCKIRQ(sock, reg & I365_IRQ_MASK); /* speaker control */ if (t->flags & IS_CIRRUS) { @@ -1160,7 +1171,7 @@ /* IO card, RESET flag, IO interrupt */ reg = t->intr; - if (state->io_irq != t->cap.pci_irq) reg |= state->io_irq; + if (state->io_irq != t->cap.pci_irq) reg |= SOCKIRQ2REG(sock, state->io_irq); reg |= (state->flags & SS_RESET) ? 0 : I365_PC_RESET; reg |= (state->flags & SS_IOCARD) ? I365_PC_IOCARD : 0; i365_set(sock, I365_INTCTL, reg); @@ -1239,7 +1250,7 @@ } /* Card status change interrupt mask */ - reg = t->cs_irq << 4; + reg = SOCKIRQ2REG(sock, t->cs_irq) << 4; if (state->csc_mask & SS_DETECT) reg |= I365_CSC_DETECT; if (state->flags & SS_IOCARD) { if (state->csc_mask & SS_STSCHG) reg |= I365_CSC_STSCHG; diff -urN orig/drivers/pcmcia/rsrc_mgr.c linux/drivers/pcmcia/rsrc_mgr.c --- orig/drivers/pcmcia/rsrc_mgr.c Thu Dec 20 11:04:06 2001 +++ linux/drivers/pcmcia/rsrc_mgr.c Tue Jan 15 21:46:50 2002 @@ -55,6 +55,10 @@ #include #include "cs_internal.h" +#ifndef ISAMEM_PHYS +#define ISAMEM_PHYS 0 +#endif + /*====================================================================*/ /* Parameters that can be set with 'insmod' */ @@ -62,7 +66,7 @@ #define INT_MODULE_PARM(n, v) static int n = v; MODULE_PARM(n, "i") INT_MODULE_PARM(probe_mem, 1); /* memory probe? */ -#ifdef CONFIG_ISA +#ifdef CONFIG_PCMCIA_PROBE INT_MODULE_PARM(probe_io, 1); /* IO port probe? */ INT_MODULE_PARM(mem_limit, 0x10000); #endif @@ -85,7 +89,7 @@ /* IO port resource database */ static resource_map_t io_db = { 0, 0, &io_db }; -#ifdef CONFIG_ISA +#ifdef CONFIG_PCMCIA_PROBE typedef struct irq_info_t { u_int Attributes; @@ -133,6 +137,7 @@ static inline int check_mem_resource(unsigned long b, unsigned long n, struct pci_dev *dev) { + b += ISAMEM_PHYS; return check_resource(resource_parent(b, n, IORESOURCE_MEM, dev), b, n); } @@ -169,10 +174,15 @@ static int request_mem_resource(unsigned long b, unsigned long n, char *name, struct pci_dev *dev) { - struct resource *res = make_resource(b, n, IORESOURCE_MEM, name); - struct resource *pr = resource_parent(b, n, IORESOURCE_MEM, dev); + struct resource *res; + struct resource *pr; int err = -ENOMEM; + b += ISAMEM_PHYS; + + res = make_resource(b, n, IORESOURCE_MEM, name); + pr = resource_parent(b, n, IORESOURCE_MEM, dev); + if (res) { err = request_resource(pr, res); if (err) @@ -181,6 +191,12 @@ return err; } +void release_mem_resource(unsigned long b, unsigned long n) +{ + b += ISAMEM_PHYS; + release_mem_region(b, n); +} + /*====================================================================== These manage the internal databases of available resources. @@ -251,7 +267,7 @@ ======================================================================*/ -#ifdef CONFIG_ISA +#ifdef CONFIG_PCMCIA_PROBE static void do_io_probe(ioaddr_t base, ioaddr_t num) { @@ -356,7 +372,7 @@ return (num - bad); } -#ifdef CONFIG_ISA +#ifdef CONFIG_PCMCIA_PROBE static u_long inv_probe(int (*is_valid)(u_long), int (*do_cksum)(u_long), @@ -414,7 +430,7 @@ } } -#else /* CONFIG_ISA */ +#else /* CONFIG_PCMCIA_PROBE */ void validate_mem(int (*is_valid)(u_long), int (*do_cksum)(u_long), int force_low, socket_info_t *s) @@ -429,7 +445,7 @@ return; } -#endif /* CONFIG_ISA */ +#endif /* CONFIG_PCMCIA_PROBE */ /*====================================================================== @@ -500,7 +516,7 @@ ======================================================================*/ -#ifdef CONFIG_ISA +#ifdef CONFIG_PCMCIA_PROBE static void fake_irq(int i, void *d, struct pt_regs *r) { } static inline int check_irq(int irq) @@ -567,7 +583,7 @@ /*====================================================================*/ -#ifdef CONFIG_ISA +#ifdef CONFIG_PCMCIA_PROBE void undo_irq(u_int Attributes, int irq) { @@ -650,7 +666,7 @@ case ADD_MANAGED_RESOURCE: if (add_interval(&io_db, base, num) != 0) return CS_IN_USE; -#ifdef CONFIG_ISA +#ifdef CONFIG_PCMCIA_PROBE if (probe_io) do_io_probe(base, num); #endif @@ -670,7 +686,7 @@ static int adjust_irq(adjust_t *adj) { -#ifdef CONFIG_ISA +#ifdef CONFIG_PCMCIA_PROBE int irq; irq_info_t *info; diff -urN orig/drivers/pcmcia/sa1100.h linux/drivers/pcmcia/sa1100.h --- orig/drivers/pcmcia/sa1100.h Mon Aug 5 13:31:16 2002 +++ linux/drivers/pcmcia/sa1100.h Fri Feb 21 15:23:45 2003 @@ -204,7 +204,9 @@ extern struct pcmcia_low_level flexanet_pcmcia_ops; extern struct pcmcia_low_level simpad_pcmcia_ops; extern struct pcmcia_low_level graphicsmaster_pcmcia_ops; +extern struct pcmcia_low_level adsagc_pcmcia_ops; extern struct pcmcia_low_level adsbitsy_pcmcia_ops; +extern struct pcmcia_low_level adsbitsyplus_pcmcia_ops; extern struct pcmcia_low_level stork_pcmcia_ops; extern struct pcmcia_low_level badge4_pcmcia_ops; diff -urN orig/drivers/pcmcia/sa1100_adsbitsy.c linux/drivers/pcmcia/sa1100_adsbitsy.c --- orig/drivers/pcmcia/sa1100_adsbitsy.c Mon Aug 5 13:31:16 2002 +++ linux/drivers/pcmcia/sa1100_adsbitsy.c Thu Feb 27 23:20:10 2003 @@ -11,28 +11,156 @@ */ #include #include +#include #include +#include +#include #include "sa1100_generic.h" #include "sa1111_generic.h" +int adsbitsy_smc91111_present(void); + +#ifndef CONFIG_SMC91111 +#define adsbitsy_smc91111_present() 0 +#endif + +static struct irqs { + int irq; + const char *str; +} irqs[] = { + { S0_CD_VALID, "SA1111 PCMCIA card detect" }, + { S0_BVD1_STSCHG, "SA1111 PCMCIA BVD1" }, + { S1_CD_VALID, "SA1111 CF card detect" }, + { S1_BVD1_STSCHG, "SA1111 CF BVD1" }, +}; + static int adsbitsy_pcmcia_init(struct pcmcia_init *init) { - /* Set GPIO_A<3:0> to be outputs for PCMCIA/CF power controller: */ - PA_DDR &= ~(GPIO_GPIO0 | GPIO_GPIO1 | GPIO_GPIO2 | GPIO_GPIO3); + int ret=0; + int nirq = 0; + int slots = 0; + int i; + + /* Set GPIO_A<1:0> to be outputs for PCMCIA power controller: */ + PA_DDR &= ~(GPIO_GPIO0 | GPIO_GPIO1); + + /* Disable Power 3.3V/5V for PCMCIA */ + PA_DWR |= GPIO_GPIO0 | GPIO_GPIO1; + + if (!request_mem_region(_PCCR, 512, "PCMCIA")) + return -1; + + + INTPOL1 |= SA1111_IRQMASK_HI(S0_READY_NINT) | + SA1111_IRQMASK_HI(S0_CD_VALID) | + SA1111_IRQMASK_HI(S0_BVD1_STSCHG); + + nirq = 2; + slots = 1; + + if (!adsbitsy_smc91111_present()) { + /* If the SMC91111 is used CF cannot be used */ + /* Set GPIO_A<3:2> to be outputs for CF power controller: */ + PA_DDR &= ~(GPIO_GPIO2 | GPIO_GPIO3); + + /* Disable Power 3.3V/5V for CF */ + PA_DWR |= GPIO_GPIO2 | GPIO_GPIO3; + + INTPOL1 |= SA1111_IRQMASK_HI(S1_READY_NINT) | + SA1111_IRQMASK_HI(S1_CD_VALID) | + SA1111_IRQMASK_HI(S1_BVD1_STSCHG); + + nirq = 4; + slots = 2; + } + + for (i = ret = 0; i < nirq; i++) { + ret = request_irq(irqs[i].irq, init->handler, SA_INTERRUPT, + irqs[i].str, NULL); + if (ret) + break; + } + + if (i < nirq) { + printk(KERN_ERR "sa1111_pcmcia: unable to grab IRQ%d (%d)\n", + irqs[i].irq, ret); + while (i--) + free_irq(irqs[i].irq, NULL); + + release_mem_region(_PCCR, 16); + } + + return ret ? -1 : slots; +} + +static int adsbitsy_pcmcia_shutdown(void) +{ + + free_irq(S0_CD_VALID, NULL); + free_irq(S0_BVD1_STSCHG, NULL); + INTPOL1 &= ~(SA1111_IRQMASK_HI(S0_CD_VALID) | SA1111_IRQMASK_HI(S0_BVD1_STSCHG)); + + if (!adsbitsy_smc91111_present()) { + free_irq(S1_CD_VALID, NULL); + free_irq(S1_BVD1_STSCHG, NULL); + INTPOL1 &= ~(SA1111_IRQMASK_HI(S1_CD_VALID) | SA1111_IRQMASK_HI(S1_BVD1_STSCHG)); + } - /* Disable Power 3.3V/5V for PCMCIA/CF */ - PA_DWR |= GPIO_GPIO0 | GPIO_GPIO1 | GPIO_GPIO2 | GPIO_GPIO3; + return 0; +} - /* Why? */ - MECR = 0x09430943; - return sa1111_pcmcia_init(init); +static int adsbitsy_pcmcia_socket_state(struct pcmcia_state_array *state) +{ + unsigned long status; + + if (adsbitsy_smc91111_present()) { + if(state->size<1) return -1; + } + else + if(state->size<2) return -1; + + memset(state->state, 0, + (state->size)*sizeof(struct pcmcia_state)); + + status = PCSR; + + state->state[0].detect = status & PCSR_S0_DETECT ? 0 : 1; + state->state[0].ready = status & PCSR_S0_READY ? 1 : 0; + state->state[0].bvd1 = status & PCSR_S0_BVD1 ? 1 : 0; + state->state[0].bvd2 = status & PCSR_S0_BVD2 ? 1 : 0; + state->state[0].wrprot = status & PCSR_S0_WP ? 1 : 0; + state->state[0].vs_3v = status & PCSR_S0_VS1 ? 0 : 1; + state->state[0].vs_Xv = status & PCSR_S0_VS2 ? 0 : 1; + + if (state->size > 1) { + if (adsbitsy_smc91111_present()) { + // If there is SMC91111 on ADS Bitsy connector board + // it returns not detect/ready/... + state->state[1].detect = 0; + state->state[1].ready = 0; + state->state[1].bvd1 = 0; + state->state[1].bvd2 = 0; + state->state[1].wrprot = 0; + state->state[1].vs_3v = 0; + state->state[1].vs_Xv = 0; + } + else { + state->state[1].detect = status & PCSR_S1_DETECT ? 0 : 1; + state->state[1].ready = status & PCSR_S1_READY ? 1 : 0; + state->state[1].bvd1 = status & PCSR_S1_BVD1 ? 1 : 0; + state->state[1].bvd2 = status & PCSR_S1_BVD2 ? 1 : 0; + state->state[1].wrprot = status & PCSR_S1_WP ? 1 : 0; + state->state[1].vs_3v = status & PCSR_S1_VS1 ? 0 : 1; + state->state[1].vs_Xv = status & PCSR_S1_VS2 ? 0 : 1; + } + } + return 1; } -static int -adsbitsy_pcmcia_configure_socket(const struct pcmcia_configure *conf) +static int adsbitsy_pcmcia_configure_socket(const struct pcmcia_configure *conf) { unsigned int pa_dwr_mask, pa_dwr_set; int ret; @@ -54,10 +182,11 @@ switch (conf->vcc) { default: - case 0: pa_dwr_set = 0; break; + case 0: pa_dwr_set = GPIO_GPIO2 | GPIO_GPIO3; break; case 33: pa_dwr_set = GPIO_GPIO2; break; case 50: pa_dwr_set = GPIO_GPIO3; break; } + break; default: return -1; @@ -83,8 +212,8 @@ struct pcmcia_low_level adsbitsy_pcmcia_ops = { init: adsbitsy_pcmcia_init, - shutdown: sa1111_pcmcia_shutdown, - socket_state: sa1111_pcmcia_socket_state, + shutdown: adsbitsy_pcmcia_shutdown, + socket_state: adsbitsy_pcmcia_socket_state, get_irq_info: sa1111_pcmcia_get_irq_info, configure_socket: adsbitsy_pcmcia_configure_socket, diff -urN orig/drivers/pcmcia/sa1100_adsbitsyplus.c linux/drivers/pcmcia/sa1100_adsbitsyplus.c --- orig/drivers/pcmcia/sa1100_adsbitsyplus.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/pcmcia/sa1100_adsbitsyplus.c Fri Feb 21 15:17:34 2003 @@ -0,0 +1,236 @@ +/* + * drivers/pcmcia/sa1100_adsbitsyplus.c + * + * PCMCIA implementation routines for ADS Bitsy Plus + * + * Created Feb 7, 2003 by Robert Whaley + * + * This file comes from sa1100_adsbitsy.c of Woojung Huh + * + */ +#include +#include +#include + +#include +#include +#include + +#include "sa1100_generic.h" +#include "sa1111_generic.h" + +int adsbitsy_smc91111_present(void); + +#ifndef CONFIG_SMC91111 +#define adsbitsy_smc91111_present() 0 +#endif + +static struct irqs { + int irq; + const char *str; +} irqs[] = { + { S0_CD_VALID, "SA1111 PCMCIA card detect" }, + { S0_BVD1_STSCHG, "SA1111 PCMCIA BVD1" }, + { S1_CD_VALID, "SA1111 CF card detect" }, + { S1_BVD1_STSCHG, "SA1111 CF BVD1" }, +}; + +#define sock0_3_3_reverse_logic() ((ADS_CPLD_IO2 & ADS_IO2_CPLD_REV_MASK) >= ADS_IO2_CPLD_REV_5_MAGIC) + +static int adsbitsyplus_pcmcia_init(struct pcmcia_init *init) +{ + int ret=0; + int nirq = 0; + int slots = 0; + int i; + + /* Set GPIO_A<1:0> to be outputs for PCMCIA power controller: */ + PA_DDR &= ~(GPIO_GPIO0 | GPIO_GPIO1); + + /* Disable Power 3.3V/5V for PCMCIA */ + if (sock0_3_3_reverse_logic()) + PA_DWR = (PA_DWR & ~GPIO_GPIO0) | GPIO_GPIO1; + else + PA_DWR |= GPIO_GPIO0 | GPIO_GPIO1; + + if (!request_mem_region(_PCCR, 512, "PCMCIA")) + return -1; + + + INTPOL1 |= SA1111_IRQMASK_HI(S0_READY_NINT) | + SA1111_IRQMASK_HI(S0_CD_VALID) | + SA1111_IRQMASK_HI(S0_BVD1_STSCHG); + + nirq = 2; + slots = 1; + + if (!adsbitsy_smc91111_present()) { + /* If the SMC91111 is used CF cannot be used */ + /* Set GPIO_A<3:2> to be outputs for CF power controller: */ + PA_DDR &= ~(GPIO_GPIO2 | GPIO_GPIO3); + + /* Disable Power 3.3V/5V for CF */ + PA_DWR |= GPIO_GPIO2 | GPIO_GPIO3; + + INTPOL1 |= SA1111_IRQMASK_HI(S1_READY_NINT) | + SA1111_IRQMASK_HI(S1_CD_VALID) | + SA1111_IRQMASK_HI(S1_BVD1_STSCHG); + + nirq = 4; + slots = 2; + } + + for (i = ret = 0; i < nirq; i++) { + ret = request_irq(irqs[i].irq, init->handler, SA_INTERRUPT, + irqs[i].str, NULL); + if (ret) + break; + } + + if (i < nirq) { + printk(KERN_ERR "sa1111_pcmcia: unable to grab IRQ%d (%d)\n", + irqs[i].irq, ret); + while (i--) + free_irq(irqs[i].irq, NULL); + + release_mem_region(_PCCR, 16); + } + + return ret ? -1 : slots; +} + +static int adsbitsyplus_pcmcia_shutdown(void) +{ + + free_irq(S0_CD_VALID, NULL); + free_irq(S0_BVD1_STSCHG, NULL); + INTPOL1 &= ~(SA1111_IRQMASK_HI(S0_CD_VALID) | SA1111_IRQMASK_HI(S0_BVD1_STSCHG)); + + if (!adsbitsy_smc91111_present()) { + free_irq(S1_CD_VALID, NULL); + free_irq(S1_BVD1_STSCHG, NULL); + INTPOL1 &= ~(SA1111_IRQMASK_HI(S1_CD_VALID) | SA1111_IRQMASK_HI(S1_BVD1_STSCHG)); + } + + return 0; +} + + +static int adsbitsyplus_pcmcia_socket_state(struct pcmcia_state_array *state) +{ + unsigned long status; + + if (adsbitsy_smc91111_present()) { + if(state->size<1) return -1; + } + else + if(state->size<2) return -1; + + memset(state->state, 0, + (state->size)*sizeof(struct pcmcia_state)); + + status = PCSR; + + state->state[0].detect = status & PCSR_S0_DETECT ? 0 : 1; + state->state[0].ready = status & PCSR_S0_READY ? 1 : 0; + state->state[0].bvd1 = status & PCSR_S0_BVD1 ? 1 : 0; + state->state[0].bvd2 = status & PCSR_S0_BVD2 ? 1 : 0; + state->state[0].wrprot = status & PCSR_S0_WP ? 1 : 0; + state->state[0].vs_3v = status & PCSR_S0_VS1 ? 0 : 1; + state->state[0].vs_Xv = status & PCSR_S0_VS2 ? 0 : 1; + + if (state->size > 1) { + if (adsbitsy_smc91111_present()) { + // If there is SMC91111 on ADS Bitsy connector board + // it returns not detect/ready/... + state->state[1].detect = 0; + state->state[1].ready = 0; + state->state[1].bvd1 = 0; + state->state[1].bvd2 = 0; + state->state[1].wrprot = 0; + state->state[1].vs_3v = 0; + state->state[1].vs_Xv = 0; + } + else { + state->state[1].detect = status & PCSR_S1_DETECT ? 0 : 1; + state->state[1].ready = status & PCSR_S1_READY ? 1 : 0; + state->state[1].bvd1 = status & PCSR_S1_BVD1 ? 1 : 0; + state->state[1].bvd2 = status & PCSR_S1_BVD2 ? 1 : 0; + state->state[1].wrprot = status & PCSR_S1_WP ? 1 : 0; + state->state[1].vs_3v = status & PCSR_S1_VS1 ? 0 : 1; + state->state[1].vs_Xv = status & PCSR_S1_VS2 ? 0 : 1; + } + } + return 1; +} + +static int adsbitsyplus_pcmcia_configure_socket(const struct pcmcia_configure *conf) +{ + unsigned int pa_dwr_mask, pa_dwr_set; + int ret; + + switch (conf->sock) { + case 0: + pa_dwr_mask = GPIO_GPIO0 | GPIO_GPIO1; + + if (sock0_3_3_reverse_logic()) { + switch (conf->vcc) { + default: + case 0: pa_dwr_set = GPIO_GPIO1; break; + case 33: pa_dwr_set = GPIO_GPIO0 | GPIO_GPIO1; break; + case 50: pa_dwr_set = 0; break; + } + } else { + switch (conf->vcc) { + default: + case 0: pa_dwr_set = GPIO_GPIO0 | GPIO_GPIO1; break; + case 33: pa_dwr_set = GPIO_GPIO1; break; + case 50: pa_dwr_set = GPIO_GPIO0; break; + } + } + break; + + case 1: + pa_dwr_mask = GPIO_GPIO2 | GPIO_GPIO3; + + switch (conf->vcc) { + default: + case 0: pa_dwr_set = GPIO_GPIO2 | GPIO_GPIO3; break; + case 33: pa_dwr_set = GPIO_GPIO2; break; + case 50: pa_dwr_set = GPIO_GPIO3; break; + } + break; + + default: + return -1; + } + + if (conf->vpp != conf->vcc && conf->vpp != 0) { + printk(KERN_ERR "%s(): CF slot cannot support VPP %u\n", + __FUNCTION__, conf->vpp); + return -1; + } + + ret = sa1111_pcmcia_configure_socket(conf); + if (ret == 0) { + unsigned long flags; + + local_irq_save(flags); + PA_DWR = (PA_DWR & ~pa_dwr_mask) | pa_dwr_set; + local_irq_restore(flags); + } + + return ret; +} + +struct pcmcia_low_level adsbitsyplus_pcmcia_ops = { + init: adsbitsyplus_pcmcia_init, + shutdown: adsbitsyplus_pcmcia_shutdown, + socket_state: adsbitsyplus_pcmcia_socket_state, + get_irq_info: sa1111_pcmcia_get_irq_info, + configure_socket: adsbitsyplus_pcmcia_configure_socket, + + socket_init: sa1111_pcmcia_socket_init, + socket_suspend: sa1111_pcmcia_socket_suspend, +}; + diff -urN orig/drivers/pcmcia/sa1100_freebird.c linux/drivers/pcmcia/sa1100_freebird.c --- orig/drivers/pcmcia/sa1100_freebird.c Mon Aug 5 13:31:16 2002 +++ linux/drivers/pcmcia/sa1100_freebird.c Fri Dec 20 15:08:10 2002 @@ -67,9 +67,6 @@ if(state_array->size<2) return -1; - memset(state_array->state, 0, - (state_array->size)*sizeof(struct pcmcia_state)); - levels = LINKUP_PRS; //printk("LINKUP_PRS=%x \n",levels); diff -urN orig/drivers/pcmcia/sa1100_generic.c linux/drivers/pcmcia/sa1100_generic.c --- orig/drivers/pcmcia/sa1100_generic.c Mon Aug 5 13:31:16 2002 +++ linux/drivers/pcmcia/sa1100_generic.c Fri Feb 21 15:17:34 2003 @@ -993,10 +993,18 @@ if(machine_is_graphicsmaster()) pcmcia_low_level = &graphicsmaster_pcmcia_ops; #endif +#ifdef CONFIG_SA1100_ADSAGC + if(machine_is_adsagc()) + pcmcia_low_level = &graphicsmaster_pcmcia_ops; +#endif #ifdef CONFIG_SA1100_ADSBITSY if(machine_is_adsbitsy()) pcmcia_low_level = &adsbitsy_pcmcia_ops; #endif +#ifdef CONFIG_SA1100_ADSBITSYPLUS + if(machine_is_adsbitsyplus()) + pcmcia_low_level = &adsbitsyplus_pcmcia_ops; +#endif #ifdef CONFIG_SA1100_STORK if(machine_is_stork()) pcmcia_low_level = &stork_pcmcia_ops; @@ -1068,7 +1076,7 @@ * We initialize the MECR to default values here, because we are * not guaranteed to see a SetIOMap operation at runtime. */ - mecr = 0; + mecr = MECR; clock = cpufreq_get(0); diff -urN orig/drivers/pcmcia/sa1100_graphicsclient.c linux/drivers/pcmcia/sa1100_graphicsclient.c --- orig/drivers/pcmcia/sa1100_graphicsclient.c Mon Aug 5 13:31:16 2002 +++ linux/drivers/pcmcia/sa1100_graphicsclient.c Fri Feb 21 15:17:35 2003 @@ -3,6 +3,8 @@ * * PCMCIA implementation routines for Graphics Client Plus * + * Nov/14/01 Woojung + * Set MECR at initializing time * 9/12/01 Woojung * Turn power OFF at startup * 1/31/2001 Woojung Huh @@ -19,11 +21,6 @@ #include #include "sa1100_generic.h" -#error This is broken! - -#define S0_CD_IRQ 60 // Socket 0 Card Detect IRQ -#define S0_STS_IRQ 55 // Socket 0 PCMCIA IRQ - static volatile unsigned long *PCMCIA_Status = ((volatile unsigned long *) ADS_p2v(_ADS_CS_STATUS)); @@ -46,20 +43,23 @@ *PCMCIA_Power &= ~0x03; /* Register interrupts */ - irq = S0_CD_IRQ; + irq = IRQ_GRAPHICSCLIENT_S0_CD; res = request_irq(irq, init->handler, SA_INTERRUPT, "PCMCIA 0 CD", NULL); if (res < 0) { - printk(KERN_ERR "%s: Request for IRQ %lu failed\n", __FUNCTION__, irq); + printk(KERN_ERR "%s: Request for IRQ %u failed\n", __FUNCTION__, irq); return -1; } + // Nov/14/01 WH + MECR = 0x00000943; + return 1; // 1 PCMCIA Slot } static int gcplus_pcmcia_shutdown(void) { /* disable IRQs */ - free_irq( S0_CD_IRQ, NULL); + free_irq( IRQ_GRAPHICSCLIENT_S0_CD, NULL); /* Shutdown PCMCIA power */ mdelay(2); // 2msec @@ -74,9 +74,6 @@ if(state_array->size<1) return -1; - memset(state_array->state, 0, - (state_array->size)*sizeof(struct pcmcia_state)); - levels=*PCMCIA_Status; state_array->state[0].detect=(levels & ADS_CS_ST_A_CD)?1:0; @@ -96,7 +93,7 @@ return -1; if (info->sock == 0) - info->irq = S0_STS_IRQ; + info->irq = IRQ_GRAPHICSCLIENT_S0_STS; return 0; } @@ -142,6 +139,11 @@ mdelay(30); restore_flags(flags); + + if (configure->irq) + enable_irq(IRQ_GRAPHICSCLIENT_S0_STS); + else + disable_irq(IRQ_GRAPHICSCLIENT_S0_STS); return 0; } diff -urN orig/drivers/pcmcia/sa1100_graphicsmaster.c linux/drivers/pcmcia/sa1100_graphicsmaster.c --- orig/drivers/pcmcia/sa1100_graphicsmaster.c Mon Aug 5 13:31:16 2002 +++ linux/drivers/pcmcia/sa1100_graphicsmaster.c Thu Dec 12 22:26:34 2002 @@ -12,13 +12,13 @@ #include #include +#include #include "sa1100_generic.h" #include "sa1111_generic.h" static int graphicsmaster_pcmcia_init(struct pcmcia_init *init) { - int return_val=0; /* Set GPIO_A<3:0> to be outputs for PCMCIA/CF power controller: */ PA_DDR &= ~(GPIO_GPIO0 | GPIO_GPIO1 | GPIO_GPIO2 | GPIO_GPIO3); @@ -26,9 +26,6 @@ /* Disable Power 3.3V/5V for PCMCIA/CF */ PA_DWR |= GPIO_GPIO0 | GPIO_GPIO1 | GPIO_GPIO2 | GPIO_GPIO3; - /* why? */ - MECR = 0x09430943; - return sa1111_pcmcia_init(init); } @@ -59,6 +56,10 @@ case 33: pa_dwr_set = GPIO_GPIO3; break; case 50: pa_dwr_set = GPIO_GPIO2; break; } + break; + + default: + return -1; } if (conf->vpp != conf->vcc && conf->vpp != 0) { diff -urN orig/drivers/pcmcia/sa1100_h3600.c linux/drivers/pcmcia/sa1100_h3600.c --- orig/drivers/pcmcia/sa1100_h3600.c Mon Aug 5 13:31:16 2002 +++ linux/drivers/pcmcia/sa1100_h3600.c Mon Aug 5 13:44:45 2002 @@ -29,6 +29,9 @@ set_GPIO_IRQ_edge(GPIO_H3600_PCMCIA_IRQ0 | GPIO_H3600_PCMCIA_IRQ1, GPIO_FALLING_EDGE); + set_GPIO_IRQ_edge(GPIO_H3600_PCMCIA_CD0 | GPIO_H3600_PCMCIA_CD1, + GPIO_NO_EDGES); + /* * Register interrupts */ diff -urN orig/drivers/pcmcia/sa1100_jornada720.c linux/drivers/pcmcia/sa1100_jornada720.c --- orig/drivers/pcmcia/sa1100_jornada720.c Mon Aug 5 13:31:16 2002 +++ linux/drivers/pcmcia/sa1100_jornada720.c Mon Dec 30 14:30:07 2002 @@ -8,6 +8,7 @@ #include #include +#include #include "sa1100_generic.h" #include "sa1111_generic.h" @@ -88,7 +89,7 @@ local_irq_save(flags); PA_DWR = (PA_DWR & ~pa_dwr_mask) | pa_dwr_set; - locla_irq_restore(flags); + local_irq_restore(flags); } return ret; diff -urN orig/drivers/pcmcia/sa1100_pangolin.c linux/drivers/pcmcia/sa1100_pangolin.c --- orig/drivers/pcmcia/sa1100_pangolin.c Mon Aug 5 13:31:16 2002 +++ linux/drivers/pcmcia/sa1100_pangolin.c Fri Dec 20 15:08:10 2002 @@ -53,9 +53,6 @@ if(state_array->size<2) return -1; - memset(state_array->state, 0, - (state_array->size)*sizeof(struct pcmcia_state)); - levels=GPLR; #ifndef CONFIG_SA1100_PANGOLIN_PCMCIA_IDE state_array->state[1].detect=((levels & GPIO_PCMCIA_CD)==0)?1:0; diff -urN orig/drivers/pcmcia/sa1100_shannon.c linux/drivers/pcmcia/sa1100_shannon.c --- orig/drivers/pcmcia/sa1100_shannon.c Mon Aug 5 13:31:16 2002 +++ linux/drivers/pcmcia/sa1100_shannon.c Fri Dec 20 15:08:10 2002 @@ -53,9 +53,6 @@ { unsigned long levels; - memset(state_array->state, 0, - state_array->size * sizeof(struct pcmcia_state)); - levels = GPLR; state_array->state[0].detect = (levels & SHANNON_GPIO_EJECT_0) ? 0 : 1; diff -urN orig/drivers/pcmcia/sa1100_simpad.c linux/drivers/pcmcia/sa1100_simpad.c --- orig/drivers/pcmcia/sa1100_simpad.c Mon Aug 5 13:31:16 2002 +++ linux/drivers/pcmcia/sa1100_simpad.c Fri Dec 20 15:08:10 2002 @@ -63,9 +63,6 @@ if(state_array->size<2) return -1; - memset(state_array->state, 0, - (state_array->size)*sizeof(struct pcmcia_state)); - levels=GPLR; state_array->state[1].detect=((levels & GPIO_CF_CD)==0)?1:0; diff -urN orig/drivers/pcmcia/sa1100_stork.c linux/drivers/pcmcia/sa1100_stork.c --- orig/drivers/pcmcia/sa1100_stork.c Mon Aug 5 13:31:16 2002 +++ linux/drivers/pcmcia/sa1100_stork.c Fri Dec 20 15:08:10 2002 @@ -80,9 +80,6 @@ if(state_array->size<2) return -1; - memset(state_array->state, 0, - (state_array->size)*sizeof(struct pcmcia_state)); - levels=GPLR; if (debug > 1) diff -urN orig/drivers/pcmcia/sa1100_yopy.c linux/drivers/pcmcia/sa1100_yopy.c --- orig/drivers/pcmcia/sa1100_yopy.c Mon Aug 5 13:31:16 2002 +++ linux/drivers/pcmcia/sa1100_yopy.c Fri Dec 20 15:08:11 2002 @@ -73,9 +73,6 @@ if (state_array->size != 1) return -1; - memset(state_array->state, 0, - state_array->size * sizeof(struct pcmcia_state)); - levels = GPLR; state_array->state[0].detect = (levels & GPIO_CF_CD) ? 0 : 1; diff -urN orig/drivers/pld/Makefile linux/drivers/pld/Makefile --- orig/drivers/pld/Makefile Thu Jan 1 01:00:00 1970 +++ linux/drivers/pld/Makefile Sun Jun 16 12:54:24 2002 @@ -0,0 +1,28 @@ +# +# Makefile for the kernel pld device drivers. +# +# Note! Dependencies are done automagically by 'make dep', which also +# removes any old dependencies. DON'T put your own dependencies here +# unless it's something special (ie not a .c file). +# +# Note 2! The CFLAGS definitions are now inherited from the +# parent makes.. +# +# $Id: $ +# + +O_TARGET := pld.o + +export-objs := pld_hotswap.o +obj-y := +obj-m := +obj-n := +obj- := + +obj-$(CONFIG_PLD) += pld_epxa.o +obj-$(CONFIG_PLD_HOTSWAP) += pld_hotswap.o + +include $(TOPDIR)/Rules.make + +fastdep: + diff -urN orig/drivers/pld/pld_epxa.c linux/drivers/pld/pld_epxa.c --- orig/drivers/pld/pld_epxa.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/pld/pld_epxa.c Thu Feb 27 23:20:12 2003 @@ -0,0 +1,375 @@ + +/* + * drivers/char/epxapld.c + * + * Copyright (C) 2001 Altera Corporation + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#define PLD_CONF00_TYPE (volatile unsigned int *) +#define MODE_CTRL00_TYPE (volatile unsigned int *) +//#define DEBUG(x) x +#define DEBUG(x) + +#include +#include +#ifdef CONFIG_PLD_HOTSWAP +#include +#endif +#include + +/* + * Macros + */ + + +#define PLD_BASE (IO_ADDRESS(EXC_PLD_CONFIG00_BASE)) +#define CLOCK_DIV_RATIO ((1 + EXC_AHB2_CLK_FREQUENCY/32000000) & CONFIG_CONTROL_CLOCK_RATIO_MSK) +/* + * STRUCTURES + */ + + +struct pld_sbihdr{ + unsigned int fpos; + unsigned int temp; +}; + +static DECLARE_MUTEX(pld_sem); + + +static void lock_pld (void) +{ + /* Lock the pld i/f */ + unsigned int tmp; + + tmp = readl(CONFIG_CONTROL(PLD_BASE)); + tmp |= CONFIG_CONTROL_LK_MSK; + + writel(tmp,CONFIG_CONTROL(PLD_BASE)); +} + +static void unlock_excalibur_pld (void) +{ + /* Unlock the pld i/f */ + + if (readl(CONFIG_CONTROL(PLD_BASE)) & CONFIG_CONTROL_LK_MSK ){ + writel(CONFIG_UNLOCK_MAGIC, CONFIG_UNLOCK(PLD_BASE)); + while (readl(CONFIG_CONTROL(PLD_BASE)) & CONFIG_CONTROL_LK_MSK); + } +} + + +static int place_pld_into_configure_mode (void) +{ + unsigned int tmp; + + + if( readl(CONFIG_CONTROL(PLD_BASE)) & CONFIG_CONTROL_CO_MSK ){ + /* + * Already being configured!!! + */ + printk(KERN_WARNING "pld0: Device already being configured!\n"); + return -EBUSY; + } + + /* Set up the config clock */ + + writel(CLOCK_DIV_RATIO,CONFIG_CONTROL_CLOCK(PLD_BASE)); + while(readl(CONFIG_CONTROL_CLOCK(PLD_BASE)) + !=CLOCK_DIV_RATIO); + /* Tell the device we wish to configure it */ + tmp = readl(CONFIG_CONTROL(PLD_BASE)); + tmp |= CONFIG_CONTROL_CO_MSK; + writel(tmp,CONFIG_CONTROL(PLD_BASE)); + + + /* + * Wait for the busy bit to clear then check for errors. + */ + + while((tmp=readl(CONFIG_CONTROL(PLD_BASE))&CONFIG_CONTROL_B_MSK )); + + if ( tmp & CONFIG_CONTROL_E_MSK ){ + if ( tmp & CONFIG_CONTROL_ES_0_MSK ){ + /* Already being programmed via JTAG */ + printk(KERN_WARNING "pld0:JTAG configuration alreay in progress\n"); + return -EBUSY; + + } + if ( tmp & CONFIG_CONTROL_ES_1_MSK ){ + /* No config clock configured */ + printk(KERN_ERR "pld0:No config clock!\n"); + BUG(); + return -EBUSY; + } + if ( tmp & CONFIG_CONTROL_ES_2_MSK ){ + /* Already being programmed via External device */ + printk(KERN_WARNING "pld0:JTAG configuration alreay in progress\n"); + return -EBUSY; + } + } + + return 0; +} + + +static int write_pld_data_word(unsigned int config_data) +{ + unsigned int tmp; + + do{ + tmp = readl(CONFIG_CONTROL(PLD_BASE)); + } + while ( ( tmp & CONFIG_CONTROL_B_MSK ) && + !( tmp & CONFIG_CONTROL_E_MSK )); + + if ( tmp & CONFIG_CONTROL_E_MSK ){ + printk("pld0: Error writing pld data, CONFIG_CONTROL=%#x\n",tmp); + + return -EILSEQ; + } + + writel(config_data,CONFIG_CONTROL_DATA(PLD_BASE)); + return 0; + +} + + +static int wait_for_device_to_configure (void) +{ + int i=0x10000; + + while(readl(CONFIG_CONTROL(PLD_BASE)) & CONFIG_CONTROL_B_MSK); + + /* + * Wait for the config bit (CO) to go low, indicating that everything + * is Ok. If it doesn't, assume that is screwed up somehow and + * clear the CO bit to abort the configuration. + */ + + while(readl(CONFIG_CONTROL(PLD_BASE)) & CONFIG_CONTROL_B_MSK); + + while (i&&(readl(CONFIG_CONTROL(PLD_BASE)) & CONFIG_CONTROL_CO_MSK)){ + i--; + } + + if (!i){ + writel(0,CONFIG_CONTROL(PLD_BASE)); + printk(KERN_WARNING "pld0: Invalid PLD config data\n"); + return -EILSEQ; + } + + return 0; +} + + + +static int pld_open(struct inode* inode, struct file *filep) +{ + + struct pld_sbihdr* sbihdr; + + /* Check the device minor number */ + if(minor(inode->i_rdev)){ + DEBUG(printk("pld0: minor=%d",minor(inode->i_rdev));) + return -ENODEV; + } + + /* Create our private data and link it to the file structure */ + sbihdr=kmalloc(sizeof(struct pld_sbihdr),GFP_KERNEL); + + if(!sbihdr) + return -ENOMEM; + + filep->private_data=sbihdr; + + sbihdr->fpos=0; + sbihdr->temp=0; + return 0; +} + +static int pld_release(struct inode* inode, struct file *filep){ + int ret_code; + + kfree(filep->private_data); + ret_code=wait_for_device_to_configure(); + lock_pld(); + return ret_code; +} + +static ssize_t pld_write(struct file* filep, const char* data, size_t count, loff_t* ppos){ + + struct pld_sbihdr* sbihdr=filep->private_data; + int bytes_left=count; + int result; + DEBUG(int i=0); + + + /* Can't seek (pwrite) on pld. */ + if (ppos != &filep->f_pos) + return -ESPIPE; + + + /* Check access to the whole are in one go */ + if(!access_ok(VERIFY_READ,(const void*)data, count)){ + return -EFAULT; + } + + /* + * We now lock against writes. + */ + if (down_interruptible(&pld_sem)) + return -ERESTARTSYS; + + if(!sbihdr->fpos){ + /* + * unlock the pld and place in configure mode + */ + unlock_excalibur_pld(); + result=place_pld_into_configure_mode(); + if(result) + return result; + } + DEBUG(printk("data= %#x count=%#x 0ffset=%#x\n",*data, count, *ppos)); + + while(bytes_left){ + char tmp; + __get_user(tmp,data); + /* read our header ! */ + sbihdr->temp|=tmp << (8*(sbihdr->fpos&3)); + if((sbihdr->fpos&3)==3){ + if(write_pld_data_word(sbihdr->temp)) + { + DEBUG(printk("pos=%d\n",sbihdr->fpos);) + return -EILSEQ; + } + DEBUG(if(i<10){) + DEBUG(printk("fpos2 :%#x data=%#x\n",sbihdr->fpos,sbihdr->temp)); + DEBUG(i++); + DEBUG(}); + sbihdr->temp=0; + DEBUG(words_written++); + } + sbihdr->fpos++; + data++; + bytes_left--; + } + + up(&pld_sem); + return (count); +} + +int pld_ioctl(struct inode *inode, struct file *filep, unsigned int cmd, unsigned long arg) +{ + + switch(cmd){ + +#ifdef CONFIG_PLD_HOTSWAP + case PLD_IOC_ADD_PLD_DEV: + { + struct pldhs_dev_desc desc; + struct pldhs_dev_info info; + char *name; + void *data; + int result=0; + + result=copy_from_user(&desc,(const void*)arg,sizeof(struct pldhs_dev_desc)); + if(result) + return -EFAULT; + result=copy_from_user(&info,(const void*)desc.info,sizeof(struct pldhs_dev_info)); + if(result) + return -EFAULT; + name=kmalloc(info.nsize,GFP_KERNEL); + if(!name) + return -ENOMEM; + + result=copy_from_user(name,(const void*)desc.name,info.nsize); + if(result){ + result=-EFAULT; + goto ioctl_out; + } + + data=kmalloc(info.pssize,GFP_KERNEL); + if(!data){ + result=-ENOMEM; + goto ioctl_out; + } + + result=copy_from_user(data,(const void*)desc.data,info.pssize); + if(result){ + result=-EFAULT; + goto ioctl_out1; + } + result=pldhs_add_device(&info,name,data); + + ioctl_out1: + kfree(data); + ioctl_out: + kfree(name); + return result; + + } + + case PLD_IOC_REMOVE_PLD_DEVS: + pldhs_remove_devices(); + return 0; + + case PLD_IOC_SET_INT_MODE: + if(cmd==3){ + printk(KERN_INFO "Interrupt mode set to 3 (Six individual interrupts)\n"); + return 0; + }else{ + printk(KERN_INFO "There is no interrupt handler available for this mode (%d). You will need to add one\n to implement whatever scheme you require\n"); + return -EACCES; + } +#endif + default: + return -ENOTTY; + } +} + + +static struct file_operations pld_fops={ + write: pld_write, + ioctl: pld_ioctl, + open: pld_open, + release: pld_release +}; + +static int __init pld_init(void){ + int major; + major=register_chrdev(0,"pld",&pld_fops); + printk(KERN_INFO "Using PLD major num=%d\n",major); + if (major<0){ + return major; + } + return 0; +} + +__initcall(pld_init); diff -urN orig/drivers/pld/pld_hotswap.c linux/drivers/pld/pld_hotswap.c --- orig/drivers/pld/pld_hotswap.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/pld/pld_hotswap.c Mon Aug 5 22:25:08 2002 @@ -0,0 +1,188 @@ +/* + * linux/drivers/pld/pld_hotswap.c + * + * Pld driver for Altera EPXA Excalibur devices + * + * + * Copyright 2001 Altera Corporation (cdavies@altera.com) + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + * + * $Id: $ + * + */ + +/* + * pld_hotswap ops contains the basic operation required for adding + * and removing devices from the system. + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + + +static struct pld_hotswap_ops pldhs_driver_list={ + list: LIST_HEAD_INIT(pldhs_driver_list.list), + name: "", +}; + +static spinlock_t list_lock=SPIN_LOCK_UNLOCKED; + + + +static struct pld_hotswap_ops * pldhs_find_driver(char* name) +{ + struct pld_hotswap_ops * ptr; + struct list_head *list_pos; + + spin_lock(&list_lock); + list_for_each(list_pos,&pldhs_driver_list.list){ + ptr=(struct pld_hotswap_ops *)list_pos; + if(!strcmp(name, ptr->name)){ + spin_unlock(&list_lock); + return ptr; + + } + } + spin_unlock(&list_lock); + return 0; +} + + + +int pldhs_register_driver(struct pld_hotswap_ops *drv) +{ + + /* Check that the device is not already on the list + * if so, do nothing */ + if(pldhs_find_driver(drv->name)){ + return 0; + } + + printk(KERN_INFO "PLD: Registering hotswap driver %s\n",drv->name); + /* Add this at the start of the list */ + spin_lock(&list_lock); + list_add((struct list_head*)drv,&pldhs_driver_list.list); + spin_unlock(&list_lock); + + return 0; +} + +int pldhs_unregister_driver(char *name) +{ + struct pld_hotswap_ops *ptr; + + ptr=pldhs_find_driver(name); + if(!ptr){ + return -ENODEV; + } + + printk(KERN_INFO "PLD: Unregistering hotswap driver%s\n",name); + spin_lock(&list_lock); + list_del((struct list_head*)ptr); + spin_unlock(&list_lock); + + return 0; +} + +int pldhs_add_device(struct pldhs_dev_info* dev_info,char *drv_name, void* dev_ps_data) +{ + struct pld_hotswap_ops * ptr; + + ptr=pldhs_find_driver(drv_name); + + if(!ptr){ + /* try requesting this module*/ + request_module(drv_name); + /* is the driver there now? */ + ptr=pldhs_find_driver(drv_name); + if(!ptr){ + printk("pld hotswap: Failed to load a driver for %s\n",drv_name); + return -ENODEV; + } + } + + if(!ptr->add_device){ + printk(KERN_WARNING "pldhs: no add_device() function for driver %s\n",drv_name); + return 0; + } + + return ptr->add_device(dev_info,dev_ps_data); +} + +int pldhs_remove_devices(void) +{ + struct list_head *list_pos; + struct pld_hotswap_ops * ptr; + + + spin_lock(&list_lock); + list_for_each(list_pos,&pldhs_driver_list.list){ + ptr=(struct pld_hotswap_ops *)list_pos; + if(ptr->remove_devices) + ptr->remove_devices(); + + } + spin_unlock(&list_lock); + + return 0; +} + +#ifdef CONFIG_PROC_FS +int pldhs_read_proc(char* buf,char** start,off_t offset,int count,int *eof,void *data){ + + + struct list_head *list_pos; + struct pld_hotswap_ops * ptr; + int i,len=0; + + *start=buf; + spin_lock(&list_lock); + list_for_each(list_pos,&pldhs_driver_list.list){ + ptr=(struct pld_hotswap_ops *)list_pos; + if(ptr->proc_read){ + i=ptr->proc_read(buf,start,offset,count,eof,data); + count-=i; + len+=i; + *start+=i; + } + } + spin_unlock(&list_lock); + + *start=NULL; + *eof=1; + return len; +} + +void __init pldhs_init(void){ + create_proc_read_entry("pld",0,NULL,pldhs_read_proc,NULL); +} + +__initcall(pldhs_init); +#endif + +EXPORT_SYMBOL(pldhs_register_driver); +EXPORT_SYMBOL(pldhs_unregister_driver); diff -urN orig/drivers/scsi/Makefile linux/drivers/scsi/Makefile --- orig/drivers/scsi/Makefile Mon Aug 5 13:31:18 2002 +++ linux/drivers/scsi/Makefile Mon Nov 11 16:19:14 2002 @@ -21,7 +21,7 @@ O_TARGET := scsidrv.o -export-objs := scsi_syms.o 53c700.o +export-objs := scsi_syms.o 53c700.o scsi_error.o mod-subdirs := pcmcia ../acorn/scsi diff -urN orig/drivers/scsi/scsi.c linux/drivers/scsi/scsi.c --- orig/drivers/scsi/scsi.c Fri Oct 25 17:39:14 2002 +++ linux/drivers/scsi/scsi.c Fri Oct 25 17:44:55 2002 @@ -1330,14 +1330,10 @@ */ int scsi_retry_command(Scsi_Cmnd * SCpnt) { - memcpy((void *) SCpnt->cmnd, (void *) SCpnt->data_cmnd, - sizeof(SCpnt->data_cmnd)); - SCpnt->request_buffer = SCpnt->buffer; - SCpnt->request_bufflen = SCpnt->bufflen; - SCpnt->use_sg = SCpnt->old_use_sg; - SCpnt->cmd_len = SCpnt->old_cmd_len; - SCpnt->sc_data_direction = SCpnt->sc_old_data_direction; - SCpnt->underflow = SCpnt->old_underflow; + /* + * Restore the SCSI command state. + */ + scsi_setup_cmd_retry(SCpnt); /* * Zero the sense information from the last time we tried diff -urN orig/drivers/scsi/scsi.h linux/drivers/scsi/scsi.h --- orig/drivers/scsi/scsi.h Fri Oct 25 17:39:14 2002 +++ linux/drivers/scsi/scsi.h Fri Oct 25 17:45:19 2002 @@ -474,6 +474,7 @@ int sectors); extern struct Scsi_Device_Template *scsi_get_request_dev(struct request *); extern int scsi_init_cmd_errh(Scsi_Cmnd * SCpnt); +extern void scsi_setup_cmd_retry(Scsi_Cmnd *SCpnt); extern int scsi_insert_special_cmd(Scsi_Cmnd * SCpnt, int); extern void scsi_io_completion(Scsi_Cmnd * SCpnt, int good_sectors, int block_sectors); @@ -597,6 +598,7 @@ unsigned changed:1; /* Data invalid due to media change */ unsigned busy:1; /* Used to prevent races */ unsigned lockable:1; /* Able to prevent media removal */ + unsigned locked:1; /* Media removal disabled */ unsigned borken:1; /* Tell the Seagate driver to be * painfully slow on this device */ unsigned tagged_supported:1; /* Supports SCSI-II tagged queuing */ diff -urN orig/drivers/scsi/scsi_dma.c linux/drivers/scsi/scsi_dma.c --- orig/drivers/scsi/scsi_dma.c Wed Feb 27 14:25:13 2002 +++ linux/drivers/scsi/scsi_dma.c Tue Feb 26 17:39:24 2002 @@ -30,6 +30,15 @@ typedef unsigned char FreeSectorBitmap; #elif SECTORS_PER_PAGE <= 32 typedef unsigned int FreeSectorBitmap; +#elif SECTORS_PER_PAGE <= 64 +#if 0 +typedef unsigned long long FreeSectorBitmap; +#else +typedef struct { + unsigned long l,h; +} FreeSectorBitmap; +#define LARGE_MALLOC +#endif #else #error You lose. #endif @@ -69,6 +78,7 @@ * to allocate more memory in order to be able to write the * data to disk, you would wedge the system. */ +#ifndef LARGE_MALLOC void *scsi_malloc(unsigned int len) { unsigned int nbits, mask; @@ -166,6 +176,97 @@ } panic("scsi_free:Bad offset"); } + +#else + +void *scsi_malloc(unsigned int len) +{ + unsigned int nbits; + unsigned long maskl, maskh, flags; + FreeSectorBitmap *fsb; + int i; + + if (len % SECTOR_SIZE != 0 || len > PAGE_SIZE) + return NULL; + + save_flags_cli (flags); + nbits = len >> 9; + if (nbits < 32) { + maskl = (1 << nbits) - 1; + maskh = 0; + } else { + maskl = (unsigned long)-1; + maskh = (1 << (nbits - 32)) - 1; + } + + fsb = dma_malloc_freelist; + + for (i = 0; i < dma_sectors / SECTORS_PER_PAGE; i++) { + unsigned long mml, mmh; + int j; + mml = maskl; + mmh = maskh; + j = 0; + do { + if ((fsb->l & mml) == 0 && (fsb->h & mmh) == 0) { + fsb->h |= mmh; + fsb->l |= mml; + restore_flags (flags); + scsi_dma_free_sectors -= nbits; +#ifdef DEBUG + printk("SMalloc: %d %p\n",len, dma_malloc_pages[i] + (j << 9)); +#endif + return (void *) ((unsigned long) dma_malloc_pages[i] + (j << 9)); + } + mmh = (mmh << 1) | (mml >> 31); + mml <<= 1; + j++; + } while (!(mmh & (1 << 31))); + fsb ++; + } + return NULL; /* Nope. No more */ +} + +int scsi_free(void *obj, unsigned int len) +{ + unsigned int page, sector, nbits; + unsigned long maskl, maskh, flags; + +#ifdef DEBUG + printk("scsi_free %p %d\n",obj, len); +#endif + + for (page = 0; page < dma_sectors / SECTORS_PER_PAGE; page++) { + unsigned long page_addr = (unsigned long) dma_malloc_pages[page]; + if ((unsigned long) obj >= page_addr && + (unsigned long) obj < page_addr + PAGE_SIZE) { + sector = (((unsigned long) obj) - page_addr) >> 9; + nbits = len >> 9; + if (nbits < 32) { + maskl = (1 << nbits) - 1; + maskh = 0; + } else { + maskl = (unsigned long)-1; + maskh = (1 << (nbits - 32)) - 1; + } + if ((sector + nbits) > SECTORS_PER_PAGE) + panic ("scsi_free:Bad memory alignment"); + maskh = (maskh << sector) | (maskl >> (32 - sector)); + maskl = maskl << sector; + save_flags_cli(flags); + if (((dma_malloc_freelist[page].l & maskl) != maskl) || + ((dma_malloc_freelist[page].h & maskh) != maskh)) + panic("scsi_free:Trying to free unused memory"); + scsi_dma_free_sectors += nbits; + dma_malloc_freelist[page].l &= ~maskl; + dma_malloc_freelist[page].h &= ~maskh; + restore_flags(flags); + return 0; + } + } + panic("scsi_free:Bad offset"); +} +#endif /* diff -urN orig/drivers/scsi/scsi_error.c linux/drivers/scsi/scsi_error.c --- orig/drivers/scsi/scsi_error.c Fri Oct 25 17:39:14 2002 +++ linux/drivers/scsi/scsi_error.c Fri Oct 25 17:45:44 2002 @@ -35,6 +35,8 @@ #include "hosts.h" #include "constants.h" +#include /* grr */ + /* * We must always allow SHUTDOWN_SIGS. Even if we are not a module, * the host drivers that we are using may be loaded as modules, and @@ -49,6 +51,13 @@ */ #define SHUTDOWN_SIGS (sigmask(SIGHUP)) +/* + * The number of times we retry a REQUEST SENSE and TEST UNIT READY + * respectively. This is arbitary. + */ +#define SENSE_RETRIES 5 +#define TUR_RETRIES 5 + #ifdef DEBUG #define SENSE_TIMEOUT SCSI_TIMEOUT #define ABORT_TIMEOUT SCSI_TIMEOUT @@ -385,16 +394,12 @@ */ STATIC int scsi_eh_retry_command(Scsi_Cmnd * SCpnt) { - memcpy((void *) SCpnt->cmnd, (void *) SCpnt->data_cmnd, - sizeof(SCpnt->data_cmnd)); - SCpnt->request_buffer = SCpnt->buffer; - SCpnt->request_bufflen = SCpnt->bufflen; - SCpnt->use_sg = SCpnt->old_use_sg; - SCpnt->cmd_len = SCpnt->old_cmd_len; - SCpnt->sc_data_direction = SCpnt->sc_old_data_direction; - SCpnt->underflow = SCpnt->old_underflow; + int tries = 0; - scsi_send_eh_cmnd(SCpnt, SCpnt->timeout_per_command); + do { + scsi_setup_cmd_retry(SCpnt); + scsi_send_eh_cmnd(SCpnt, SCpnt->timeout_per_command); + } while (SCpnt->eh_state == NEEDS_RETRY && tries++ < SCpnt->allowed); /* * Hey, we are done. Let's look to see what happened. @@ -421,16 +426,10 @@ static unsigned char generic_sense[6] = {REQUEST_SENSE, 0, 0, 0, 255, 0}; unsigned char scsi_result0[256], *scsi_result = NULL; - int saved_result; + int saved_result, tries; ASSERT_LOCK(&io_request_lock, 0); - memcpy((void *) SCpnt->cmnd, (void *) generic_sense, - sizeof(generic_sense)); - - if (SCpnt->device->scsi_level <= SCSI_2) - SCpnt->cmnd[1] = SCpnt->lun << 5; - scsi_result = (!SCpnt->host->hostt->unchecked_isa_dma) ? &scsi_result0[0] : kmalloc(512, GFP_ATOMIC | GFP_DMA); @@ -438,24 +437,41 @@ printk("cannot allocate scsi_result in scsi_request_sense.\n"); return FAILED; } - /* - * Zero the sense buffer. Some host adapters automatically always request - * sense, so it is not a good idea that SCpnt->request_buffer and - * SCpnt->sense_buffer point to the same address (DB). - * 0 is not a valid sense code. - */ - memset((void *) SCpnt->sense_buffer, 0, sizeof(SCpnt->sense_buffer)); - memset((void *) scsi_result, 0, 256); saved_result = SCpnt->result; - SCpnt->request_buffer = scsi_result; - SCpnt->request_bufflen = 256; - SCpnt->use_sg = 0; - SCpnt->cmd_len = COMMAND_SIZE(SCpnt->cmnd[0]); - SCpnt->sc_data_direction = SCSI_DATA_READ; - SCpnt->underflow = 0; - scsi_send_eh_cmnd(SCpnt, SENSE_TIMEOUT); + tries = 0; + do { + memcpy((void *) SCpnt->cmnd, (void *) generic_sense, + sizeof(generic_sense)); + + if (SCpnt->device->scsi_level <= SCSI_2) + SCpnt->cmnd[1] = SCpnt->lun << 5; + + /* + * Zero the sense buffer. Some host adapters automatically + * always request sense, so it is not a good idea that + * SCpnt->request_buffer and SCpnt->sense_buffer point to + * the same address (DB). 0 is not a valid sense code. + */ + memset((void *) SCpnt->sense_buffer, 0, + sizeof(SCpnt->sense_buffer)); + memset((void *) scsi_result, 0, 256); + + SCpnt->request_buffer = scsi_result; + SCpnt->request_bufflen = 256; + SCpnt->use_sg = 0; + SCpnt->cmd_len = COMMAND_SIZE(SCpnt->cmnd[0]); + SCpnt->sc_data_direction = SCSI_DATA_READ; + SCpnt->underflow = 0; + + scsi_send_eh_cmnd(SCpnt, SENSE_TIMEOUT); + /* + * If the SCSI device responded with "logical unit + * is in process of becoming ready", we need to + * retry this command. + */ + } while (SCpnt->eh_state == NEEDS_RETRY && tries++ < SENSE_RETRIES); /* Last chance to have valid sense data */ if (!scsi_sense_valid(SCpnt)) @@ -470,15 +486,8 @@ * When we eventually call scsi_finish, we really wish to complete * the original request, so let's restore the original data. (DB) */ - memcpy((void *) SCpnt->cmnd, (void *) SCpnt->data_cmnd, - sizeof(SCpnt->data_cmnd)); SCpnt->result = saved_result; - SCpnt->request_buffer = SCpnt->buffer; - SCpnt->request_bufflen = SCpnt->bufflen; - SCpnt->use_sg = SCpnt->old_use_sg; - SCpnt->cmd_len = SCpnt->old_cmd_len; - SCpnt->sc_data_direction = SCpnt->sc_old_data_direction; - SCpnt->underflow = SCpnt->old_underflow; + scsi_setup_cmd_retry(SCpnt); /* * Hey, we are done. Let's look to see what happened. @@ -496,40 +505,42 @@ { static unsigned char tur_command[6] = {TEST_UNIT_READY, 0, 0, 0, 0, 0}; + int tries = 0; - memcpy((void *) SCpnt->cmnd, (void *) tur_command, - sizeof(tur_command)); + do { + memcpy((void *) SCpnt->cmnd, (void *) tur_command, + sizeof(tur_command)); - if (SCpnt->device->scsi_level <= SCSI_2) - SCpnt->cmnd[1] = SCpnt->lun << 5; + if (SCpnt->device->scsi_level <= SCSI_2) + SCpnt->cmnd[1] = SCpnt->lun << 5; - /* - * Zero the sense buffer. The SCSI spec mandates that any - * untransferred sense data should be interpreted as being zero. - */ - memset((void *) SCpnt->sense_buffer, 0, sizeof(SCpnt->sense_buffer)); + /* + * Zero the sense buffer. The SCSI spec mandates that any + * untransferred sense data should be interpreted as being zero. + */ + memset((void *) SCpnt->sense_buffer, 0, + sizeof(SCpnt->sense_buffer)); - SCpnt->request_buffer = NULL; - SCpnt->request_bufflen = 0; - SCpnt->use_sg = 0; - SCpnt->cmd_len = COMMAND_SIZE(SCpnt->cmnd[0]); - SCpnt->underflow = 0; - SCpnt->sc_data_direction = SCSI_DATA_NONE; + SCpnt->request_buffer = NULL; + SCpnt->request_bufflen = 0; + SCpnt->use_sg = 0; + SCpnt->cmd_len = COMMAND_SIZE(SCpnt->cmnd[0]); + SCpnt->underflow = 0; + SCpnt->sc_data_direction = SCSI_DATA_NONE; - scsi_send_eh_cmnd(SCpnt, SENSE_TIMEOUT); + scsi_send_eh_cmnd(SCpnt, SENSE_TIMEOUT); + /* + * If the SCSI device responded with "logical unit + * is in process of becoming ready", we need to + * retry this command. + */ + } while (SCpnt->eh_state == NEEDS_RETRY && tries++ < TUR_RETRIES); /* * When we eventually call scsi_finish, we really wish to complete * the original request, so let's restore the original data. (DB) */ - memcpy((void *) SCpnt->cmnd, (void *) SCpnt->data_cmnd, - sizeof(SCpnt->data_cmnd)); - SCpnt->request_buffer = SCpnt->buffer; - SCpnt->request_bufflen = SCpnt->bufflen; - SCpnt->use_sg = SCpnt->old_use_sg; - SCpnt->cmd_len = SCpnt->old_cmd_len; - SCpnt->sc_data_direction = SCpnt->sc_old_data_direction; - SCpnt->underflow = SCpnt->old_underflow; + scsi_setup_cmd_retry(SCpnt); /* * Hey, we are done. Let's look to see what happened. @@ -589,7 +600,6 @@ host = SCpnt->host; - retry: /* * We will use a queued command if possible, otherwise we will emulate the * queuing and calling of completion function ourselves. @@ -672,14 +682,13 @@ SCSI_LOG_ERROR_RECOVERY(3, printk("scsi_send_eh_cmnd: scsi_eh_completed_normally %x\n", ret)); switch (ret) { - case SUCCESS: - SCpnt->eh_state = SUCCESS; - break; - case NEEDS_RETRY: - goto retry; - case FAILED: default: - SCpnt->eh_state = FAILED; + ret = FAILED; + /*FALLTHROUGH*/ + case FAILED: + case NEEDS_RETRY: + case SUCCESS: + SCpnt->eh_state = ret; break; } } else { @@ -1220,6 +1229,82 @@ /* + * Function: scsi_eh_lock_done + * + * Purpose: free the command and request structures associated + * with the error handlers door lock request + * + * Arguments: SCpnt - the SCSI command block for the door lock request. + * + * Returns: Nothing + * + * Notes: We completed the asynchronous door lock request, and + * it has either locked the door or failed. We must free + * the command structures associated with this request. + */ +static void scsi_eh_lock_done(struct scsi_cmnd *SCpnt) +{ + struct scsi_request *SRpnt = SCpnt->sc_request; + + SCpnt->sc_request = NULL; + SRpnt->sr_command = NULL; + + scsi_release_command(SCpnt); + scsi_release_request(SRpnt); +} + + +/* + * Function: scsi_eh_lock_door + * + * Purpose: Prevent medium removal for the specified device + * + * Arguments: dev - SCSI device to prevent medium removal + * + * Locking: We must be called from process context; + * scsi_allocate_request() may sleep. + * + * Returns: Nothing + * + * Notes: We queue up an asynchronous "ALLOW MEDIUM REMOVAL" request + * on the head of the devices request queue, and continue. + * + * Bugs: scsi_allocate_request() may sleep waiting for existing + * requests to be processed. However, since we haven't + * kicked off any request processing for this host, this + * may deadlock. + * + * If scsi_allocate_request() fails for what ever reason, + * we completely forget to lock the door. + */ +STATIC void scsi_eh_lock_door(struct scsi_device *dev) +{ + struct scsi_request *SRpnt = scsi_allocate_request(dev); + + if (SRpnt == NULL) { + /* what now? */ + return; + } + + SRpnt->sr_cmnd[0] = ALLOW_MEDIUM_REMOVAL; + SRpnt->sr_cmnd[1] = (dev->scsi_level <= SCSI_2) ? (dev->lun << 5) : 0; + SRpnt->sr_cmnd[2] = 0; + SRpnt->sr_cmnd[3] = 0; + SRpnt->sr_cmnd[4] = SCSI_REMOVAL_PREVENT; + SRpnt->sr_cmnd[5] = 0; + SRpnt->sr_data_direction = SCSI_DATA_NONE; + SRpnt->sr_bufflen = 0; + SRpnt->sr_buffer = NULL; + SRpnt->sr_allowed = 5; + SRpnt->sr_done = scsi_eh_lock_done; + SRpnt->sr_timeout_per_command = 10 * HZ; + SRpnt->sr_cmd_len = COMMAND_SIZE(SRpnt->sr_cmnd[0]); + + scsi_insert_special_req(SRpnt, 1); +} + + +/* * Function: scsi_restart_operations * * Purpose: Restart IO operations to the specified host. @@ -1241,6 +1326,15 @@ ASSERT_LOCK(&io_request_lock, 0); /* + * If the door was locked, we need to insert a door lock request + * onto the head of the SCSI request queue for the device. There + * is no point trying to lock the door of an off-line device. + */ + for (SDpnt = host->host_queue; SDpnt; SDpnt = SDpnt->next) + if (SDpnt->online && SDpnt->locked) + scsi_eh_lock_door(SDpnt); + + /* * Next free up anything directly waiting upon the host. This will be * requests for character device operations, and also for ioctls to queued * block devices. @@ -1260,8 +1354,7 @@ request_queue_t *q; if ((host->can_queue > 0 && (host->host_busy >= host->can_queue)) || (host->host_blocked) - || (host->host_self_blocked) - || (SDpnt->device_blocked)) { + || (host->host_self_blocked)) { break; } q = &SDpnt->request_queue; @@ -1271,136 +1364,202 @@ } /* - * Function: scsi_unjam_host + * Function: scsi_eh_find_failed_command * - * Purpose: Attempt to fix a host which has a command that failed for - * some reason. + * Purpose: Find a failed Scsi_Cmnd structure on a device. * - * Arguments: host - host that needs unjamming. - * - * Returns: Nothing + * Arguments: SDpnt - Scsi_Device structure * - * Notes: When we come in here, we *know* that all commands on the - * bus have either completed, failed or timed out. We also - * know that no further commands are being sent to the host, - * so things are relatively quiet and we have freedom to - * fiddle with things as we wish. + * Returns: Pointer to Scsi_Cmnd structure, or NULL on failure + */ +STATIC Scsi_Cmnd *scsi_eh_find_failed_command(Scsi_Device *SDpnt) +{ + Scsi_Cmnd *SCpnt; + + for (SCpnt = SDpnt->device_queue; SCpnt; SCpnt = SCpnt->next) + if (SCpnt->state == SCSI_STATE_FAILED || + SCpnt->state == SCSI_STATE_TIMEOUT) + return SCpnt; + + return NULL; +} + + +/* + * Function: scsi_eh_test_and_retry * - * Additional note: This is only the *default* implementation. It is possible - * for individual drivers to supply their own version of this - * function, and if the maintainer wishes to do this, it is - * strongly suggested that this function be taken as a template - * and modified. This function was designed to correctly handle - * problems for about 95% of the different cases out there, and - * it should always provide at least a reasonable amount of error - * recovery. + * Purpose: Try to retry a failed command. * - * Note3: Any command marked 'FAILED' or 'TIMEOUT' must eventually - * have scsi_finish_command() called for it. We do all of - * the retry stuff here, so when we restart the host after we - * return it should have an empty queue. + * Arguments: SCpnt - scsi command structure + * done - list of commands that have been successfully + * completed. + * + * Returns: SUCCESS or FAILED + * + * Note: If the TEST UNIT READY command successfully executes, + * but returns some form of "device not ready", we wait + * a while, and retry 3 times. The device could be still + * re-initialising. */ -STATIC int scsi_unjam_host(struct Scsi_Host *host) +STATIC int scsi_eh_test_and_retry(Scsi_Cmnd *SCpnt, Scsi_Cmnd **done) { - int devices_failed; - int numfailed; - int ourrtn; - int rtn = FALSE; - int result; - Scsi_Cmnd *SCloop; - Scsi_Cmnd *SCpnt; - Scsi_Device *SDpnt; - Scsi_Device *SDloop; - Scsi_Cmnd *SCdone; - int timed_out; + int rtn, tries = 3; - ASSERT_LOCK(&io_request_lock, 0); + do { + rtn = scsi_test_unit_ready(SCpnt); + if (rtn != SUCCESS) + return rtn; - SCdone = NULL; + if (scsi_unit_is_ready(SCpnt)) + break; + + if (tries-- == 0) + return FAILED; + + scsi_sleep(5 * HZ); + } while (1); + + rtn = scsi_eh_retry_command(SCpnt); + if (rtn == SUCCESS) { + SCpnt->host->host_failed--; + scsi_eh_finish_command(done, SCpnt); + } + + return rtn; +} - /* - * First, protect against any sort of race condition. If any of the outstanding - * commands are in states that indicate that we are not yet blocked (i.e. we are - * not in a quiet state) then we got woken up in error. If we ever end up here, - * we need to re-examine some of the assumptions. - */ - for (SDpnt = host->host_queue; SDpnt; SDpnt = SDpnt->next) { - for (SCpnt = SDpnt->device_queue; SCpnt; SCpnt = SCpnt->next) { - if (SCpnt->state == SCSI_STATE_FAILED - || SCpnt->state == SCSI_STATE_TIMEOUT - || SCpnt->state == SCSI_STATE_INITIALIZING - || SCpnt->state == SCSI_STATE_UNUSED) { - continue; - } - /* - * Rats. Something is still floating around out there. This could - * be the result of the fact that the upper level drivers are still frobbing - * commands that might have succeeded. There are two outcomes. One is that - * the command block will eventually be freed, and the other one is that - * the command will be queued and will be finished along the way. - */ - SCSI_LOG_ERROR_RECOVERY(1, printk("Error handler prematurely woken - commands still active (%p %x %d)\n", SCpnt, SCpnt->state, SCpnt->target)); /* - * panic("SCSI Error handler woken too early\n"); + * Function: scsi_eh_restart_device * - * This is no longer a problem, since now the code cares only about - * SCSI_STATE_TIMEOUT and SCSI_STATE_FAILED. - * Other states are useful only to release active commands when devices are - * set offline. If (host->host_active == host->host_busy) we can safely assume - * that there are no commands in state other then TIMEOUT od FAILED. (DB) + * Purpose: Retry all failed or timed out commands for a device * - * FIXME: - * It is not easy to release correctly commands according to their state when - * devices are set offline, when the state is neither TIMEOUT nor FAILED. - * When a device is set offline, we can have some command with - * rq_status=RQ_SCSY_BUSY, owner=SCSI_STATE_HIGHLEVEL, - * state=SCSI_STATE_INITIALIZING and the driver module cannot be released. - * (DB, 17 May 1998) + * Arguments: SDpnt - SCSI device to retry + * done - list of commands that have been successfully + * completed. + * + * Returns: SUCCESS or failure code + */ +STATIC int scsi_eh_restart_device(Scsi_Device *SDpnt, Scsi_Cmnd **done) +{ + Scsi_Cmnd *SCpnt, *SCnext; + int rtn = SUCCESS; + + for (SCpnt = SDpnt->device_queue; SCpnt; SCpnt = SCnext) { + SCnext = SCpnt->next; + + if (SCpnt->state == SCSI_STATE_FAILED || + SCpnt->state == SCSI_STATE_TIMEOUT) { + rtn = scsi_eh_test_and_retry(SCpnt, done); + if (rtn != SUCCESS) + break; + } + } + + return rtn; +} + +/* + * Function: scsi_eh_set_device_offline + * + * Purpose: set a device off line + * + * Arguments: SDpnt - SCSI device to take off line + * done - list of commands that have been successfully + * completed. + * reason - text string describing why the device is off-line + * + * Returns: Nothing + * + * Notes: In addition, we complete each failed or timed out command + * attached to this device. */ +STATIC void scsi_eh_set_device_offline(Scsi_Device *SDpnt, Scsi_Cmnd **done, + const char *reason) +{ + Scsi_Cmnd *SCpnt, *SCnext; + + printk(KERN_ERR "scsi: device set offline - %s: " + "host %d channel %d id %d lun %d\n", + reason, SDpnt->host->host_no, SDpnt->channel, + SDpnt->id, SDpnt->lun); + + SDpnt->online = FALSE; + + for (SCpnt = SDpnt->device_queue; SCpnt; SCpnt = SCnext) { + SCnext = SCpnt->next; + + switch (SCpnt->state) { + case SCSI_STATE_TIMEOUT: + SCpnt->result |= DRIVER_TIMEOUT; + /*FALLTHROUGH*/ + + case SCSI_STATE_FAILED: + SCSI_LOG_ERROR_RECOVERY(3, + printk("Finishing command for device %d %x\n", + SDpnt->id, SCpnt->result)); + + SDpnt->host->host_failed--; + scsi_eh_finish_command(done, SCpnt); + break; + + default: + break; } } +} + +static void scsi_unjam_request_sense(struct Scsi_Host *host, Scsi_Cmnd **done) +{ + int rtn; + int result; + Scsi_Cmnd *SCpnt; + Scsi_Device *SDpnt; /* * Next, see if we need to request sense information. if so, * then get it now, so we have a better idea of what to do. - * FIXME(eric) this has the unfortunate side effect that if a host - * adapter does not automatically request sense information, that we end - * up shutting it down before we request it. All hosts should be doing this - * anyways, so for now all I have to say is tough noogies if you end up in here. - * On second thought, this is probably a good idea. We *really* want to give - * authors an incentive to automatically request this. + * FIXME(eric) this has the unfortunate side effect that if a + * host adapter does not automatically request sense information, + * that we end up shutting it down before we request it. All + * hosts should be doing this anyways, so for now all I have + * to say is tough noogies if you end up in here. On second + * thought, this is probably a good idea. We *really* want + * to give authors an incentive to automatically request this. */ - SCSI_LOG_ERROR_RECOVERY(3, printk("scsi_unjam_host: Checking to see if we need to request sense\n")); + SCSI_LOG_ERROR_RECOVERY(3, + printk("scsi_unjam_host: Checking to see if we need to request sense\n")); for (SDpnt = host->host_queue; SDpnt; SDpnt = SDpnt->next) { for (SCpnt = SDpnt->device_queue; SCpnt; SCpnt = SCpnt->next) { - if (SCpnt->state != SCSI_STATE_FAILED || scsi_sense_valid(SCpnt)) { + if (SCpnt->state != SCSI_STATE_FAILED || scsi_sense_valid(SCpnt)) continue; - } - SCSI_LOG_ERROR_RECOVERY(2, printk("scsi_unjam_host: Requesting sense for %d\n", - SCpnt->target)); + + SCSI_LOG_ERROR_RECOVERY(2, + printk("scsi_unjam_host: Requesting sense for %d\n", + SCpnt->target)); rtn = scsi_request_sense(SCpnt); - if (rtn != SUCCESS) { + if (rtn != SUCCESS) continue; - } - SCSI_LOG_ERROR_RECOVERY(3, printk("Sense requested for %p - result %x\n", - SCpnt, SCpnt->result)); + + SCSI_LOG_ERROR_RECOVERY(3, + printk("Sense requested for %p - result %x\n", + SCpnt, SCpnt->result)); SCSI_LOG_ERROR_RECOVERY(3, print_sense("bh", SCpnt)); result = scsi_decide_disposition(SCpnt); /* - * If the result was normal, then just pass it along to the - * upper level. + * If the result was normal, then just pass + * it along to the upper level. */ if (result == SUCCESS) { SCpnt->host->host_failed--; - scsi_eh_finish_command(&SCdone, SCpnt); + scsi_eh_finish_command(done, SCpnt); } - if (result != NEEDS_RETRY) { + if (result != NEEDS_RETRY) continue; - } + /* * We only come in here if we want to retry a * command. The test to see whether the command @@ -1410,20 +1569,29 @@ */ SCpnt->state = NEEDS_RETRY; rtn = scsi_eh_retry_command(SCpnt); - if (rtn != SUCCESS) { + if (rtn != SUCCESS) continue; - } + /* * We eventually hand this one back to the top level. */ SCpnt->host->host_failed--; - scsi_eh_finish_command(&SCdone, SCpnt); + scsi_eh_finish_command(done, SCpnt); } } +} + +static void scsi_unjam_count(struct Scsi_Host *host, Scsi_Cmnd **done) +{ + Scsi_Device *SDpnt; + Scsi_Cmnd *SCpnt; + int devices_failed; + int numfailed; + int timed_out; /* - * Go through the list of commands and figure out where we stand and how bad things - * really are. + * Go through the list of commands and figure out where we + * stand and how bad things really are. */ numfailed = 0; timed_out = 0; @@ -1433,359 +1601,478 @@ for (SCpnt = SDpnt->device_queue; SCpnt; SCpnt = SCpnt->next) { if (SCpnt->state == SCSI_STATE_FAILED) { - SCSI_LOG_ERROR_RECOVERY(5, printk("Command to ID %d failed\n", - SCpnt->target)); + SCSI_LOG_ERROR_RECOVERY(5, + printk("Command to ID %d failed\n", + SCpnt->target)); numfailed++; device_error++; } if (SCpnt->state == SCSI_STATE_TIMEOUT) { - SCSI_LOG_ERROR_RECOVERY(5, printk("Command to ID %d timedout\n", - SCpnt->target)); + SCSI_LOG_ERROR_RECOVERY(5, + printk("Command to ID %d timedout\n", + SCpnt->target)); timed_out++; device_error++; } } - if (device_error > 0) { + if (device_error > 0) devices_failed++; - } } - SCSI_LOG_ERROR_RECOVERY(2, printk("Total of %d+%d commands on %d devices require eh work\n", - numfailed, timed_out, devices_failed)); + SCSI_LOG_ERROR_RECOVERY(2, + printk("Total of %d+%d commands on %d devices require eh work\n", + numfailed, timed_out, devices_failed)); +} + +static void scsi_unjam_abort(struct Scsi_Host *host, Scsi_Cmnd **done) +{ + Scsi_Device *SDpnt; + Scsi_Cmnd *SCpnt; + int rtn; - if (host->host_failed == 0) { - ourrtn = TRUE; - goto leave; - } /* - * Next, try and see whether or not it makes sense to try and abort - * the running command. This only works out to be the case if we have - * one command that has timed out. If the command simply failed, it - * makes no sense to try and abort the command, since as far as the - * host adapter is concerned, it isn't running. + * Next, try and see whether or not it makes sense to try and + * abort the running command. This only works out to be the + * case if we have one command that has timed out. If the + * command simply failed, it makes no sense to try and abort + * the command, since as far as the host adapter is concerned, + * it isn't running. */ - SCSI_LOG_ERROR_RECOVERY(3, printk("scsi_unjam_host: Checking to see if we want to try abort\n")); + SCSI_LOG_ERROR_RECOVERY(3, + printk("scsi_unjam_host: Checking to see if we want to try abort\n")); for (SDpnt = host->host_queue; SDpnt; SDpnt = SDpnt->next) { - for (SCloop = SDpnt->device_queue; SCloop; SCloop = SCloop->next) { - if (SCloop->state != SCSI_STATE_TIMEOUT) { + for (SCpnt = SDpnt->device_queue; SCpnt; SCpnt = SCpnt->next) { + if (SCpnt->state != SCSI_STATE_TIMEOUT) continue; - } - rtn = scsi_try_to_abort_command(SCloop, ABORT_TIMEOUT); - if (rtn == SUCCESS) { - rtn = scsi_test_unit_ready(SCloop); - - if (rtn == SUCCESS && scsi_unit_is_ready(SCloop)) { - rtn = scsi_eh_retry_command(SCloop); - - if (rtn == SUCCESS) { - SCloop->host->host_failed--; - scsi_eh_finish_command(&SCdone, SCloop); - } - } - } + + rtn = scsi_try_to_abort_command(SCpnt, ABORT_TIMEOUT); + if (rtn == SUCCESS) + scsi_eh_test_and_retry(SCpnt, done); } } +} + +static void scsi_unjam_device_reset(struct Scsi_Host *host, Scsi_Cmnd **done) +{ + Scsi_Device *SDpnt; + Scsi_Cmnd *SCpnt; + int rtn; - /* - * If we have corrected all of the problems, then we are done. - */ - if (host->host_failed == 0) { - ourrtn = TRUE; - goto leave; - } /* * Either the abort wasn't appropriate, or it didn't succeed. - * Now try a bus device reset. Still, look to see whether we have - * multiple devices that are jammed or not - if we have multiple devices, - * it makes no sense to try BUS_DEVICE_RESET - we really would need - * to try a BUS_RESET instead. + * Now try a bus device reset. Still, look to see whether we + * have multiple devices that are jammed or not - if we have + * multiple devices, it makes no sense to try BUS_DEVICE_RESET + * - we really would need to try a BUS_RESET instead. * - * Does this make sense - should we try BDR on each device individually? - * Yes, definitely. + * Does this make sense - should we try BDR on each device + * individually? Yes, definitely. */ - SCSI_LOG_ERROR_RECOVERY(3, printk("scsi_unjam_host: Checking to see if we want to try BDR\n")); + SCSI_LOG_ERROR_RECOVERY(3, + printk("scsi_unjam_host: Checking to see if we want to try BDR\n")); for (SDpnt = host->host_queue; SDpnt; SDpnt = SDpnt->next) { - for (SCloop = SDpnt->device_queue; SCloop; SCloop = SCloop->next) { - if (SCloop->state == SCSI_STATE_FAILED - || SCloop->state == SCSI_STATE_TIMEOUT) { - break; - } - } - - if (SCloop == NULL) { + SCpnt = scsi_eh_find_failed_command(SDpnt); + if (SCpnt == NULL) continue; - } + /* - * OK, we have a device that is having problems. Try and send - * a bus device reset to it. + * OK, we have a device that is having problems. + * Try and send a bus device reset to it. + */ + rtn = scsi_try_bus_device_reset(SCpnt, RESET_TIMEOUT); + + /* + * A successful bus device reset causes all commands + * currently executing on the device to terminate. + * We expect the HBA driver to "forget" all commands + * associated with this device. * - * FIXME(eric) - make sure we handle the case where multiple - * commands to the same device have failed. They all must - * get properly restarted. - */ - rtn = scsi_try_bus_device_reset(SCloop, RESET_TIMEOUT); - - if (rtn == SUCCESS) { - rtn = scsi_test_unit_ready(SCloop); - - if (rtn == SUCCESS && scsi_unit_is_ready(SCloop)) { - rtn = scsi_eh_retry_command(SCloop); - - if (rtn == SUCCESS) { - SCloop->host->host_failed--; - scsi_eh_finish_command(&SCdone, SCloop); - } - } - } + * Retry each failed or timed out command currently + * outstanding for this device. + * + * If any command fails, bail out. We will try a + * bus reset instead. + */ + if (rtn == SUCCESS) + scsi_eh_restart_device(SDpnt, done); } +} + +static void scsi_unjam_bus_reset(struct Scsi_Host *host, Scsi_Cmnd **done) +{ + Scsi_Device *SDpnt; + Scsi_Cmnd *SCpnt; + int rtn, channel, max_channel = 0; - if (host->host_failed == 0) { - ourrtn = TRUE; - goto leave; - } /* - * If we ended up here, we have serious problems. The only thing left - * to try is a full bus reset. If someone has grabbed the bus and isn't - * letting go, then perhaps this will help. + * If we ended up here, we have serious problems. The only thing + * left to try is a full bus reset. If someone has grabbed the + * bus and isn't letting go, then perhaps this will help. */ - SCSI_LOG_ERROR_RECOVERY(3, printk("scsi_unjam_host: Try hard bus reset\n")); + SCSI_LOG_ERROR_RECOVERY(3, + printk("scsi_unjam_host: Try hard bus reset\n")); - /* - * We really want to loop over the various channels, and do this on - * a channel by channel basis. We should also check to see if any - * of the failed commands are on soft_reset devices, and if so, skip - * the reset. + /* + * Find the maximum channel number for this host. */ - for (SDpnt = host->host_queue; SDpnt; SDpnt = SDpnt->next) { - next_device: - for (SCpnt = SDpnt->device_queue; SCpnt; SCpnt = SCpnt->next) { - if (SCpnt->state != SCSI_STATE_FAILED - && SCpnt->state != SCSI_STATE_TIMEOUT) { + for (SDpnt = host->host_queue; SDpnt; SDpnt = SDpnt->next) + if (SDpnt->channel > max_channel) + max_channel = SDpnt->channel; + + /* + * Loop over each channel, and see if it any device on + * each channel has failed. + */ + for (channel = 0; channel <= max_channel; channel++) { + Scsi_Cmnd *failed_command; + int soft_reset; + + try_again: + failed_command = NULL; + soft_reset = 0; + + /* + * Loop over each device on this channel locating any + * failed command. We need a Scsi_Cmnd structure to + * call the bus reset function. + * + * We also need to check if any of the failed commands + * are on soft_reset devices, and if so, skip the reset. + */ + for (SDpnt = host->host_queue; SDpnt; SDpnt = SDpnt->next) { + if (SDpnt->channel != channel) continue; - } - /* - * We have a failed command. Make sure there are no other failed - * commands on the same channel that are timed out and implement a - * soft reset. - */ - for (SDloop = host->host_queue; SDloop; SDloop = SDloop->next) { - for (SCloop = SDloop->device_queue; SCloop; SCloop = SCloop->next) { - if (SCloop->channel != SCpnt->channel) { - continue; - } - if (SCloop->state != SCSI_STATE_FAILED - && SCloop->state != SCSI_STATE_TIMEOUT) { - continue; - } - if (SDloop->soft_reset && SCloop->state == SCSI_STATE_TIMEOUT) { - /* - * If this device uses the soft reset option, and this - * is one of the devices acting up, then our only - * option is to wait a bit, since the command is - * supposedly still running. - * - * FIXME(eric) - right now we will just end up falling - * through to the 'take device offline' case. - * - * FIXME(eric) - It is possible that the command completed - * *after* the error recovery procedure started, and if this - * is the case, we are worrying about nothing here. - */ - - scsi_sleep(1 * HZ); - goto next_device; - } - } - } + + SCpnt = scsi_eh_find_failed_command(SDpnt); + if (SCpnt) + failed_command = SCpnt; /* - * We now know that we are able to perform a reset for the - * bus that SCpnt points to. There are no soft-reset devices - * with outstanding timed out commands. + * If this device has timed out or failed commands, + * and uses the soft_reset option. */ - rtn = scsi_try_bus_reset(SCpnt); - if (rtn == SUCCESS) { - for (SDloop = host->host_queue; SDloop; SDloop = SDloop->next) { - for (SCloop = SDloop->device_queue; SCloop; SCloop = SCloop->next) { - if (SCloop->channel != SCpnt->channel) { - continue; - } - if (SCloop->state != SCSI_STATE_FAILED - && SCloop->state != SCSI_STATE_TIMEOUT) { - continue; - } - rtn = scsi_test_unit_ready(SCloop); - - if (rtn == SUCCESS && scsi_unit_is_ready(SCloop)) { - rtn = scsi_eh_retry_command(SCloop); - - if (rtn == SUCCESS) { - SCpnt->host->host_failed--; - scsi_eh_finish_command(&SCdone, SCloop); - } - } - /* - * If the bus reset worked, but we are still unable to - * talk to the device, take it offline. - * FIXME(eric) - is this really the correct thing to do? - */ - if (rtn != SUCCESS) { - printk(KERN_INFO "scsi: device set offline - not ready or command retry failed after bus reset: host %d channel %d id %d lun %d\n", SDloop->host->host_no, SDloop->channel, SDloop->id, SDloop->lun); - - SDloop->online = FALSE; - SDloop->host->host_failed--; - scsi_eh_finish_command(&SCdone, SCloop); - } - } - } + if (SCpnt && SDpnt->soft_reset) + soft_reset = 1; + } + + /* + * If this channel hasn't failed, we + * don't need to reset it. + */ + if (!failed_command) + continue; + + /* + * If this device uses the soft reset option, and this + * is one of the devices acting up, then our only + * option is to wait a bit, since the command is + * supposedly still running. + * + * FIXME(eric) - right now we will just end up falling + * through to the 'take device offline' case. + * + * FIXME(eric) - It is possible that the command completed + * *after* the error recovery procedure started, and if + * this is the case, we are worrying about nothing here. + * + * FIXME(rmk) - This should be bounded; we shouldn't wait + * for an infinite amount of time for any device. + */ + if (soft_reset) { + SCSI_LOG_ERROR_RECOVERY(3, + printk("scsi_unjam_host: unable to try bus " + "reset for host %d channel %d\n", + host->host_no, channel)); + scsi_sleep(1 * HZ); + goto try_again; + } + + /* + * We now know that we are able to perform a reset for the + * bus that SCpnt points to. There are no soft-reset devices + * with outstanding timed out commands. + */ + rtn = scsi_try_bus_reset(failed_command); + + /* + * If we failed to reset the bus, move on to the next bus. + */ + if (rtn != SUCCESS) + continue; + + /* + * We succeeded. Retry each failed command. + */ + for (SDpnt = host->host_queue; SDpnt; SDpnt = SDpnt->next) { + if (SDpnt->channel != channel) + continue; + + rtn = scsi_eh_restart_device(SDpnt, done); + + if (rtn != SUCCESS) { + SCpnt = scsi_eh_find_failed_command(SDpnt); + + /* + * This device failed again. Since a bus + * reset freed it up, chances are we've + * hit the same problem, so try the same + * solution. We also need to ensure that + * the SCSI bus is in the BUS FREE state + * so we can try to talk to other devices. + */ + scsi_try_bus_reset(SCpnt); + scsi_eh_set_device_offline(SDpnt, done, + "not ready or command retry " + "failed after bus reset"); } } } +} + +static void scsi_unjam_host_reset(struct Scsi_Host *host, Scsi_Cmnd **done) +{ + Scsi_Device *SDpnt; + Scsi_Cmnd *SCpnt; + Scsi_Cmnd *failed_command = NULL; + int rtn, soft_reset; - if (host->host_failed == 0) { - ourrtn = TRUE; - goto leave; - } /* - * If we ended up here, we have serious problems. The only thing left - * to try is a full host reset - perhaps the firmware on the device - * crashed, or something like that. + * If we ended up here, we have serious problems. The only thing + * left to try is a full host reset - perhaps the firmware on the + * device crashed, or something like that. * - * It is assumed that a succesful host reset will cause *all* information - * about the command to be flushed from both the host adapter *and* the - * device. + * It is assumed that a succesful host reset will cause *all* + * information about the command to be flushed from both the host + * adapter *and* the device. * - * FIXME(eric) - it isn't clear that devices that implement the soft reset - * option can ever be cleared except via cycling the power. The problem is - * that sending the host reset command will cause the host to forget - * about the pending command, but the device won't forget. For now, we - * skip the host reset option if any of the failed devices are configured - * to use the soft reset option. + * FIXME(eric) - it isn't clear that devices that implement the + * soft reset option can ever be cleared except via cycling the + * power. The problem is that sending the host reset command will + * cause the host to forget about the pending command, but the + * device won't forget. For now, we skip the host reset option + * if any of the failed devices are configured to use the soft + * reset option. */ + SCSI_LOG_ERROR_RECOVERY(3, + printk("scsi_unjam_host: Try host reset\n")); + + try_again: + failed_command = NULL; + soft_reset = 0; + for (SDpnt = host->host_queue; SDpnt; SDpnt = SDpnt->next) { - next_device2: - for (SCpnt = SDpnt->device_queue; SCpnt; SCpnt = SCpnt->next) { - if (SCpnt->state != SCSI_STATE_FAILED - && SCpnt->state != SCSI_STATE_TIMEOUT) { - continue; - } - if (SDpnt->soft_reset && SCpnt->state == SCSI_STATE_TIMEOUT) { - /* - * If this device uses the soft reset option, and this - * is one of the devices acting up, then our only - * option is to wait a bit, since the command is - * supposedly still running. - * - * FIXME(eric) - right now we will just end up falling - * through to the 'take device offline' case. - */ - SCSI_LOG_ERROR_RECOVERY(3, - printk("scsi_unjam_host: Unable to try hard host reset\n")); + /* + * Locate any failed commands for this device. + */ + SCpnt = scsi_eh_find_failed_command(SDpnt); + if (SCpnt) + failed_command = SCpnt; - /* - * Due to the spinlock, we will never get out of this - * loop without a proper wait. (DB) - */ - scsi_sleep(1 * HZ); + /* + * If this device has timed out or failed commands, + * and uses the soft_reset option. + */ + if (SCpnt && SDpnt->soft_reset) + soft_reset = 1; + } - goto next_device2; - } - SCSI_LOG_ERROR_RECOVERY(3, printk("scsi_unjam_host: Try hard host reset\n")); + /* + * If this device uses the soft reset option, and this + * is one of the devices acting up, then our only + * option is to wait a bit, since the command is + * supposedly still running. + * + * FIXME(eric) - right now we will just end up falling + * through to the 'take device offline' case. + * + * FIXME(rmk) - This should be bounded; we shouldn't wait + * for an infinite amount of time for any device. + */ + if (soft_reset) { + SCSI_LOG_ERROR_RECOVERY(3, + printk("scsi_unjam_host: unable to try " + "hard host reset\n")); /* - * FIXME(eric) - we need to obtain a valid SCpnt to perform this call. + * Due to the spinlock, we will never get out of this + * loop without a proper wait. (DB) */ - rtn = scsi_try_host_reset(SCpnt); - if (rtn == SUCCESS) { - /* - * FIXME(eric) we assume that all commands are flushed from the - * controller. We should get a DID_RESET for all of the commands - * that were pending. We should ignore these so that we can - * guarantee that we are in a consistent state. - * - * I believe this to be the case right now, but this needs to be - * tested. - */ - for (SDloop = host->host_queue; SDloop; SDloop = SDloop->next) { - for (SCloop = SDloop->device_queue; SCloop; SCloop = SCloop->next) { - if (SCloop->state != SCSI_STATE_FAILED - && SCloop->state != SCSI_STATE_TIMEOUT) { - continue; - } - rtn = scsi_test_unit_ready(SCloop); - - if (rtn == SUCCESS && scsi_unit_is_ready(SCloop)) { - rtn = scsi_eh_retry_command(SCloop); - - if (rtn == SUCCESS) { - SCpnt->host->host_failed--; - scsi_eh_finish_command(&SCdone, SCloop); - } - } - if (rtn != SUCCESS) { - printk(KERN_INFO "scsi: device set offline - not ready or command retry failed after host reset: host %d channel %d id %d lun %d\n", SDloop->host->host_no, SDloop->channel, SDloop->id, SDloop->lun); - SDloop->online = FALSE; - SDloop->host->host_failed--; - scsi_eh_finish_command(&SCdone, SCloop); - } - } - } - } - } - } + scsi_sleep(1 * HZ); - /* - * If we solved all of the problems, then let's rev up the engines again. - */ - if (host->host_failed == 0) { - ourrtn = TRUE; - goto leave; + goto try_again; } + + SCSI_LOG_ERROR_RECOVERY(3, + printk("scsi_unjam_host: Try hard host reset\n")); + /* - * If the HOST RESET failed, then for now we assume that the entire host - * adapter is too hosed to be of any use. For our purposes, however, it is - * easier to simply take the devices offline that correspond to commands - * that failed. + * FIXME(eric) - we need to obtain a valid SCpnt to perform this call. */ - SCSI_LOG_ERROR_RECOVERY(1, printk("scsi_unjam_host: Take device offline\n")); + rtn = scsi_try_host_reset(failed_command); + if (rtn == SUCCESS) { + /* + * FIXME(eric) we assume that all commands are flushed from + * the controller. We should get a DID_RESET for all of the + * commands that were pending. We should ignore these so + * that we can guarantee that we are in a consistent state. + * + * I believe this to be the case right now, but this needs + * to be tested. + */ + for (SDpnt = host->host_queue; SDpnt; SDpnt = SDpnt->next) { + rtn = scsi_eh_restart_device(SDpnt, done); - for (SDpnt = host->host_queue; SDpnt; SDpnt = SDpnt->next) { - for (SCloop = SDpnt->device_queue; SCloop; SCloop = SCloop->next) { - if (SCloop->state == SCSI_STATE_FAILED || SCloop->state == SCSI_STATE_TIMEOUT) { - SDloop = SCloop->device; - if (SDloop->online == TRUE) { - printk(KERN_INFO "scsi: device set offline - command error recover failed: host %d channel %d id %d lun %d\n", SDloop->host->host_no, SDloop->channel, SDloop->id, SDloop->lun); - SDloop->online = FALSE; - } + if (rtn != SUCCESS) { + SCpnt = scsi_eh_find_failed_command(SDpnt); /* - * This should pass the failure up to the top level driver, and - * it will have to try and do something intelligent with it. + * This device failed again. Since a host + * reset freed it up, chances are we've + * hit the same problem, so try the same + * solution. We also need to ensure that + * the SCSI bus is in the BUS FREE state + * so we can try to talk to other devices. */ - SCloop->host->host_failed--; - - if (SCloop->state == SCSI_STATE_TIMEOUT) { - SCloop->result |= (DRIVER_TIMEOUT << 24); - } - SCSI_LOG_ERROR_RECOVERY(3, printk("Finishing command for device %d %x\n", - SDloop->id, SCloop->result)); - - scsi_eh_finish_command(&SCdone, SCloop); + scsi_try_host_reset(SCpnt); + scsi_eh_set_device_offline(SDpnt, done, + "not ready or command retry " + "failed after host reset"); } } } +} + +static void scsi_unjam_failure(struct Scsi_Host *host, Scsi_Cmnd **done) +{ + Scsi_Device *SDpnt; - if (host->host_failed != 0) { + /* + * If the HOST RESET failed, then for now we assume that the + * entire host adapter is too hosed to be of any use. For our + * purposes, however, it is easier to simply take the devices + * offline that correspond to commands that failed. + */ + SCSI_LOG_ERROR_RECOVERY(1, + printk("scsi_unjam_host: Take device offline\n")); + + for (SDpnt = host->host_queue; SDpnt; SDpnt = SDpnt->next) + scsi_eh_set_device_offline(SDpnt, done, + "command error recover failed"); + + if (host->host_failed != 0) panic("scsi_unjam_host: Miscount of number of failed commands.\n"); - } + SCSI_LOG_ERROR_RECOVERY(3, printk("scsi_unjam_host: Returning\n")); +} + +static void (*unjam_method[])(struct Scsi_Host *, Scsi_Cmnd **) = { + scsi_unjam_request_sense, + scsi_unjam_count, + scsi_unjam_abort, + scsi_unjam_device_reset, + scsi_unjam_bus_reset, + scsi_unjam_host_reset, + scsi_unjam_failure, +}; + +/* + * Function: scsi_unjam_host + * + * Purpose: Attempt to fix a host which has a command that failed for + * some reason. + * + * Arguments: host - host that needs unjamming. + * + * Returns: Nothing + * + * Notes: When we come in here, we *know* that all commands on the + * bus have either completed, failed or timed out. We also + * know that no further commands are being sent to the host, + * so things are relatively quiet and we have freedom to + * fiddle with things as we wish. + * + * Additional note: This is only the *default* implementation. It is possible + * for individual drivers to supply their own version of this + * function, and if the maintainer wishes to do this, it is + * strongly suggested that this function be taken as a template + * and modified. This function was designed to correctly handle + * problems for about 95% of the different cases out there, and + * it should always provide at least a reasonable amount of error + * recovery. + * + * Note3: Any command marked 'FAILED' or 'TIMEOUT' must eventually + * have scsi_finish_command() called for it. We do all of + * the retry stuff here, so when we restart the host after we + * return it should have an empty queue. + */ +STATIC int scsi_unjam_host(struct Scsi_Host *host) +{ + Scsi_Cmnd *SCdone = NULL; + Scsi_Cmnd *SCpnt; + Scsi_Device *SDpnt; + int ourrtn = FALSE; + int i; - ourrtn = FALSE; + ASSERT_LOCK(&io_request_lock, 0); - leave: + /* + * First, protect against any sort of race condition. If any of the outstanding + * commands are in states that indicate that we are not yet blocked (i.e. we are + * not in a quiet state) then we got woken up in error. If we ever end up here, + * we need to re-examine some of the assumptions. + */ + for (SDpnt = host->host_queue; SDpnt; SDpnt = SDpnt->next) { + for (SCpnt = SDpnt->device_queue; SCpnt; SCpnt = SCpnt->next) { + if (SCpnt->state == SCSI_STATE_FAILED + || SCpnt->state == SCSI_STATE_TIMEOUT + || SCpnt->state == SCSI_STATE_INITIALIZING + || SCpnt->state == SCSI_STATE_UNUSED) { + continue; + } + /* + * Rats. Something is still floating around out there. This could + * be the result of the fact that the upper level drivers are still frobbing + * commands that might have succeeded. There are two outcomes. One is that + * the command block will eventually be freed, and the other one is that + * the command will be queued and will be finished along the way. + */ + SCSI_LOG_ERROR_RECOVERY(1, printk("Error handler prematurely woken - commands still active (%p %x %d)\n", SCpnt, SCpnt->state, SCpnt->target)); + +/* + * panic("SCSI Error handler woken too early\n"); + * + * This is no longer a problem, since now the code cares only about + * SCSI_STATE_TIMEOUT and SCSI_STATE_FAILED. + * Other states are useful only to release active commands when devices are + * set offline. If (host->host_active == host->host_busy) we can safely assume + * that there are no commands in state other then TIMEOUT od FAILED. (DB) + * + * FIXME: + * It is not easy to release correctly commands according to their state when + * devices are set offline, when the state is neither TIMEOUT nor FAILED. + * When a device is set offline, we can have some command with + * rq_status=RQ_SCSY_BUSY, owner=SCSI_STATE_HIGHLEVEL, + * state=SCSI_STATE_INITIALIZING and the driver module cannot be released. + * (DB, 17 May 1998) + */ + } + } + + for (i = 0; i < ARRAY_SIZE(unjam_method); i++) { + unjam_method[i](host, &SCdone); + + /* + * If we solved all of the problems, then + * let's rev up the engines again. + */ + if (host->host_failed == 0) { + ourrtn = TRUE; + break; + } + } /* * We should have a list of commands that we 'finished' during the course of @@ -2025,3 +2312,17 @@ * tab-width: 8 * End: */ + +EXPORT_SYMBOL(scsi_eh_times_out); +EXPORT_SYMBOL(scsi_eh_retry_command); +EXPORT_SYMBOL(scsi_request_sense); +EXPORT_SYMBOL(scsi_test_unit_ready); +EXPORT_SYMBOL(scsi_unit_is_ready); +EXPORT_SYMBOL(scsi_eh_finish_command); +EXPORT_SYMBOL(scsi_try_to_abort_command); +EXPORT_SYMBOL(scsi_try_bus_device_reset); +EXPORT_SYMBOL(scsi_try_bus_reset); +EXPORT_SYMBOL(scsi_try_host_reset); +EXPORT_SYMBOL(scsi_sense_valid); +EXPORT_SYMBOL(scsi_done); +EXPORT_SYMBOL(scsi_decide_disposition); diff -urN orig/drivers/scsi/scsi_ioctl.c linux/drivers/scsi/scsi_ioctl.c --- orig/drivers/scsi/scsi_ioctl.c Fri Oct 25 17:38:56 2002 +++ linux/drivers/scsi/scsi_ioctl.c Fri Oct 25 17:45:19 2002 @@ -153,6 +153,29 @@ return result; } +int scsi_set_medium_removal(Scsi_Device *dev, char state) +{ + char scsi_cmd[MAX_COMMAND_SIZE]; + int ret; + + if (!dev->removable || !dev->lockable) + return 0; + + scsi_cmd[0] = ALLOW_MEDIUM_REMOVAL; + scsi_cmd[1] = (dev->scsi_level <= SCSI_2) ? (dev->lun << 5) : 0; + scsi_cmd[2] = 0; + scsi_cmd[3] = 0; + scsi_cmd[4] = state; + scsi_cmd[5] = 0; + + ret = ioctl_internal_command(dev, scsi_cmd, IOCTL_NORMAL_TIMEOUT, NORMAL_RETRIES); + + if (ret == 0) + dev->locked = state == SCSI_REMOVAL_PREVENT; + + return ret; +} + /* * This interface is depreciated - users should use the scsi generic (sg) * interface instead, as this is a more flexible approach to performing @@ -449,24 +472,9 @@ return scsi_ioctl_send_command((Scsi_Device *) dev, (Scsi_Ioctl_Command *) arg); case SCSI_IOCTL_DOORLOCK: - if (!dev->removable || !dev->lockable) - return 0; - scsi_cmd[0] = ALLOW_MEDIUM_REMOVAL; - scsi_cmd[1] = cmd_byte1; - scsi_cmd[2] = scsi_cmd[3] = scsi_cmd[5] = 0; - scsi_cmd[4] = SCSI_REMOVAL_PREVENT; - return ioctl_internal_command((Scsi_Device *) dev, scsi_cmd, - IOCTL_NORMAL_TIMEOUT, NORMAL_RETRIES); - break; + return scsi_set_medium_removal(dev, SCSI_REMOVAL_PREVENT); case SCSI_IOCTL_DOORUNLOCK: - if (!dev->removable || !dev->lockable) - return 0; - scsi_cmd[0] = ALLOW_MEDIUM_REMOVAL; - scsi_cmd[1] = cmd_byte1; - scsi_cmd[2] = scsi_cmd[3] = scsi_cmd[5] = 0; - scsi_cmd[4] = SCSI_REMOVAL_ALLOW; - return ioctl_internal_command((Scsi_Device *) dev, scsi_cmd, - IOCTL_NORMAL_TIMEOUT, NORMAL_RETRIES); + return scsi_set_medium_removal(dev, SCSI_REMOVAL_ALLOW); case SCSI_IOCTL_TEST_UNIT_READY: scsi_cmd[0] = TEST_UNIT_READY; scsi_cmd[1] = cmd_byte1; diff -urN orig/drivers/scsi/scsi_lib.c linux/drivers/scsi/scsi_lib.c --- orig/drivers/scsi/scsi_lib.c Fri Oct 25 17:39:14 2002 +++ linux/drivers/scsi/scsi_lib.c Fri Oct 25 17:45:22 2002 @@ -208,6 +208,30 @@ } /* + * Function: scsi_setup_cmd_retry() + * + * Purpose: Restore the command state for a retry + * + * Arguments: SCpnt - command to be restored + * + * Returns: Nothing + * + * Notes: Immediately prior to retrying a command, we need + * to restore certain fields that we saved above. + */ +void scsi_setup_cmd_retry(Scsi_Cmnd *SCpnt) +{ + memcpy((void *) SCpnt->cmnd, (void *) SCpnt->data_cmnd, + sizeof(SCpnt->data_cmnd)); + SCpnt->request_buffer = SCpnt->buffer; + SCpnt->request_bufflen = SCpnt->bufflen; + SCpnt->use_sg = SCpnt->old_use_sg; + SCpnt->cmd_len = SCpnt->old_cmd_len; + SCpnt->sc_data_direction = SCpnt->sc_old_data_direction; + SCpnt->underflow = SCpnt->old_underflow; +} + +/* * Function: scsi_queue_next_request() * * Purpose: Handle post-processing of completed commands. @@ -726,7 +750,7 @@ printk("scsi%d: ERROR on channel %d, id %d, lun %d, CDB: ", SCpnt->host->host_no, (int) SCpnt->channel, (int) SCpnt->target, (int) SCpnt->lun); - print_command(SCpnt->cmnd); + print_command(SCpnt->data_cmnd); print_sense("sd", SCpnt); SCpnt = scsi_end_request(SCpnt, 0, block_sectors); return; @@ -901,8 +925,17 @@ * space. Technically the error handling thread should be * doing this crap, but the error handler isn't used by * most hosts. + * + * (rmk) + * Trying to lock the door can cause deadlocks. We therefore + * only use this for old hosts; our door locking is now done + * by the error handler in scsi_restart_operations for new + * eh hosts. + * + * Note that we don't clear was_reset here; this is used by + * st.c, and either one or other has to die. */ - if (SDpnt->was_reset) { + if (SHpnt->hostt->use_new_eh_code == 0 && SDpnt->was_reset) { /* * We need to relock the door, but we might * be in an interrupt handler. Only do this @@ -913,7 +946,7 @@ * this work. */ SDpnt->was_reset = 0; - if (SDpnt->removable && !in_interrupt()) { + if (SDpnt->removable && SDpnt->locked && !in_interrupt()) { spin_unlock_irq(&io_request_lock); scsi_ioctl(SDpnt, SCSI_IOCTL_DOORLOCK, 0); spin_lock_irq(&io_request_lock); diff -urN orig/drivers/scsi/scsi_syms.c linux/drivers/scsi/scsi_syms.c --- orig/drivers/scsi/scsi_syms.c Mon Aug 5 13:31:24 2002 +++ linux/drivers/scsi/scsi_syms.c Mon Jan 20 23:12:35 2003 @@ -103,3 +103,6 @@ extern int scsi_delete_timer(Scsi_Cmnd *); EXPORT_SYMBOL(scsi_add_timer); EXPORT_SYMBOL(scsi_delete_timer); + +extern int scsi_set_medium_removal(Scsi_Device *dev, char state); +EXPORT_SYMBOL(scsi_set_medium_removal); diff -urN orig/drivers/scsi/sd.c linux/drivers/scsi/sd.c --- orig/drivers/scsi/sd.c Fri Oct 25 17:38:56 2002 +++ linux/drivers/scsi/sd.c Fri Oct 25 17:45:19 2002 @@ -399,6 +399,7 @@ this_count = 0xffff; SCpnt->cmnd[0] += READ_10 - READ_6; + SCpnt->cmnd[1] |= 1 << 3; /* Set FUA --rmk */ SCpnt->cmnd[2] = (unsigned char) (block >> 24) & 0xff; SCpnt->cmnd[3] = (unsigned char) (block >> 16) & 0xff; SCpnt->cmnd[4] = (unsigned char) (block >> 8) & 0xff; @@ -524,7 +525,7 @@ if (SDev->removable) if (SDev->access_count==1) if (scsi_block_when_processing_errors(SDev)) - scsi_ioctl(SDev, SCSI_IOCTL_DOORLOCK, NULL); + scsi_set_medium_removal(SDev, SCSI_REMOVAL_PREVENT); return 0; @@ -553,7 +554,7 @@ if (SDev->removable) { if (!SDev->access_count) if (scsi_block_when_processing_errors(SDev)) - scsi_ioctl(SDev, SCSI_IOCTL_DOORUNLOCK, NULL); + scsi_set_medium_removal(SDev, SCSI_REMOVAL_ALLOW); } if (SDev->host->hostt->module) __MOD_DEC_USE_COUNT(SDev->host->hostt->module); diff -urN orig/drivers/scsi/sr_ioctl.c linux/drivers/scsi/sr_ioctl.c --- orig/drivers/scsi/sr_ioctl.c Fri Oct 25 17:38:57 2002 +++ linux/drivers/scsi/sr_ioctl.c Fri Oct 25 17:45:19 2002 @@ -216,9 +216,8 @@ int sr_lock_door(struct cdrom_device_info *cdi, int lock) { - return scsi_ioctl(scsi_CDs[MINOR(cdi->dev)].device, - lock ? SCSI_IOCTL_DOORLOCK : SCSI_IOCTL_DOORUNLOCK, - 0); + return scsi_set_medium_removal(scsi_CDs[MINOR(cdi->dev)].device, + lock ? SCSI_REMOVAL_PREVENT : SCSI_REMOVAL_ALLOW); } int sr_drive_status(struct cdrom_device_info *cdi, int slot) diff -urN orig/drivers/serial/21285.c linux/drivers/serial/21285.c --- orig/drivers/serial/21285.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/serial/21285.c Fri Feb 21 16:03:44 2003 @@ -0,0 +1,600 @@ +/* + * linux/drivers/char/serial_21285.c + * + * Driver for the serial port on the 21285 StrongArm-110 core logic chip. + * + * Based on drivers/char/serial.c + * + * $Id: 21285.c,v 1.4.2.1 2002/10/24 09:53:23 rmk Exp $ + */ +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include + +#define BAUD_BASE (mem_fclk_21285/64) + +#define SERIAL_21285_NAME "ttyFB" +#define SERIAL_21285_MAJOR 204 +#define SERIAL_21285_MINOR 4 + +#define SERIAL_21285_AUXNAME "cuafb" +#define SERIAL_21285_AUXMAJOR 205 +#define SERIAL_21285_AUXMINOR 4 + +#ifdef CONFIG_SERIAL_21285_OLD +#include +/* + * Compatability with a mistake made a long time ago. + * Note - the use of "ttyI", "/dev/ttyS0" and major/minor 5,64 + * is HIGHLY DEPRECIATED, and will be removed in the 2.5 + * kernel series. + * -- rmk 15/04/2000 + */ +#define SERIAL_21285_OLD_NAME "ttyI" +#define SERIAL_21285_OLD_MAJOR TTY_MAJOR +#define SERIAL_21285_OLD_MINOR 64 + +static struct tty_driver rs285_old_driver; +#endif + +static struct tty_driver rs285_driver, callout_driver; +static int rs285_refcount; +static struct tty_struct *rs285_table[1]; + +static struct termios *rs285_termios[1]; +static struct termios *rs285_termios_locked[1]; + +static char wbuf[1000], *putp = wbuf, *getp = wbuf, x_char; +static struct tty_struct *rs285_tty; +static DECLARE_MUTEX(rs285_sem); +static int rs285_use_count; +static unsigned long rs285_irq_enabled; + +#define TX_IRQ_BIT (0) +#define RX_IRQ_BIT (1) + +static void rs285_stop_tx(void) +{ + if (test_and_clear_bit(TX_IRQ_BIT, &rs285_irq_enabled)) + disable_irq(IRQ_CONTX); +} + +static void rs285_start_tx(void) +{ + if (!test_and_set_bit(TX_IRQ_BIT, &rs285_irq_enabled)) + enable_irq(IRQ_CONTX); +} + +static void rs285_stop_rx(void) +{ + if (test_and_clear_bit(RX_IRQ_BIT, &rs285_irq_enabled)) + disable_irq(IRQ_CONRX); +} + +static void rs285_start_rx(void) +{ + if (!test_and_set_bit(RX_IRQ_BIT, &rs285_irq_enabled)) + enable_irq(IRQ_CONRX); +} + +static int rs285_write_room(struct tty_struct *tty) +{ + return putp >= getp ? (sizeof(wbuf) - (long) putp + (long) getp) : ((long) getp - (long) putp - 1); +} + +static void rs285_rx_int(int irq, void *dev_id, struct pt_regs *regs) +{ + if (!rs285_tty) { + rs285_stop_rx(); + return; + } + while (!(*CSR_UARTFLG & 0x10)) { + int ch, flag; + ch = *CSR_UARTDR; + flag = *CSR_RXSTAT; + if (flag & 4) + tty_insert_flip_char(rs285_tty, 0, TTY_OVERRUN); + if (flag & 2) + flag = TTY_PARITY; + else if (flag & 1) + flag = TTY_FRAME; + tty_insert_flip_char(rs285_tty, ch, flag); + } + tty_flip_buffer_push(rs285_tty); +} + +static void rs285_send_xchar(struct tty_struct *tty, char ch) +{ + x_char = ch; + rs285_start_tx(); +} + +static void rs285_throttle(struct tty_struct *tty) +{ + if (I_IXOFF(tty)) + rs285_send_xchar(tty, STOP_CHAR(tty)); +} + +static void rs285_unthrottle(struct tty_struct *tty) +{ + if (I_IXOFF(tty)) { + if (x_char) + x_char = 0; + else + rs285_send_xchar(tty, START_CHAR(tty)); + } +} + +static void rs285_tx_int(int irq, void *dev_id, struct pt_regs *regs) +{ + while (!(*CSR_UARTFLG & 0x20)) { + if (x_char) { + *CSR_UARTDR = x_char; + x_char = 0; + continue; + } + if (putp == getp) { + rs285_stop_tx(); + break; + } + *CSR_UARTDR = *getp; + if (++getp >= wbuf + sizeof(wbuf)) + getp = wbuf; + } + if (rs285_tty) + wake_up_interruptible(&rs285_tty->write_wait); +} + +static inline int rs285_xmit(int ch) +{ + if (putp + 1 == getp || (putp + 1 == wbuf + sizeof(wbuf) && getp == wbuf)) + return 0; + *putp = ch; + if (++putp >= wbuf + sizeof(wbuf)) + putp = wbuf; + rs285_start_tx(); + return 1; +} + +static int rs285_write(struct tty_struct *tty, int from_user, + const u_char * buf, int count) +{ + int i; + + if (from_user && verify_area(VERIFY_READ, buf, count)) + return -EINVAL; + + for (i = 0; i < count; i++) { + char ch; + if (from_user) + __get_user(ch, buf + i); + else + ch = buf[i]; + if (!rs285_xmit(ch)) + break; + } + return i; +} + +static void rs285_put_char(struct tty_struct *tty, u_char ch) +{ + rs285_xmit(ch); +} + +static int rs285_chars_in_buffer(struct tty_struct *tty) +{ + return sizeof(wbuf) - rs285_write_room(tty); +} + +static void rs285_flush_buffer(struct tty_struct *tty) +{ + rs285_stop_tx(); + putp = getp = wbuf; + if (x_char) + rs285_start_tx(); +} + +static inline void rs285_set_cflag(int cflag) +{ + int h_lcr, baud, quot; + + switch (cflag & CSIZE) { + case CS5: + h_lcr = 0x10; + break; + case CS6: + h_lcr = 0x30; + break; + case CS7: + h_lcr = 0x50; + break; + default: /* CS8 */ + h_lcr = 0x70; + break; + + } + if (cflag & CSTOPB) + h_lcr |= 0x08; + if (cflag & PARENB) + h_lcr |= 0x02; + if (!(cflag & PARODD)) + h_lcr |= 0x04; + + switch (cflag & CBAUD) { + case B200: baud = 200; break; + case B300: baud = 300; break; + case B1200: baud = 1200; break; + case B1800: baud = 1800; break; + case B2400: baud = 2400; break; + case B4800: baud = 4800; break; + default: + case B9600: baud = 9600; break; + case B19200: baud = 19200; break; + case B38400: baud = 38400; break; + case B57600: baud = 57600; break; + case B115200: baud = 115200; break; + } + + /* + * The documented expression for selecting the divisor is: + * BAUD_BASE / baud - 1 + * However, typically BAUD_BASE is not divisible by baud, so + * we want to select the divisor that gives us the minimum + * error. Therefore, we want: + * int(BAUD_BASE / baud - 0.5) -> + * int(BAUD_BASE / baud - (baud >> 1) / baud) -> + * int((BAUD_BASE - (baud >> 1)) / baud) + */ + quot = (BAUD_BASE - (baud >> 1)) / baud; + + *CSR_UARTCON = 0; + *CSR_L_UBRLCR = quot & 0xff; + *CSR_M_UBRLCR = (quot >> 8) & 0x0f; + *CSR_H_UBRLCR = h_lcr; + *CSR_UARTCON = 1; +} + +static void rs285_set_termios(struct tty_struct *tty, struct termios *old) +{ + if (old && tty->termios->c_cflag == old->c_cflag) + return; + rs285_set_cflag(tty->termios->c_cflag); +} + + +static void rs285_stop(struct tty_struct *tty) +{ + rs285_stop_tx(); +} + +static void rs285_start(struct tty_struct *tty) +{ + rs285_start_tx(); +} + +static void rs285_wait_until_sent(struct tty_struct *tty, int timeout) +{ + int orig_jiffies = jiffies; + while (*CSR_UARTFLG & 8) { + set_current_state(TASK_INTERRUPTIBLE); + schedule_timeout(1); + if (signal_pending(current)) + break; + if (timeout && time_after(jiffies, orig_jiffies + timeout)) + break; + } + set_current_state(TASK_RUNNING); +} + +static int rs285_open(struct tty_struct *tty, struct file *filp) +{ + int line, ret; + + MOD_INC_USE_COUNT; + + line = MINOR(tty->device) - tty->driver.minor_start; + if (line) + return -ENODEV; + + ret = down_interruptible(&rs285_sem); + if (ret) + return ret; + + tty->driver_data = NULL; + rs285_tty = tty; + + if (rs285_use_count == 0) { + rs285_irq_enabled = 3; + ret = request_irq(IRQ_CONRX, rs285_rx_int, 0, "rs285", NULL); + if (ret == 0) { + ret = request_irq(IRQ_CONTX, rs285_tx_int, 0, "rs285", + NULL); + if (ret) + free_irq(IRQ_CONRX, NULL); + } + } + + if (ret == 0) + rs285_use_count++; + + up(&rs285_sem); + + return ret; +} + +static void rs285_close(struct tty_struct *tty, struct file *filp) +{ + down(&rs285_sem); + if (!--rs285_use_count) { + rs285_wait_until_sent(tty, 0); + rs285_stop_rx(); + rs285_stop_tx(); + rs285_tty = NULL; + free_irq(IRQ_CONTX, NULL); + free_irq(IRQ_CONRX, NULL); + } + up(&rs285_sem); + MOD_DEC_USE_COUNT; +} + +static int __init rs285_init(void) +{ + int baud = B9600; + + if (machine_is_personal_server()) + baud = B57600; + + rs285_driver.magic = TTY_DRIVER_MAGIC; + rs285_driver.driver_name = "serial_21285"; + rs285_driver.name = SERIAL_21285_NAME; + rs285_driver.major = SERIAL_21285_MAJOR; + rs285_driver.minor_start = SERIAL_21285_MINOR; + rs285_driver.num = 1; + rs285_driver.type = TTY_DRIVER_TYPE_SERIAL; + rs285_driver.subtype = SERIAL_TYPE_NORMAL; + rs285_driver.init_termios = tty_std_termios; + rs285_driver.init_termios.c_cflag = baud | CS8 | CREAD | HUPCL | CLOCAL; + rs285_driver.flags = TTY_DRIVER_REAL_RAW; + rs285_driver.refcount = &rs285_refcount; + rs285_driver.table = rs285_table; + rs285_driver.termios = rs285_termios; + rs285_driver.termios_locked = rs285_termios_locked; + + rs285_driver.open = rs285_open; + rs285_driver.close = rs285_close; + rs285_driver.write = rs285_write; + rs285_driver.put_char = rs285_put_char; + rs285_driver.write_room = rs285_write_room; + rs285_driver.chars_in_buffer = rs285_chars_in_buffer; + rs285_driver.flush_buffer = rs285_flush_buffer; + rs285_driver.throttle = rs285_throttle; + rs285_driver.unthrottle = rs285_unthrottle; + rs285_driver.send_xchar = rs285_send_xchar; + rs285_driver.set_termios = rs285_set_termios; + rs285_driver.stop = rs285_stop; + rs285_driver.start = rs285_start; + rs285_driver.wait_until_sent = rs285_wait_until_sent; + + callout_driver = rs285_driver; + callout_driver.name = SERIAL_21285_AUXNAME; + callout_driver.major = SERIAL_21285_AUXMAJOR; + callout_driver.subtype = SERIAL_TYPE_CALLOUT; + +#ifdef CONFIG_SERIAL_21285_OLD + if (!machine_is_ebsa285() && !machine_is_netwinder()) { + rs285_old_driver = rs285_driver; + rs285_old_driver.name = SERIAL_21285_OLD_NAME; + rs285_old_driver.major = SERIAL_21285_OLD_MAJOR; + rs285_old_driver.minor_start = SERIAL_21285_OLD_MINOR; + + if (tty_register_driver(&rs285_old_driver)) + printk(KERN_ERR "Couldn't register old 21285 serial driver\n"); + } +#endif + + if (tty_register_driver(&rs285_driver)) + printk(KERN_ERR "Couldn't register 21285 serial driver\n"); + if (tty_register_driver(&callout_driver)) + printk(KERN_ERR "Couldn't register 21285 callout driver\n"); + + return 0; +} + +static void __exit rs285_fini(void) +{ + unsigned long flags; + int ret; + + save_flags(flags); + cli(); + ret = tty_unregister_driver(&callout_driver); + if (ret) + printk(KERN_ERR "Unable to unregister 21285 callout driver " + "(%d)\n", ret); + ret = tty_unregister_driver(&rs285_driver); + if (ret) + printk(KERN_ERR "Unable to unregister 21285 driver (%d)\n", + ret); +#ifdef CONFIG_SERIAL_21285_OLD + if (!machine_is_ebsa285() && !machine_is_netwinder()) { + ret = tty_unregister_driver(&rs285_old_driver); + if (ret) + printk(KERN_ERR "Unable to unregister old 21285 " + "driver (%d)\n", ret); + } +#endif + free_irq(IRQ_CONTX, NULL); + free_irq(IRQ_CONRX, NULL); + restore_flags(flags); +} + +module_init(rs285_init); +module_exit(rs285_fini); + +#ifdef CONFIG_SERIAL_21285_CONSOLE +/************** console driver *****************/ + +static void rs285_console_write(struct console *co, const char *s, u_int count) +{ + int i; + + rs285_stop_tx(); + for (i = 0; i < count; i++) { + while (*CSR_UARTFLG & 0x20); + *CSR_UARTDR = s[i]; + if (s[i] == '\n') { + while (*CSR_UARTFLG & 0x20); + *CSR_UARTDR = '\r'; + } + } + rs285_start_tx(); +} + +static kdev_t rs285_console_device(struct console *c) +{ + return MKDEV(SERIAL_21285_MAJOR, SERIAL_21285_MINOR); +} + +static int __init rs285_console_setup(struct console *co, char *options) +{ + int baud = 9600; + int bits = 8; + int parity = 'n'; + int cflag = CREAD | HUPCL | CLOCAL; + + if (machine_is_personal_server()) + baud = 57600; + + if (options) { + char *s = options; + baud = simple_strtoul(options, NULL, 10); + while (*s >= '0' && *s <= '9') + s++; + if (*s) + parity = *s++; + if (*s) + bits = *s - '0'; + } + + /* + * Now construct a cflag setting. + */ + switch (baud) { + case 1200: + cflag |= B1200; + break; + case 2400: + cflag |= B2400; + break; + case 4800: + cflag |= B4800; + break; + case 9600: + cflag |= B9600; + break; + case 19200: + cflag |= B19200; + break; + case 38400: + cflag |= B38400; + break; + case 57600: + cflag |= B57600; + break; + case 115200: + cflag |= B115200; + break; + default: + cflag |= B9600; + break; + } + switch (bits) { + case 7: + cflag |= CS7; + break; + default: + cflag |= CS8; + break; + } + switch (parity) { + case 'o': + case 'O': + cflag |= PARODD; + break; + case 'e': + case 'E': + cflag |= PARENB; + break; + } + co->cflag = cflag; + rs285_set_cflag(cflag); + rs285_console_write(NULL, "\e[2J\e[Hboot ", 12); + if (options) + rs285_console_write(NULL, options, strlen(options)); + else + rs285_console_write(NULL, "no options", 10); + rs285_console_write(NULL, "\n", 1); + + return 0; +} + +#ifdef CONFIG_SERIAL_21285_OLD +static struct console rs285_old_cons = +{ + SERIAL_21285_OLD_NAME, + rs285_console_write, + NULL, + rs285_console_device, + NULL, + rs285_console_setup, + CON_PRINTBUFFER, + -1, + 0, + NULL +}; +#endif + +static struct console rs285_cons = +{ + name: SERIAL_21285_NAME, + write: rs285_console_write, + device: rs285_console_device, + setup: rs285_console_setup, + flags: CON_PRINTBUFFER, + index: -1, +}; + +void __init rs285_console_init(void) +{ +#ifdef CONFIG_SERIAL_21285_OLD + if (!machine_is_ebsa285() && !machine_is_netwinder()) + register_console(&rs285_old_cons); +#endif + register_console(&rs285_cons); +} + +#endif /* CONFIG_SERIAL_21285_CONSOLE */ + +EXPORT_NO_SYMBOLS; + +MODULE_LICENSE("GPL"); +MODULE_DESCRIPTION("Intel Footbridge (21285) serial driver"); diff -urN orig/drivers/serial/8250.c linux/drivers/serial/8250.c --- orig/drivers/serial/8250.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/serial/8250.c Fri Oct 25 15:08:53 2002 @@ -0,0 +1,1921 @@ +/* + * linux/drivers/char/serial_8250.c + * + * Driver for 8250/16550-type serial ports + * + * Based on drivers/char/serial.c, by Linus Torvalds, Theodore Ts'o. + * + * Copyright (C) 2001 Russell King. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * $Id: 8250.c,v 1.14.2.8 2002/10/24 14:31:31 rmk Exp $ + * + * A note about mapbase / membase + * + * mapbase is the physical address of the IO port. Currently, we don't + * support this very well, and it may well be dropped from this driver + * in future. As such, mapbase should be NULL. + * + * membase is an 'ioremapped' cookie. This is compatible with the old + * serial.c driver, and is currently the preferred form. + */ +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include + +#include "8250.h" + +/* + * This converts from our new CONFIG_ symbols to the symbols + * that asm/serial.h expects. You _NEED_ to comment out the + * linux/config.h include contained inside asm/serial.h for + * this to work. + */ +#undef CONFIG_SERIAL_MANY_PORTS +#undef CONFIG_SERIAL_DETECT_IRQ +#undef CONFIG_SERIAL_MULTIPORT +#undef CONFIG_HUB6 + +#ifdef CONFIG_SERIAL_8250_MANY_PORTS +#define CONFIG_SERIAL_MANY_PORTS 1 +#endif +#ifdef CONFIG_SERIAL_8250_DETECT_IRQ +#define CONFIG_SERIAL_DETECT_IRQ 1 +#endif +#ifdef CONFIG_SERIAL_8250_MULTIPORT +#define CONFIG_SERIAL_MULTIPORT 1 +#endif +#ifdef CONFIG_SERIAL_8250_HUB6 +#define CONFIG_HUB6 1 +#endif + +#include + +static struct old_serial_port old_serial_port[] = { + SERIAL_PORT_DFNS /* defined in asm/serial.h */ +}; + +#define UART_NR ARRAY_SIZE(old_serial_port) + +static struct tty_driver normal, callout; +static struct tty_struct *serial8250_table[UART_NR]; +static struct termios *serial8250_termios[UART_NR], *serial8250_termios_locked[UART_NR]; +#ifdef CONFIG_SERIAL_8250_CONSOLE +static struct console serial8250_console; +static unsigned int lsr_break_flag; +#endif +static struct uart_info *IRQ_ports[NR_IRQS]; + +#if defined(CONFIG_SERIAL_RSA) && defined(MODULE) + +#define PORT_RSA_MAX 4 +static int probe_rsa[PORT_RSA_MAX]; +static int force_rsa[PORT_RSA_MAX]; + +MODULE_PARM(probe_rsa, "1-" __MODULE_STRING(PORT_RSA_MAX) "i"); +MODULE_PARM_DESC(probe_rsa, "Probe I/O ports for RSA"); +MODULE_PARM(force_rsa, "1-" __MODULE_STRING(PORT_RSA_MAX) "i"); +MODULE_PARM_DESC(force_rsa, "Force I/O ports for RSA"); +#endif /* CONFIG_SERIAL_RSA */ + +#define port_acr unused[0] /* 8bit */ +#define port_ier unused[1] /* 8bit */ +#define port_rev unused[2] /* 8bit */ +#define port_lcr unused[3] /* 8bit */ + +/* + * Here we define the default xmit fifo size used for each type of UART. + */ +static const struct serial_uart_config uart_config[PORT_MAX_8250+1] = { + { "unknown", 1, 0 }, + { "8250", 1, 0 }, + { "16450", 1, 0 }, + { "16550", 1, 0 }, + { "16550A", 16, UART_CLEAR_FIFO | UART_USE_FIFO }, + { "Cirrus", 1, 0 }, + { "ST16650", 1, UART_CLEAR_FIFO | UART_STARTECH }, + { "ST16650V2", 32, UART_CLEAR_FIFO | UART_USE_FIFO | UART_STARTECH }, + { "TI16750", 64, UART_CLEAR_FIFO | UART_USE_FIFO }, + { "Startech", 1, 0 }, + { "16C950/954", 128, UART_CLEAR_FIFO | UART_USE_FIFO }, + { "ST16654", 64, UART_CLEAR_FIFO | UART_USE_FIFO | UART_STARTECH }, + { "XR16850", 128, UART_CLEAR_FIFO | UART_USE_FIFO | UART_STARTECH }, + { "RSA", 2048, UART_CLEAR_FIFO | UART_USE_FIFO } +}; + +static _INLINE_ unsigned int serial_in(struct uart_port *port, int offset) +{ + offset <<= port->regshift; + + switch (port->iotype) { +#ifdef CONFIG_SERIAL_8250_HUB6 + case SERIAL_IO_HUB6: + outb(port->hub6 - 1 + offset, port->iobase); + return inb(port->iobase + 1); +#endif + + case SERIAL_IO_MEM: + return readb((unsigned long)port->membase + offset); + + default: + return inb(port->iobase + offset); + } +} + +static _INLINE_ void +serial_out(struct uart_port *port, int offset, int value) +{ + offset <<= port->regshift; + + switch (port->iotype) { +#ifdef CONFIG_SERIAL_8250_HUB6 + case SERIAL_IO_HUB6: + outb(port->hub6 - 1 + offset, port->iobase); + outb(value, port->iobase + 1); + break; +#endif + + case SERIAL_IO_MEM: + writeb(value, (unsigned long)port->membase + offset); + break; + + default: + outb(value, port->iobase + offset); + } +} + +/* + * We used to support using pause I/O for certain machines. We + * haven't supported this for a while, but just in case it's badly + * needed for certain old 386 machines, I've left these #define's + * in.... + */ +#define serial_inp(port, offset) serial_in(port, offset) +#define serial_outp(port, offset, value) serial_out(port, offset, value) + + +/* + * For the 16C950 + */ +static void serial_icr_write(struct uart_port *port, int offset, int value) +{ + serial_out(port, UART_SCR, offset); + serial_out(port, UART_ICR, value); +} + +static unsigned int serial_icr_read(struct uart_port *port, int offset) +{ + unsigned int value; + + serial_icr_write(port, UART_ACR, port->port_acr | UART_ACR_ICRRD); + serial_out(port, UART_SCR, offset); + value = serial_in(port, UART_ICR); + serial_icr_write(port, UART_ACR, port->port_acr); + + return value; +} + +#ifdef CONFIG_SERIAL_RSA +/* Attempts to turn on the RSA FIFO. Returns zero on failure */ +static int enable_rsa(struct uart_port *port) +{ + unsigned char mode; + int result; + unsigned long flags; + + save_flags(flags); cli(); + mode = serial_inp(port, UART_RSA_MSR); + result = mode & UART_RSA_MSR_FIFO; + + if (!result) { + serial_outp(port, UART_RSA_MSR, mode | UART_RSA_MSR_FIFO); + mode = serial_inp(port, UART_RSA_MSR); + result = mode & UART_RSA_MSR_FIFO; + } + + restore_flags(flags); + return result; +} + +/* Attempts to turn off the RSA FIFO. Returns zero on failure */ +static int disable_rsa(struct uart_port *port) +{ + unsigned char mode; + int result; + unsigned long flags; + + save_flags(flags); cli(); + mode = serial_inp(port, UART_RSA_MSR); + result = !(mode & UART_RSA_MSR_FIFO); + + if (!result) { + serial_outp(port, UART_RSA_MSR, mode & ~UART_RSA_MSR_FIFO); + mode = serial_inp(port, UART_RSA_MSR); + result = !(mode & UART_RSA_MSR_FIFO); + } + + restore_flags(flags); + return result; +} +#endif /* CONFIG_SERIAL_RSA */ + +/* + * This is a quickie test to see how big the FIFO is. + * It doesn't work at all the time, more's the pity. + */ +static int size_fifo(struct uart_port *port) +{ + unsigned char old_fcr, old_mcr, old_dll, old_dlm; + int count; + + old_fcr = serial_inp(port, UART_FCR); + old_mcr = serial_inp(port, UART_MCR); + serial_outp(port, UART_FCR, UART_FCR_ENABLE_FIFO | + UART_FCR_CLEAR_RCVR | UART_FCR_CLEAR_XMIT); + serial_outp(port, UART_MCR, UART_MCR_LOOP); + serial_outp(port, UART_LCR, UART_LCR_DLAB); + old_dll = serial_inp(port, UART_DLL); + old_dlm = serial_inp(port, UART_DLM); + serial_outp(port, UART_DLL, 0x01); + serial_outp(port, UART_DLM, 0x00); + serial_outp(port, UART_LCR, 0x03); + for (count = 0; count < 256; count++) + serial_outp(port, UART_TX, count); + mdelay(20); + for (count = 0; (serial_inp(port, UART_LSR) & UART_LSR_DR) && + (count < 256); count++) + serial_inp(port, UART_RX); + serial_outp(port, UART_FCR, old_fcr); + serial_outp(port, UART_MCR, old_mcr); + serial_outp(port, UART_LCR, UART_LCR_DLAB); + serial_outp(port, UART_DLL, old_dll); + serial_outp(port, UART_DLM, old_dlm); + + return count; +} + +/* + * This is a helper routine to autodetect StarTech/Exar/Oxsemi UART's. + * When this function is called we know it is at least a StarTech + * 16650 V2, but it might be one of several StarTech UARTs, or one of + * its clones. (We treat the broken original StarTech 16650 V1 as a + * 16550, and why not? Startech doesn't seem to even acknowledge its + * existence.) + * + * What evil have men's minds wrought... + */ +static void +autoconfig_startech_uarts(struct uart_port *port) +{ + unsigned char scratch, scratch2, scratch3, scratch4; + + /* + * First we check to see if it's an Oxford Semiconductor UART. + * + * If we have to do this here because some non-National + * Semiconductor clone chips lock up if you try writing to the + * LSR register (which serial_icr_read does) + */ + if (port->type == PORT_16550A) { + /* + * EFR [4] must be set else this test fails + * + * This shouldn't be necessary, but Mike Hudson + * (Exoray@isys.ca) claims that it's needed for 952 + * dual UART's (which are not recommended for new designs). + */ + port->port_acr = 0; + serial_out(port, UART_LCR, 0xBF); + serial_out(port, UART_EFR, 0x10); + serial_out(port, UART_LCR, 0x00); + /* Check for Oxford Semiconductor 16C950 */ + scratch = serial_icr_read(port, UART_ID1); + scratch2 = serial_icr_read(port, UART_ID2); + scratch3 = serial_icr_read(port, UART_ID3); + + if (scratch == 0x16 && scratch2 == 0xC9 && + (scratch3 == 0x50 || scratch3 == 0x52 || + scratch3 == 0x54)) { + port->type = PORT_16C950; + port->port_rev = serial_icr_read(port, UART_REV) | + (scratch3 << 8); + return; + } + } + + /* + * We check for a XR16C850 by setting DLL and DLM to 0, and then + * reading back DLL and DLM. The chip type depends on the DLM + * value read back: + * 0x10 - XR16C850 and the DLL contains the chip revision. + * 0x12 - XR16C2850. + * 0x14 - XR16C854. + */ + + /* Save the DLL and DLM */ + + serial_outp(port, UART_LCR, UART_LCR_DLAB); + scratch3 = serial_inp(port, UART_DLL); + scratch4 = serial_inp(port, UART_DLM); + + serial_outp(port, UART_DLL, 0); + serial_outp(port, UART_DLM, 0); + scratch2 = serial_inp(port, UART_DLL); + scratch = serial_inp(port, UART_DLM); + serial_outp(port, UART_LCR, 0); + + if (scratch == 0x10 || scratch == 0x12 || scratch == 0x14) { + if (scratch == 0x10) + port->port_rev = scratch2; + port->type = PORT_16850; + return; + } + + /* Restore the DLL and DLM */ + + serial_outp(port, UART_LCR, UART_LCR_DLAB); + serial_outp(port, UART_DLL, scratch3); + serial_outp(port, UART_DLM, scratch4); + serial_outp(port, UART_LCR, 0); + + /* + * We distinguish between the '654 and the '650 by counting + * how many bytes are in the FIFO. I'm using this for now, + * since that's the technique that was sent to me in the + * serial driver update, but I'm not convinced this works. + * I've had problems doing this in the past. -TYT + */ + if (size_fifo(port) == 64) + port->type = PORT_16654; + else + port->type = PORT_16650V2; +} + +/* + * This routine is called by rs_init() to initialize a specific serial + * port. It determines what type of UART chip this serial port is + * using: 8250, 16450, 16550, 16550A. The important question is + * whether or not this UART is a 16550A or not, since this will + * determine whether or not we can use its FIFO features or not. + */ +static void autoconfig(struct uart_port *port, unsigned int probeflags) +{ + unsigned char status1, status2, scratch, scratch2, scratch3; + unsigned char save_lcr, save_mcr; + unsigned long flags; + +#ifdef SERIAL_DEBUG_AUTOCONF + printk("Testing ttyS%d (0x%04x, 0x%08lx)...\n", + port->line, port->iobase, port->membase); +#endif + + if (!port->iobase && !port->membase) + return; + + save_flags(flags); cli(); + + if (!(port->flags & ASYNC_BUGGY_UART)) { + /* + * Do a simple existence test first; if we fail this, + * there's no point trying anything else. + * + * 0x80 is used as a nonsense port to prevent against + * false positives due to ISA bus float. The + * assumption is that 0x80 is a non-existent port; + * which should be safe since include/asm/io.h also + * makes this assumption. + */ + scratch = serial_inp(port, UART_IER); + serial_outp(port, UART_IER, 0); +#ifdef __i386__ + outb(0xff, 0x080); +#endif + scratch2 = serial_inp(port, UART_IER); + serial_outp(port, UART_IER, 0x0F); +#ifdef __i386__ + outb(0, 0x080); +#endif + scratch3 = serial_inp(port, UART_IER); + serial_outp(port, UART_IER, scratch); + if (scratch2 || scratch3 != 0x0F) { +#ifdef SERIAL_DEBUG_AUTOCONF + printk("serial: ttyS%d: simple autoconfig failed " + "(%02x, %02x)\n", port->line, + scratch2, scratch3); +#endif + restore_flags(flags); + return; /* We failed; there's nothing here */ + } + } + + save_mcr = serial_in(port, UART_MCR); + save_lcr = serial_in(port, UART_LCR); + + /* + * Check to see if a UART is really there. Certain broken + * internal modems based on the Rockwell chipset fail this + * test, because they apparently don't implement the loopback + * test mode. So this test is skipped on the COM 1 through + * COM 4 ports. This *should* be safe, since no board + * manufacturer would be stupid enough to design a board + * that conflicts with COM 1-4 --- we hope! + */ + if (!(port->flags & ASYNC_SKIP_TEST)) { + serial_outp(port, UART_MCR, UART_MCR_LOOP | 0x0A); + status1 = serial_inp(port, UART_MSR) & 0xF0; + serial_outp(port, UART_MCR, save_mcr); + if (status1 != 0x90) { +#ifdef SERIAL_DEBUG_AUTOCONF + printk("serial: ttyS%d: no UART loopback failed\n", + port->line); +#endif + restore_flags(flags); + return; + } + } + serial_outp(port, UART_LCR, 0xBF); /* set up for StarTech test */ + serial_outp(port, UART_EFR, 0); /* EFR is the same as FCR */ + serial_outp(port, UART_LCR, 0); + serial_outp(port, UART_FCR, UART_FCR_ENABLE_FIFO); + scratch = serial_in(port, UART_IIR) >> 6; + switch (scratch) { + case 0: + port->type = PORT_16450; + break; + case 1: + port->type = PORT_UNKNOWN; + break; + case 2: + port->type = PORT_16550; + break; + case 3: + port->type = PORT_16550A; + break; + } + if (port->type == PORT_16550A) { + /* Check for Startech UART's */ + serial_outp(port, UART_LCR, UART_LCR_DLAB); + if (serial_in(port, UART_EFR) == 0) { + port->type = PORT_16650; + } else { + serial_outp(port, UART_LCR, 0xBF); + if (serial_in(port, UART_EFR) == 0) + autoconfig_startech_uarts(port); + } + } + if (port->type == PORT_16550A) { + /* Check for TI 16750 */ + serial_outp(port, UART_LCR, save_lcr | UART_LCR_DLAB); + serial_outp(port, UART_FCR, + UART_FCR_ENABLE_FIFO | UART_FCR7_64BYTE); + scratch = serial_in(port, UART_IIR) >> 5; + if (scratch == 7) { + /* + * If this is a 16750, and not a cheap UART + * clone, then it should only go into 64 byte + * mode if the UART_FCR7_64BYTE bit was set + * while UART_LCR_DLAB was latched. + */ + serial_outp(port, UART_FCR, UART_FCR_ENABLE_FIFO); + serial_outp(port, UART_LCR, 0); + serial_outp(port, UART_FCR, + UART_FCR_ENABLE_FIFO | UART_FCR7_64BYTE); + scratch = serial_in(port, UART_IIR) >> 5; + if (scratch == 6) + port->type = PORT_16750; + } + serial_outp(port, UART_FCR, UART_FCR_ENABLE_FIFO); + } +#if defined(CONFIG_SERIAL_RSA) && defined(MODULE) + /* + * Only probe for RSA ports if we got the region. + */ + if (port->type == PORT_16550A && probeflags & PROBE_RSA) { + int i; + + for (i = 0 ; i < PORT_RSA_MAX ; ++i) { + if (!probe_rsa[i] && !force_rsa[i]) + break; + if (((probe_rsa[i] != port->iobase) || + check_region(port->iobase + UART_RSA_BASE, 16)) && + (force_rsa[i] != port->iobase)) + continue; + if (!enable_rsa(port)) + continue; + port->type = PORT_RSA; + port->uartclk = SERIAL_RSA_BAUD_BASE * 16; + break; + } + } +#endif + serial_outp(port, UART_LCR, save_lcr); + if (port->type == PORT_16450) { + scratch = serial_in(port, UART_SCR); + serial_outp(port, UART_SCR, 0xa5); + status1 = serial_in(port, UART_SCR); + serial_outp(port, UART_SCR, 0x5a); + status2 = serial_in(port, UART_SCR); + serial_outp(port, UART_SCR, scratch); + + if ((status1 != 0xa5) || (status2 != 0x5a)) + port->type = PORT_8250; + } + port->fifosize = uart_config[port->type].dfl_xmit_fifo_size; + + if (port->type == PORT_UNKNOWN) { + restore_flags(flags); + return; + } + +#ifdef CONFIG_SERIAL_RSA + if (port->iobase && port->type == PORT_RSA) { + release_region(port->iobase, 8); + request_region(port->iobase + UART_RSA_BASE, 16, + "serial_rsa"); + } +#endif + + /* + * Reset the UART. + */ +#ifdef CONFIG_SERIAL_RSA + if (port->type == PORT_RSA) + serial_outp(port, UART_RSA_FRR, 0); +#endif + serial_outp(port, UART_MCR, save_mcr); + serial_outp(port, UART_FCR, (UART_FCR_ENABLE_FIFO | + UART_FCR_CLEAR_RCVR | + UART_FCR_CLEAR_XMIT)); + serial_outp(port, UART_FCR, 0); + (void)serial_in(port, UART_RX); + serial_outp(port, UART_IER, 0); + + restore_flags(flags); +} + +static void autoconfig_irq(struct uart_port *port) +{ + unsigned char save_mcr, save_ier; + unsigned long irqs; + int irq; + +#ifdef CONFIG_SERIAL_MANY_PORTS + unsigned char save_ICP = 0; + unsigned short ICP = 0; + + if (port->flags & ASYNC_FOURPORT) { + ICP = (port->iobase & 0xfe0) | 0x1f; + save_ICP = inb_p(ICP); + outb_p(0x80, ICP); + (void) inb_p(ICP); + } +#endif + + /* forget possible initially masked and pending IRQ */ + probe_irq_off(probe_irq_on()); + save_mcr = serial_inp(port, UART_MCR); + save_ier = serial_inp(port, UART_IER); + serial_outp(port, UART_MCR, UART_MCR_OUT1 | UART_MCR_OUT2); + + irqs = probe_irq_on(); + serial_outp(port, UART_MCR, 0); + udelay (10); + if (port->flags & ASYNC_FOURPORT) { + serial_outp(port, UART_MCR, + UART_MCR_DTR | UART_MCR_RTS); + } else { + serial_outp(port, UART_MCR, + UART_MCR_DTR | UART_MCR_RTS | UART_MCR_OUT2); + } + serial_outp(port, UART_IER, 0x0f); /* enable all intrs */ + (void)serial_inp(port, UART_LSR); + (void)serial_inp(port, UART_RX); + (void)serial_inp(port, UART_IIR); + (void)serial_inp(port, UART_MSR); + serial_outp(port, UART_TX, 0xFF); + udelay (20); + irq = probe_irq_off(irqs); + + serial_outp(port, UART_MCR, save_mcr); + serial_outp(port, UART_IER, save_ier); +#ifdef CONFIG_SERIAL_MANY_PORTS + if (port->flags & ASYNC_FOURPORT) + outb_p(save_ICP, ICP); +#endif + port->irq = (irq > 0)? irq : 0; +} + +static void serial8250_stop_tx(struct uart_port *port, u_int from_tty) +{ + if (port->port_ier & UART_IER_THRI) { + port->port_ier &= ~UART_IER_THRI; + serial_out(port, UART_IER, port->port_ier); + } + if (port->type == PORT_16C950) { + port->port_acr |= UART_ACR_TXDIS; + serial_icr_write(port, UART_ACR, port->port_acr); + } +} + +static void serial8250_start_tx(struct uart_port *port, u_int nonempty, u_int from_tty) +{ + if (nonempty && !(port->port_ier & UART_IER_THRI)) { + port->port_ier |= UART_IER_THRI; + serial_out(port, UART_IER, port->port_ier); + } + /* + * We only do this from uart_start + */ + if (from_tty && port->type == PORT_16C950) { + port->port_acr &= ~UART_ACR_TXDIS; + serial_icr_write(port, UART_ACR, port->port_acr); + } +} + +static void serial8250_stop_rx(struct uart_port *port) +{ + port->port_ier &= ~UART_IER_RLSI; + port->read_status_mask &= ~UART_LSR_DR; + serial_out(port, UART_IER, port->port_ier); +} + +static void serial8250_enable_ms(struct uart_port *port) +{ + port->port_ier |= UART_IER_MSI; + serial_out(port, UART_IER, port->port_ier); +} + +static _INLINE_ void +receive_chars(struct uart_info *info, int *status, struct pt_regs *regs) +{ + struct tty_struct *tty = info->tty; + struct uart_port *port = info->port; + unsigned char ch; + int max_count = 256; + + do { + if (tty->flip.count >= TTY_FLIPBUF_SIZE) { + tty->flip.tqueue.routine((void *)tty); + if (tty->flip.count >= TTY_FLIPBUF_SIZE) + return; // if TTY_DONT_FLIP is set + } + ch = serial_inp(port, UART_RX); + *tty->flip.char_buf_ptr = ch; + *tty->flip.flag_buf_ptr = TTY_NORMAL; + port->icount.rx++; + + if (*status & (UART_LSR_BI | UART_LSR_PE | + UART_LSR_FE | UART_LSR_OE)) { + /* + * For statistics only + */ + if (*status & UART_LSR_BI) { + *status &= ~(UART_LSR_FE | UART_LSR_PE); + port->icount.brk++; + /* + * We do the SysRQ and SAK checking + * here because otherwise the break + * may get masked by ignore_status_mask + * or read_status_mask. + */ + uart_handle_break(info, &serial8250_console); + } else if (*status & UART_LSR_PE) + port->icount.parity++; + else if (*status & UART_LSR_FE) + port->icount.frame++; + if (*status & UART_LSR_OE) + port->icount.overrun++; + + /* + * Mask off conditions which should be ingored. + */ + *status &= port->read_status_mask; + +#ifdef CONFIG_SERIAL_8250_CONSOLE + if (port->line == serial8250_console.index) { + /* Recover the break flag from console xmit */ + *status |= lsr_break_flag; + lsr_break_flag = 0; + } +#endif + if (*status & UART_LSR_BI) { +#ifdef SERIAL_DEBUG_INTR + printk("handling break...."); +#endif + *tty->flip.flag_buf_ptr = TTY_BREAK; + } else if (*status & UART_LSR_PE) + *tty->flip.flag_buf_ptr = TTY_PARITY; + else if (*status & UART_LSR_FE) + *tty->flip.flag_buf_ptr = TTY_FRAME; + } + if (uart_handle_sysrq_char(info, ch, regs)) + goto ignore_char; + if ((*status & port->ignore_status_mask) == 0) { + tty->flip.flag_buf_ptr++; + tty->flip.char_buf_ptr++; + tty->flip.count++; + } + if ((*status & UART_LSR_OE) && + tty->flip.count < TTY_FLIPBUF_SIZE) { + /* + * Overrun is special, since it's reported + * immediately, and doesn't affect the current + * character. + */ + *tty->flip.flag_buf_ptr = TTY_OVERRUN; + tty->flip.flag_buf_ptr++; + tty->flip.char_buf_ptr++; + tty->flip.count++; + } + ignore_char: + *status = serial_inp(port, UART_LSR); + } while ((*status & UART_LSR_DR) && (max_count-- > 0)); + tty_flip_buffer_push(tty); +} + +static _INLINE_ void transmit_chars(struct uart_info *info, int *intr_done) +{ + struct uart_port *port = info->port; + int count; + + if (port->x_char) { + serial_outp(port, UART_TX, port->x_char); + port->icount.tx++; + port->x_char = 0; + if (intr_done) + *intr_done = 0; + return; + } + if (info->xmit.head == info->xmit.tail + || info->tty->stopped + || info->tty->hw_stopped) { + serial8250_stop_tx(port, 0); + return; + } + + count = port->fifosize; + do { + serial_out(port, UART_TX, info->xmit.buf[info->xmit.tail]); + info->xmit.tail = (info->xmit.tail + 1) & (UART_XMIT_SIZE - 1); + port->icount.tx++; + if (info->xmit.head == info->xmit.tail) + break; + } while (--count > 0); + + if (CIRC_CNT(info->xmit.head, info->xmit.tail, UART_XMIT_SIZE) < + WAKEUP_CHARS) + uart_event(info, EVT_WRITE_WAKEUP); + +#ifdef SERIAL_DEBUG_INTR + printk("THRE..."); +#endif + if (intr_done) + *intr_done = 0; + + if (info->xmit.head == info->xmit.tail) + serial8250_stop_tx(info->port, 0); +} + +static _INLINE_ void check_modem_status(struct uart_info *info) +{ + struct uart_port *port = info->port; + int status; + + status = serial_in(port, UART_MSR); + + if (status & UART_MSR_ANY_DELTA) { + if (status & UART_MSR_TERI) + port->icount.rng++; + if (status & UART_MSR_DDSR) + port->icount.dsr++; + if (status & UART_MSR_DDCD) + uart_handle_dcd_change(info, status & UART_MSR_DCD); + if (status & UART_MSR_DCTS) + uart_handle_cts_change(info, status & UART_MSR_CTS); + + wake_up_interruptible(&info->delta_msr_wait); + } +} + +/* + * This handles the interrupt from one port. + */ +static inline void +serial8250_handle_port(struct uart_info *info, struct pt_regs *regs) +{ + int status = serial_inp(info->port, UART_LSR); +#ifdef SERIAL_DEBUG_INTR + printk("status = %x...", status); +#endif + if (status & UART_LSR_DR) + receive_chars(info, &status, regs); + check_modem_status(info); + if (status & UART_LSR_THRE) + transmit_chars(info, 0); +} + +#ifdef CONFIG_SERIAL_8250_SHARE_IRQ +/* + * This is the serial driver's generic interrupt routine + */ +static void rs_interrupt(int irq, void *dev_id, struct pt_regs *regs) +{ + struct uart_info *info, *end_mark = NULL; + int pass_counter = 0; +#ifdef CONFIG_SERIAL_8250_MULTIPORT + int first_multi = 0; + unsigned long port_monitor = rs_multiport[irq].port_monitor; +#endif + +#ifdef SERIAL_DEBUG_INTR + printk("rs_interrupt(%d)...", irq); +#endif + + info = *(struct uart_info **)dev_id; + if (!info) + return; + +#ifdef CONFIG_SERIAL_8250_MULTIPORT + if (port_monitor) + first_multi = inb(port_monitor); +#endif + + do { + if (!info->tty || + (serial_in(info->port, UART_IIR) & UART_IIR_NO_INT)) { + if (!end_mark) + end_mark = info; + goto next; + } +#ifdef SERIAL_DEBUG_INTR + printk("IIR = %x...", serial_in(info->port, UART_IIR)); +#endif + end_mark = NULL; + + serial8250_handle_port(info, regs); + + next: + info = info->next_info; + if (info) + continue; + info = *(struct uart_info **)dev_id; + if (pass_counter++ > RS_ISR_PASS_LIMIT) { +#if 0 + printk("rs loop break\n"); +#endif + break; /* Prevent infinite loops */ + } + } while (end_mark != info); +#ifdef CONFIG_SERIAL_8250_MULTIPORT + if (port_monitor) + printk("rs port monitor (normal) irq %d: 0x%x, 0x%x\n", + info->port->irq, first_multi, inb(port_monitor)); +#endif +#ifdef SERIAL_DEBUG_INTR + printk("end.\n"); +#endif +} +#endif + +/* + * This is the serial driver's interrupt routine for a single port + */ +static void rs_interrupt_single(int irq, void *dev_id, struct pt_regs *regs) +{ + struct uart_info *info; + int pass_counter = 0; +#ifdef CONFIG_SERIAL_8250_MULTIPORT + int first_multi = 0; + unsigned long port_monitor = rs_multiport[irq].port_monitor; +#endif + +#ifdef SERIAL_DEBUG_INTR + printk("rs_interrupt_single(%d)...", irq); +#endif + + info = *(struct uart_info **)dev_id; + if (!info || !info->tty) + return; + +#ifdef CONFIG_SERIAL_8250_MULTIPORT + if (port_monitor) + first_multi = inb(port_monitor); +#endif + + do { + serial8250_handle_port(info, regs); + if (pass_counter++ > RS_ISR_PASS_LIMIT) { +#if 0 + printk("rs_single loop break.\n"); +#endif + break; + } +#ifdef SERIAL_DEBUG_INTR + printk("IIR = %x...", serial_in(info->port, UART_IIR)); +#endif + } while (!(serial_in(info->port, UART_IIR) & UART_IIR_NO_INT)); +#ifdef CONFIG_SERIAL_8250_MULTIPORT + if (port_monitor) + printk("rs port monitor (single) irq %d: 0x%x, 0x%x\n", + info->port->irq, first_multi, inb(port_monitor)); +#endif +#ifdef SERIAL_DEBUG_INTR + printk("end.\n"); +#endif +} + +#ifdef CONFIG_SERIAL_8250_MULTIPORT +/* + * This is the serial driver's interrupt routine for multiport boards + */ +static void rs_interrupt_multi(int irq, void *dev_id, struct pt_regs *regs) +{ + struct uart_info *info; + int pass_counter = 0; + struct rs_multiport_struct *multi = &rs_multiport[irq]; + int first_multi = 0; + +#ifdef SERIAL_DEBUG_INTR + printk("rs_interrupt_multi(%d)...", irq); +#endif + + info = *(struct uart_info **)dev_id; + if (!info) + return; + + if (!multi->port1) { + /* should never happen */ + printk("rs_interrupt_multi: port1 NULL!\n"); + return; + } + if (multi->port_monitor) + first_multi = inb(multi->port_monitor); + + while (1) { + if (!info->tty || + (serial_in(info->port, UART_IIR) & UART_IIR_NO_INT)) + goto next; + + serial8250_handle_port(info, regs); + + next: + info = info->next; + if (info) + continue; + info = *(struct uart_info **)dev_id; + + /* + * The user was a bonehead, and misconfigured their + * multiport info. Rather than lock up the kernel + * in an infinite loop, if we loop too many times, + * print a message and break out of the loop. + */ + if (pass_counter++ > RS_ISR_PASS_LIMIT) { + printk("Misconfigured multiport serial info " + "for irq %d. Breaking out irq loop\n", irq); + break; + } + if (multi->port_monitor) + printk("rs port monitor irq %d: 0x%x, 0x%x\n", + info->port->irq, first_multi, + inb(multi->port_monitor)); + if ((inb(multi->port1) & multi->mask1) != multi->match1) + continue; + if (!multi->port2) + break; + if ((inb(multi->port2) & multi->mask2) != multi->match2) + continue; + if (!multi->port3) + break; + if ((inb(multi->port3) & multi->mask3) != multi->match3) + continue; + if (!multi->port4) + break; + if ((inb(multi->port4) & multi->mask4) != multi->match4) + continue; + break; + } +#ifdef SERIAL_DEBUG_INTR + printk("end.\n"); +#endif +} +#endif + +static u_int serial8250_tx_empty(struct uart_port *port) +{ + return serial_in(port, UART_LSR) & UART_LSR_TEMT ? TIOCSER_TEMT : 0; +} + +static u_int serial8250_get_mctrl(struct uart_port *port) +{ + unsigned long flags; + unsigned char status; + unsigned int ret; + + save_flags(flags); cli(); + status = serial_in(port, UART_MSR); + restore_flags(flags); + + ret = 0; + if (status & UART_MSR_DCD) + ret |= TIOCM_CAR; + if (status & UART_MSR_RI) + ret |= TIOCM_RNG; + if (status & UART_MSR_DSR) + ret |= TIOCM_DSR; + if (status & UART_MSR_CTS) + ret |= TIOCM_CTS; + return ret; +} + +static void serial8250_set_mctrl(struct uart_port *port, u_int mctrl) +{ + unsigned char mcr = 0; + + if (mctrl & TIOCM_RTS) + mcr |= UART_MCR_RTS; + if (mctrl & TIOCM_DTR) + mcr |= UART_MCR_DTR; + if (mctrl & TIOCM_OUT1) + mcr |= UART_MCR_OUT1; + if (mctrl & TIOCM_OUT2) + mcr |= UART_MCR_OUT2; + if (mctrl & TIOCM_LOOP) + mcr |= UART_MCR_LOOP; + + serial_out(port, UART_MCR, mcr); +} + +static void serial8250_break_ctl(struct uart_port *port, int break_state) +{ + if (break_state == -1) + port->port_lcr |= UART_LCR_SBC; + else + port->port_lcr &= ~UART_LCR_SBC; + serial_out(port, UART_LCR, port->port_lcr); +} + +static int serial8250_startup(struct uart_port *port, struct uart_info *info) +{ + void (*handler)(int, void *, struct pt_regs *); + int retval; + + if (port->type == PORT_16C950) { + /* Wake up and initialize UART */ + port->port_acr = 0; + serial_outp(port, UART_LCR, 0xBF); + serial_outp(port, UART_EFR, UART_EFR_ECB); + serial_outp(port, UART_IER, 0); + serial_outp(port, UART_LCR, 0); + serial_icr_write(port, UART_CSR, 0); /* Reset the UART */ + serial_outp(port, UART_LCR, 0xBF); + serial_outp(port, UART_EFR, UART_EFR_ECB); + serial_outp(port, UART_LCR, 0); + } + +#ifdef CONFIG_SERIAL_RSA + /* + * If this is an RSA port, see if we can kick it up to the + * higher speed clock. + */ + if (port->type == PORT_RSA) { + if (port->uartclk != SERIAL_RSA_BAUD_BASE * 16 && + enable_rsa(port)) + port->uartclk = SERIAL_RSA_BAUD_BASE * 16; + if (port->uartclk == SERIAL_RSA_BAUD_BASE * 16) + serial_outp(port, UART_RSA_FRR, 0); + } +#endif + + /* + * Clear the FIFO buffers and disable them. + * (they will be reeanbled in change_speed()) + */ + if (uart_config[port->type].flags & UART_CLEAR_FIFO) { + serial_outp(port, UART_FCR, UART_FCR_ENABLE_FIFO); + serial_outp(port, UART_FCR, UART_FCR_ENABLE_FIFO | + UART_FCR_CLEAR_RCVR | UART_FCR_CLEAR_XMIT); + serial_outp(port, UART_FCR, 0); + } + + /* + * Clear the interrupt registers. + */ + (void) serial_inp(port, UART_LSR); + (void) serial_inp(port, UART_RX); + (void) serial_inp(port, UART_IIR); + (void) serial_inp(port, UART_MSR); + + /* + * At this point, there's no way the LSR could still be 0xff; + * if it is, then bail out, because there's likely no UART + * here. + */ + if (!(port->flags & ASYNC_BUGGY_UART) && + (serial_inp(port, UART_LSR) == 0xff)) { + printk("ttyS%d: LSR safety check engaged!\n", port->line); + return -ENODEV; + } + + /* + * Allocate the IRQ if necessary + */ + if (port->irq && (!IRQ_ports[port->irq] || + !IRQ_ports[port->irq]->next_info)) { + handler = rs_interrupt_single; + if (IRQ_ports[port->irq]) { +#ifdef CONFIG_SERIAL_8250_SHARE_IRQ + handler = rs_interrupt; + free_irq(port->irq, &IRQ_ports[port->irq]); +#ifdef CONFIG_SERIAL_8250_MULTIPORT + if (rs_multiport[port->irq].port1) + handler = serial8250_interrupt_multi; +#endif +#else + return -EBUSY; +#endif /* CONFIG_SERIAL_8250_SHARE_IRQ */ + } + + retval = request_irq(port->irq, handler, SA_SHIRQ, + "serial", &IRQ_ports[port->irq]); + if (retval) + return retval; + } + + /* + * Insert serial port into IRQ chain. + */ + info->next_info = IRQ_ports[port->irq]; + IRQ_ports[port->irq] = info; + + /* + * Now, initialize the UART + */ + serial_outp(port, UART_LCR, UART_LCR_WLEN8); + +#ifdef CONFIG_SERIAL_MANY_PORTS + if (port->flags & ASYNC_FOURPORT) { + if (port->irq == 0) + info->mctrl |= TIOCM_OUT1; + } else +#endif + /* + * Most PC uarts need OUT2 raised to enable interrupts. + */ + if (port->irq != 0) + info->mctrl |= TIOCM_OUT2; + + /* FIXME: ALPHA_KLUDGE_MCR; */ + serial8250_set_mctrl(port, info->mctrl); + + /* + * Finally, enable interrupts. Note: Modem status interrupts + * are set via change_speed(), which will be occuring imminently + * anyway, so we don't enable them here. + */ + port->port_ier = UART_IER_RLSI | UART_IER_RDI; + serial_outp(port, UART_IER, port->port_ier); + +#ifdef CONFIG_SERIAL_MANY_PORTS + if (port->flags & ASYNC_FOURPORT) { + unsigned int ICP; + /* + * Enable interrupts on the AST Fourport board + */ + ICP = (port->iobase & 0xfe0) | 0x01f; + outb_p(0x80, ICP); + (void) inb_p(ICP); + } +#endif + + /* + * And clear the interrupt registers again for luck. + */ + (void) serial_inp(port, UART_LSR); + (void) serial_inp(port, UART_RX); + (void) serial_inp(port, UART_IIR); + (void) serial_inp(port, UART_MSR); + + return 0; +} + +static void serial8250_shutdown(struct uart_port *port, struct uart_info *info) +{ + struct uart_info **infop; + int retval; + + /* + * First, disable all intrs from the port. + */ + port->port_ier = 0; + serial_outp(port, UART_IER, 0); + + synchronize_irq(); + + /* + * unlink the serial port from the IRQ chain... + */ + for (infop = &IRQ_ports[port->irq]; *infop; infop = &(*infop)->next_info) + if (*infop == info) + break; + + if (*infop == info) + *infop = info->next_info; + + /* + * Free the IRQ, if necessary + */ + if (port->irq && (!IRQ_ports[port->irq] || + !IRQ_ports[port->irq]->next_info)) { + free_irq(port->irq, &IRQ_ports[port->irq]); + if (IRQ_ports[port->irq]) { + retval = request_irq(port->irq, rs_interrupt_single, + SA_SHIRQ, "serial", &IRQ_ports[port->irq]); + if (retval) + printk("serial shutdown: request_irq: error %d" + " couldn't reacquire IRQ.\n", retval); + } + } + +#ifdef CONFIG_SERIAL_MANY_PORTS + if (port->flags & ASYNC_FOURPORT) { + /* reset interrupts on the AST Fourport board */ + inb((port->iobase & 0xfe0) | 0x1f); + info->mctrl |= TIOCM_OUT1; + } else +#endif + info->mctrl &= ~TIOCM_OUT2; + + /* FIXME: ALPHA_KLUDGE_MCR; */ + serial8250_set_mctrl(port, info->mctrl); + + /* + * Disable break condition and FIFOs + */ + serial_out(port, UART_LCR, serial_inp(port, UART_LCR) & ~UART_LCR_SBC); + serial_outp(port, UART_FCR, UART_FCR_ENABLE_FIFO | + UART_FCR_CLEAR_RCVR | + UART_FCR_CLEAR_XMIT); + serial_outp(port, UART_FCR, 0); + +#ifdef CONFIG_SERIAL_RSA + /* + * Reset the RSA board back to 115kbps compat mode. + */ + if (port->type == PORT_RSA && + port->uartclk == SERIAL_RSA_BAUD_BASE * 16 && + disable_rsa(port)) + port->uartclk = SERIAL_RSA_BAUD_BASE_LO * 16; +#endif + + /* + * Read data port to reset things + */ + (void) serial_in(port, UART_RX); +} + +static void serial8250_change_speed(struct uart_port *port, u_int cflag, u_int iflag, u_int quot) +{ + unsigned char cval, fcr = 0; + unsigned long flags; + + switch (cflag & CSIZE) { + case CS5: cval = 0x00; break; + case CS6: cval = 0x01; break; + case CS7: cval = 0x02; break; + default: + case CS8: cval = 0x03; break; + } + + if (cflag & CSTOPB) + cval |= 0x04; + if (cflag & PARENB) + cval |= UART_LCR_PARITY; + if (!(cflag & PARODD)) + cval |= UART_LCR_EPAR; +#ifdef CMSPAR + if (cflag & CMSPAR) + cval |= UART_LCR_SPAR; +#endif + + /* + * Work around a bug in the Oxford Semiconductor 952 rev B + * chip which causes it to seriously miscalculate baud rates + * when DLL is 0. + */ + if ((quot & 0xff) == 0 && port->type == PORT_16C950 && + port->port_rev == 0x5201) + quot ++; + + if (uart_config[port->type].flags & UART_USE_FIFO) { + if ((port->uartclk / quot) < (2400 * 16)) + fcr = UART_FCR_ENABLE_FIFO | UART_FCR_TRIGGER_1; +#ifdef CONFIG_SERIAL_RSA + else if (port->type == PORT_RSA) + fcr = UART_FCR_ENABLE_FIFO | UART_FCR_TRIGGER_14; +#endif + else + fcr = UART_FCR_ENABLE_FIFO | UART_FCR_TRIGGER_8; + } + if (port->type == PORT_16750) + fcr |= UART_FCR7_64BYTE; + + port->read_status_mask = UART_LSR_OE | UART_LSR_THRE | UART_LSR_DR; + if (iflag & IGNPAR) + port->read_status_mask |= UART_LSR_FE | UART_LSR_PE; + if (iflag & (BRKINT | PARMRK)) + port->read_status_mask |= UART_LSR_BI; + + /* + * Characteres to ignore + */ + port->ignore_status_mask = 0; + if (iflag & IGNPAR) + port->ignore_status_mask |= UART_LSR_PE | UART_LSR_FE; + if (iflag & IGNBRK) { + port->ignore_status_mask |= UART_LSR_BI; + /* + * If we're ignoring parity and break indicators, + * ignore overruns too (for real raw support). + */ + if (iflag & IGNPAR) + port->ignore_status_mask |= UART_LSR_OE; + } + + /* + * ignore all characters if CREAD is not set + */ + if ((cflag & CREAD) == 0) + port->ignore_status_mask |= UART_LSR_DR; + + /* + * CTS flow control flag and modem status interrupts + */ + port->port_ier &= ~UART_IER_MSI; + if (port->flags & ASYNC_HARDPPS_CD || cflag & CRTSCTS || + !(cflag & CLOCAL)) + port->port_ier |= UART_IER_MSI; + + /* + * Ok, we're now changing the port state. Do it with + * interrupts disabled. + */ + save_flags(flags); cli(); + serial_out(port, UART_IER, port->port_ier); + + if (uart_config[port->type].flags & UART_STARTECH) { + serial_outp(port, UART_LCR, 0xBF); + serial_outp(port, UART_EFR, cflag & CRTSCTS ? UART_EFR_CTS :0); + } + serial_outp(port, UART_LCR, cval | UART_LCR_DLAB); /* set DLAB */ + serial_outp(port, UART_DLL, quot & 0xff); /* LS of divisor */ + serial_outp(port, UART_DLM, quot >> 8); /* MS of divisor */ + if (port->type == PORT_16750) + serial_outp(port, UART_FCR, fcr); /* set fcr */ + serial_outp(port, UART_LCR, cval); /* reset DLAB */ + port->port_lcr = cval; /* Save LCR */ + if (port->type != PORT_16750) { + if (fcr & UART_FCR_ENABLE_FIFO) { + /* emulated UARTs (Lucent Venus 167x) need two steps */ + serial_outp(port, UART_FCR, UART_FCR_ENABLE_FIFO); + } + serial_outp(port, UART_FCR, fcr); /* set fcr */ + } + restore_flags(flags); +} + +static void serial8250_pm(struct uart_port *port, u_int state, u_int oldstate) +{ + if (state) { + /* sleep */ + if (uart_config[port->type].flags & UART_STARTECH) { + /* Arrange to enter sleep mode */ + serial_outp(port, UART_LCR, 0xBF); + serial_outp(port, UART_EFR, UART_EFR_ECB); + serial_outp(port, UART_LCR, 0); + serial_outp(port, UART_IER, UART_IERX_SLEEP); + serial_outp(port, UART_LCR, 0xBF); + serial_outp(port, UART_EFR, 0); + serial_outp(port, UART_LCR, 0); + } + if (port->type == PORT_16750) { + /* Arrange to enter sleep mode */ + serial_outp(port, UART_IER, UART_IERX_SLEEP); + } + } else { + /* wake */ + if (uart_config[port->type].flags & UART_STARTECH) { + /* Wake up UART */ + serial_outp(port, UART_LCR, 0xBF); + serial_outp(port, UART_EFR, UART_EFR_ECB); + /* + * Turn off LCR == 0xBF so we actually set the IER + * register on the XR16C850 + */ + serial_outp(port, UART_LCR, 0); + serial_outp(port, UART_IER, 0); + /* + * Now reset LCR so we can turn off the ECB bit + */ + serial_outp(port, UART_LCR, 0xBF); + serial_outp(port, UART_EFR, 0); + /* + * For a XR16C850, we need to set the trigger levels + */ + if (port->type == PORT_16850) { + unsigned char fctr; + + fctr = serial_inp(port, UART_FCTR) & + ~(UART_FCTR_RX | UART_FCTR_TX); + serial_outp(port, UART_FCTR, fctr | + UART_FCTR_TRGD | + UART_FCTR_RX); + serial_outp(port, UART_TRG, UART_TRG_96); + serial_outp(port, UART_FCTR, fctr | + UART_FCTR_TRGD | + UART_FCTR_TX); + serial_outp(port, UART_TRG, UART_TRG_96); + } + serial_outp(port, UART_LCR, 0); + } + + if (port->type == PORT_16750) { + /* Wake up UART */ + serial_outp(port, UART_IER, 0); + } + } +} + +/* + * Resource handling. This is complicated by the fact that resources + * depend on the port type. Maybe we should be claiming the standard + * 8250 ports, and then trying to get other resources as necessary? + */ +static int +serial8250_request_std_resource(struct uart_port *port, struct resource **res) +{ + unsigned int size = 8 << port->regshift; + int ret = 0; + + switch (port->iotype) { + case SERIAL_IO_MEM: + if (port->mapbase) { + *res = request_mem_region(port->mapbase, size, "serial"); + if (!*res) + ret = -EBUSY; + } + break; + + case SERIAL_IO_HUB6: + case SERIAL_IO_PORT: + *res = request_region(port->iobase, size, "serial"); + if (!*res) + ret = -EBUSY; + break; + } + return ret; +} + +static int +serial8250_request_rsa_resource(struct uart_port *port, struct resource **res) +{ + unsigned long start, size = 8 << port->regshift; + int ret = 0; + + switch (port->iotype) { + case SERIAL_IO_MEM: + if (port->mapbase) { + start = port->mapbase; + start += UART_RSA_BASE << port->regshift; + *res = request_mem_region(start, size, "serial-rsa"); + if (!*res) + ret = -EBUSY; + } + break; + + case SERIAL_IO_HUB6: + case SERIAL_IO_PORT: + start = port->iobase; + start += UART_RSA_BASE << port->regshift; + *res = request_region(start, size, "serial-rsa"); + if (!*res) + ret = -EBUSY; + break; + } + + return ret; +} + +static void serial8250_release_port(struct uart_port *port) +{ + unsigned long start, offset = 0, size = 0; + + if (port->type == PORT_RSA) { + offset = UART_RSA_BASE << port->regshift; + size = 8; + } + + offset <<= port->regshift; + size <<= port->regshift; + + switch (port->iotype) { + case SERIAL_IO_MEM: + if (port->mapbase) { + /* + * Unmap the area. + */ + iounmap(port->membase); + port->membase = NULL; + + start = port->mapbase; + + if (size) + release_mem_region(start + offset, size); + release_mem_region(start, 8 << port->regshift); + } + break; + + case SERIAL_IO_HUB6: + case SERIAL_IO_PORT: + start = port->iobase; + + if (size) + release_region(start + offset, size); + release_region(start + offset, 8 << port->regshift); + break; + + default: + break; + } +} + +static int serial8250_request_port(struct uart_port *port) +{ + struct resource *res = NULL, *res_rsa = NULL; + int ret = -EBUSY; + + if (port->type == PORT_RSA) { + ret = serial8250_request_rsa_resource(port, &res_rsa); + if (ret) + return ret; + } + + ret = serial8250_request_std_resource(port, &res); + + /* + * If we have a mapbase, then request that as well. + */ + if (res != NULL && port->iotype == SERIAL_IO_MEM && + port->mapbase) { + int size = res->end - res->start + 1; + + port->membase = ioremap(port->mapbase, size); + if (!port->membase) + ret = -ENOMEM; + } + + if (ret) { + if (res_rsa) + release_resource(res_rsa); + if (res) + release_resource(res); + } + return ret; +} + +static void serial8250_config_port(struct uart_port *port, int flags) +{ + struct resource *res_std = NULL, *res_rsa = NULL; + int probeflags = PROBE_ANY; + int ret; + +#ifdef CONFIG_MCA + /* + * Don't probe for MCA ports on non-MCA machines. + */ + if (port->flags & ASYNC_BOOT_ONLYMCA && !MCA_bus) + return; +#endif + + /* + * Find the region that we can probe for. This in turn + * tells us whether we can probe for the type of port. + */ + ret = serial8250_request_std_resource(port, &res_std); + if (ret) + return; + + ret = serial8250_request_rsa_resource(port, &res_rsa); + if (ret) + probeflags &= ~PROBE_RSA; + + if (flags & UART_CONFIG_TYPE) + autoconfig(port, probeflags); + if (port->type != PORT_UNKNOWN && flags & UART_CONFIG_IRQ) + autoconfig_irq(port); + + /* + * If the port wasn't an RSA port, release the resource. + */ + if (port->type != PORT_RSA && res_rsa) + release_resource(res_rsa); + + if (port->type == PORT_UNKNOWN) + release_resource(res_std); +} + +static int +serial8250_verify_port(struct uart_port *port, struct serial_struct *ser) +{ + if (ser->irq >= NR_IRQS || ser->irq < 0 || + ser->baud_base < 9600 || ser->type < PORT_UNKNOWN || + ser->type > PORT_MAX_8250 || ser->type == PORT_CIRRUS || + ser->type == PORT_STARTECH) + return -EINVAL; + return 0; +} + +static const char * +serial8250_type(struct uart_port *port) +{ + int type = port->type; + + if (type >= PORT_MAX_8250) + type = 0; + return uart_config[type].name; +} + +static struct uart_ops serial8250_pops = { + tx_empty: serial8250_tx_empty, + set_mctrl: serial8250_set_mctrl, + get_mctrl: serial8250_get_mctrl, + stop_tx: serial8250_stop_tx, + start_tx: serial8250_start_tx, + stop_rx: serial8250_stop_rx, + enable_ms: serial8250_enable_ms, + break_ctl: serial8250_break_ctl, + startup: serial8250_startup, + shutdown: serial8250_shutdown, + change_speed: serial8250_change_speed, + pm: serial8250_pm, + type: serial8250_type, + release_port: serial8250_release_port, + request_port: serial8250_request_port, + config_port: serial8250_config_port, + verify_port: serial8250_verify_port, +}; + +static struct uart_port serial8250_ports[UART_NR]; + +static void __init serial8250_isa_init_ports(void) +{ + static int first = 1; + int i; + + if (!first) + return; + first = 0; + + for (i = 0; i < ARRAY_SIZE(old_serial_port); i++) { + serial8250_ports[i].iobase = old_serial_port[i].port; + serial8250_ports[i].irq = irq_cannonicalize(old_serial_port[i].irq); + serial8250_ports[i].uartclk = old_serial_port[i].base_baud * 16; + serial8250_ports[i].flags = old_serial_port[i].flags; + serial8250_ports[i].ops = &serial8250_pops; + } +} + +#ifdef CONFIG_SERIAL_8250_CONSOLE + +#define BOTH_EMPTY (UART_LSR_TEMT | UART_LSR_THRE) + +/* + * Wait for transmitter & holding register to empty + */ +static inline void wait_for_xmitr(struct uart_port *port) +{ + unsigned int status, tmout = 1000000; + + do { + status = serial_in(port, UART_LSR); + + if (status & UART_LSR_BI) + lsr_break_flag = UART_LSR_BI; + + if (--tmout == 0) + break; + } while ((status & BOTH_EMPTY) != BOTH_EMPTY); + + /* Wait for flow control if necessary */ + if (port->flags & ASYNC_CONS_FLOW) { + tmout = 1000000; + while (--tmout && + ((serial_in(port, UART_MSR) & UART_MSR_CTS) == 0)); + } +} + +/* + * Print a string to the serial port trying not to disturb + * any possible real use of the port... + * + * The console_lock must be held when we get here. + */ +static void serial8250_console_write(struct console *co, const char *s, u_int count) +{ + struct uart_port *port = serial8250_ports + co->index; + unsigned int ier; + int i; + + /* + * First save the UER then disable the interrupts + */ + ier = serial_in(port, UART_IER); + serial_out(port, UART_IER, 0); + + /* + * Now, do each character + */ + for (i = 0; i < count; i++, s++) { + wait_for_xmitr(port); + + /* + * Send the character out. + * If a LF, also do CR... + */ + serial_out(port, UART_TX, *s); + if (*s == 10) { + wait_for_xmitr(port); + serial_out(port, UART_TX, 13); + } + } + + /* + * Finally, wait for transmitter to become empty + * and restore the IER + */ + wait_for_xmitr(port); + serial_out(port, UART_IER, ier); +} + +static kdev_t serial8250_console_device(struct console *co) +{ + return MKDEV(TTY_MAJOR, 64 + co->index); +} + +static int __init serial8250_console_setup(struct console *co, char *options) +{ + struct uart_port *port; + int baud = 9600; + int bits = 8; + int parity = 'n'; + int flow = 'n'; + + /* + * Check whether an invalid uart number has been specified, and + * if so, search for the first available port that does have + * console support. + */ + port = uart_get_console(serial8250_ports, UART_NR, co); + + if (options) + uart_parse_options(options, &baud, &parity, &bits, &flow); + + return uart_set_options(port, co, baud, parity, bits, flow); +} + +static struct console serial8250_console = { + name: "ttyS", + write: serial8250_console_write, + device: serial8250_console_device, + setup: serial8250_console_setup, + flags: CON_PRINTBUFFER, + index: -1, +}; + +void __init serial8250_console_init(void) +{ + serial8250_isa_init_ports(); + register_console(&serial8250_console); +} + +#define SERIAL8250_CONSOLE &serial8250_console +#else +#define SERIAL8250_CONSOLE NULL +#endif + +static struct uart_driver serial8250_reg = { + owner: THIS_MODULE, +#ifdef CONFIG_DEVFS_FS + normal_name: "tts/%d", + callout_name: "cua/%d", +#else + normal_name: "ttyS", + callout_name: "cua", +#endif + normal_major: TTY_MAJOR, + callout_major: TTYAUX_MAJOR, + normal_driver: &normal, + callout_driver: &callout, + table: serial8250_table, + termios: serial8250_termios, + termios_locked: serial8250_termios_locked, + minor: 64, + nr: ARRAY_SIZE(old_serial_port), + port: serial8250_ports, + cons: SERIAL8250_CONSOLE, +}; + +/* + * register_serial and unregister_serial allows for 16x50 serial ports to be + * configured at run-time, to support PCMCIA modems. + */ + +/** + * register_serial - configure a 16x50 serial port at runtime + * @req: request structure + * + * Configure the serial port specified by the request. If the + * port exists and is in use an error is returned. If the port + * is not currently in the table it is added. + * + * The port is then probed and if neccessary the IRQ is autodetected + * If this fails an error is returned. + * + * On success the port is ready to use and the line number is returned. + */ +int register_serial(struct serial_struct *req) +{ + struct uart_port port; + + port.iobase = req->port; + port.membase = req->iomem_base; + port.irq = req->irq; + port.uartclk = req->baud_base * 16; + port.fifosize = req->xmit_fifo_size; + port.regshift = req->iomem_reg_shift; + port.iotype = req->io_type; + port.flags = req->flags | ASYNC_BOOT_AUTOCONF; + + if (HIGH_BITS_OFFSET) + port.iobase |= req->port_high << HIGH_BITS_OFFSET; + + /* + * If a clock rate wasn't specified by the low level + * driver, then default to the standard clock rate. + */ + if (port.uartclk == 0) + port.uartclk = BASE_BAUD * 16; + + return uart_register_port(&serial8250_reg, &port); +} + +void unregister_serial(int line) +{ + uart_unregister_port(&serial8250_reg, line); +} + +static int __init serial8250_init(void) +{ + serial8250_isa_init_ports(); + return uart_register_driver(&serial8250_reg); +} + +static void __exit serial8250_exit(void) +{ + uart_unregister_driver(&serial8250_reg); +} + +module_init(serial8250_init); +module_exit(serial8250_exit); + +EXPORT_SYMBOL(register_serial); +EXPORT_SYMBOL(unregister_serial); + +MODULE_LICENSE("GPL"); +MODULE_DESCRIPTION("Generic 8250/16x50 serial driver"); + diff -urN orig/drivers/serial/8250.h linux/drivers/serial/8250.h --- orig/drivers/serial/8250.h Thu Jan 1 01:00:00 1970 +++ linux/drivers/serial/8250.h Thu Oct 24 10:53:24 2002 @@ -0,0 +1,101 @@ +/* + * linux/drivers/char/serial_8250.h + * + * Driver for 8250/16550-type serial ports + * + * Based on drivers/char/serial.c, by Linus Torvalds, Theodore Ts'o. + * + * Copyright (C) 2001 Russell King. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * $Id: 8250.h,v 1.1.1.1.2.1 2002/10/24 09:53:24 rmk Exp $ + */ + +struct serial8250_probe { + struct module *owner; + int (*pci_init_one)(struct pci_dev *dev); + void (*pci_remove_one)(struct pci_dev *dev); + void (*pnp_init)(void); +}; + +int serial8250_register_probe(struct serial8250_probe *probe); +void serial8250_unregister_probe(struct serial8250_probe *probe); + +struct old_serial_port { + unsigned int uart; + unsigned int base_baud; + unsigned int port; + unsigned int irq; + unsigned int flags; +}; + +#undef SERIAL_PARANOIA_CHECK +#define CONFIG_SERIAL_NOPAUSE_IO +#define SERIAL_DO_RESTART + +#ifdef CONFIG_PCI +#ifndef CONFIG_SERIAL_SHARE_IRQ +#define CONFIG_SERIAL_SHARE_IRQ +#endif +#ifndef CONFIG_SERIAL_MANY_PORTS +#define CONFIG_SERIAL_MANY_PORTS +#endif +#endif + +#if defined(CONFIG_ISAPNP)|| (defined(CONFIG_ISAPNP_MODULE) && defined(MODULE)) +#ifndef ENABLE_SERIAL_PNP +#define ENABLE_SERIAL_PNP +#endif +#endif + +/* Set of debugging defines */ + +#undef SERIAL_DEBUG_INTR +#undef SERIAL_DEBUG_PCI +#undef SERIAL_DEBUG_AUTOCONF + +/* Sanity checks */ + +#ifdef CONFIG_SERIAL_MULTIPORT +#ifndef CONFIG_SERIAL_SHARE_IRQ +#define CONFIG_SERIAL_SHARE_IRQ +#endif +#endif + +#ifdef CONFIG_HUB6 +#ifndef CONFIG_SERIAL_MANY_PORTS +#define CONFIG_SERIAL_MANY_PORTS +#endif +#ifndef CONFIG_SERIAL_SHARE_IRQ +#define CONFIG_SERIAL_SHARE_IRQ +#endif +#endif + +#ifdef MODULE +#undef CONFIG_SERIAL_CONSOLE +#endif + +#define CONFIG_SERIAL_RSA + +#define RS_ISR_PASS_LIMIT 256 + +#if defined(__i386__) && (defined(CONFIG_M386) || defined(CONFIG_M486)) +#define SERIAL_INLINE +#endif + +#ifdef SERIAL_INLINE +#define _INLINE_ inline +#else +#define _INLINE_ +#endif + +#define PROBE_RSA (1 << 0) +#define PROBE_ANY (~0) + +#define HIGH_BITS_OFFSET ((sizeof(long)-sizeof(int))*8) + + diff -urN orig/drivers/serial/8250_pci.c linux/drivers/serial/8250_pci.c --- orig/drivers/serial/8250_pci.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/serial/8250_pci.c Wed Mar 19 16:54:20 2003 @@ -0,0 +1,1080 @@ +/* + * linux/drivers/char/serial_8250_pci.c + * + * Probe module for 8250/16550-type PCI serial ports. + * + * Based on drivers/char/serial.c, by Linus Torvalds, Theodore Ts'o. + * + * Copyright (C) 2001 Russell King, All Rights Reserved. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License. + * + * $Id: 8250_pci.c,v 1.8.2.1 2002/10/24 09:53:24 rmk Exp $ + */ +#include +#include +#include +#include +#include +#include +#include +#include +#include + +/* 2.4.6 compatibility cruft ;( */ +#define pci_board __pci_board +#include +#undef pci_board + +#include +#include +#include + +#include "8250.h" + +#ifndef IS_PCI_REGION_IOPORT +#define IS_PCI_REGION_IOPORT(dev, r) (pci_resource_flags((dev), (r)) & \ + IORESOURCE_IO) +#endif +#ifndef IS_PCI_REGION_IOMEM +#define IS_PCI_REGION_IOMEM(dev, r) (pci_resource_flags((dev), (r)) & \ + IORESOURCE_MEM) +#endif +#ifndef PCI_IRQ_RESOURCE +#define PCI_IRQ_RESOURCE(dev, r) ((dev)->irq_resource[r].start) +#endif + +#ifndef pci_get_subvendor +#define pci_get_subvendor(dev) ((dev)->subsystem_vendor) +#define pci_get_subdevice(dev) ((dev)->subsystem_device) +#endif + +struct serial_private { + unsigned int nr; + struct pci_board *board; + int line[0]; +}; + +struct pci_board { + int flags; + int num_ports; + int base_baud; + int uart_offset; + int reg_shift; + int (*init_fn)(struct pci_dev *dev, struct pci_board *board, + int enable); + int first_uart_offset; +}; + +static int +get_pci_port(struct pci_dev *dev, struct pci_board *board, + struct serial_struct *req, int idx) +{ + unsigned long port; + int base_idx; + int max_port; + int offset; + + base_idx = SPCI_FL_GET_BASE(board->flags); + if (board->flags & SPCI_FL_BASE_TABLE) + base_idx += idx; + + if (board->flags & SPCI_FL_REGION_SZ_CAP) { + max_port = pci_resource_len(dev, base_idx) / 8; + if (idx >= max_port) + return 1; + } + + offset = board->first_uart_offset; + + /* Timedia/SUNIX uses a mixture of BARs and offsets */ + /* Ugh, this is ugly as all hell --- TYT */ + if(dev->vendor == PCI_VENDOR_ID_TIMEDIA ) /* 0x1409 */ + switch(idx) { + case 0: base_idx=0; + break; + case 1: base_idx=0; offset=8; + break; + case 2: base_idx=1; + break; + case 3: base_idx=1; offset=8; + break; + case 4: /* BAR 2*/ + case 5: /* BAR 3 */ + case 6: /* BAR 4*/ + case 7: base_idx=idx-2; /* BAR 5*/ + } + + /* Some Titan cards are also a little weird */ + if (dev->vendor == PCI_VENDOR_ID_TITAN && + (dev->device == PCI_DEVICE_ID_TITAN_400L || + dev->device == PCI_DEVICE_ID_TITAN_800L)) { + switch (idx) { + case 0: base_idx = 1; + break; + case 1: base_idx = 2; + break; + default: + base_idx = 4; + offset = 8 * (idx - 2); + } + } + + port = pci_resource_start(dev, base_idx) + offset; + + if ((board->flags & SPCI_FL_BASE_TABLE) == 0) + port += idx * (board->uart_offset ? board->uart_offset : 8); + + if (IS_PCI_REGION_IOPORT(dev, base_idx)) { + req->port = port; + if (HIGH_BITS_OFFSET) + req->port_high = port >> HIGH_BITS_OFFSET; + else + req->port_high = 0; + return 0; + } + req->io_type = SERIAL_IO_MEM; + req->iomem_base = ioremap(port, board->uart_offset); + req->iomem_reg_shift = board->reg_shift; + req->port = 0; + return 0; +} + +static _INLINE_ int get_pci_irq(struct pci_dev *dev, + struct pci_board *board, + int idx) +{ + int base_idx; + + if ((board->flags & SPCI_FL_IRQRESOURCE) == 0) + return dev->irq; + + base_idx = SPCI_FL_GET_IRQBASE(board->flags); + if (board->flags & SPCI_FL_IRQ_TABLE) + base_idx += idx; + + return PCI_IRQ_RESOURCE(dev, base_idx); +} + +/* + * Some PCI serial cards using the PLX 9050 PCI interface chip require + * that the card interrupt be explicitly enabled or disabled. This + * seems to be mainly needed on card using the PLX which also use I/O + * mapped memory. + */ +static int __devinit +pci_plx9050_fn(struct pci_dev *dev, struct pci_board *board, int enable) +{ + u8 data, *p, irq_config; + int pci_config; + + irq_config = 0x41; + pci_config = PCI_COMMAND_MEMORY; + if (dev->vendor == PCI_VENDOR_ID_PANACOM) + irq_config = 0x43; + if ((dev->vendor == PCI_VENDOR_ID_PLX) && + (dev->device == PCI_DEVICE_ID_PLX_ROMULUS)) { + /* + * As the megawolf cards have the int pins active + * high, and have 2 UART chips, both ints must be + * enabled on the 9050. Also, the UARTS are set in + * 16450 mode by default, so we have to enable the + * 16C950 'enhanced' mode so that we can use the deep + * FIFOs + */ + irq_config = 0x5b; + pci_config = PCI_COMMAND_MEMORY | PCI_COMMAND_IO; + } + + pci_read_config_byte(dev, PCI_COMMAND, &data); + + if (enable) + pci_write_config_byte(dev, PCI_COMMAND, + data | pci_config); + + /* enable/disable interrupts */ + p = ioremap(pci_resource_start(dev, 0), 0x80); + writel(enable ? irq_config : 0x00, (unsigned long)p + 0x4c); + iounmap(p); + + if (!enable) + pci_write_config_byte(dev, PCI_COMMAND, + data & ~pci_config); + return 0; +} + + +/* + * SIIG serial cards have an PCI interface chip which also controls + * the UART clocking frequency. Each UART can be clocked independently + * (except cards equiped with 4 UARTs) and initial clocking settings + * are stored in the EEPROM chip. It can cause problems because this + * version of serial driver doesn't support differently clocked UART's + * on single PCI card. To prevent this, initialization functions set + * high frequency clocking for all UART's on given card. It is safe (I + * hope) because it doesn't touch EEPROM settings to prevent conflicts + * with other OSes (like M$ DOS). + * + * SIIG support added by Andrey Panin , 10/1999 + * + * There is two family of SIIG serial cards with different PCI + * interface chip and different configuration methods: + * - 10x cards have control registers in IO and/or memory space; + * - 20x cards have control registers in standard PCI configuration space. + */ + +#define PCI_DEVICE_ID_SIIG_1S_10x (PCI_DEVICE_ID_SIIG_1S_10x_550 & 0xfffc) +#define PCI_DEVICE_ID_SIIG_2S_10x (PCI_DEVICE_ID_SIIG_2S_10x_550 & 0xfff8) + +static int __devinit +pci_siig10x_fn(struct pci_dev *dev, struct pci_board *board, int enable) +{ + u16 data, *p; + + if (!enable) return 0; + + p = ioremap(pci_resource_start(dev, 0), 0x80); + + switch (dev->device & 0xfff8) { + case PCI_DEVICE_ID_SIIG_1S_10x: /* 1S */ + data = 0xffdf; + break; + case PCI_DEVICE_ID_SIIG_2S_10x: /* 2S, 2S1P */ + data = 0xf7ff; + break; + default: /* 1S1P, 4S */ + data = 0xfffb; + break; + } + + writew(readw((unsigned long) p + 0x28) & data, (unsigned long) p + 0x28); + iounmap(p); + return 0; +} + +#define PCI_DEVICE_ID_SIIG_2S_20x (PCI_DEVICE_ID_SIIG_2S_20x_550 & 0xfffc) +#define PCI_DEVICE_ID_SIIG_2S1P_20x (PCI_DEVICE_ID_SIIG_2S1P_20x_550 & 0xfffc) + +static int __devinit +pci_siig20x_fn(struct pci_dev *dev, struct pci_board *board, int enable) +{ + u8 data; + + if (!enable) return 0; + + /* Change clock frequency for the first UART. */ + pci_read_config_byte(dev, 0x6f, &data); + pci_write_config_byte(dev, 0x6f, data & 0xef); + + /* If this card has 2 UART, we have to do the same with second UART. */ + if (((dev->device & 0xfffc) == PCI_DEVICE_ID_SIIG_2S_20x) || + ((dev->device & 0xfffc) == PCI_DEVICE_ID_SIIG_2S1P_20x)) { + pci_read_config_byte(dev, 0x73, &data); + pci_write_config_byte(dev, 0x73, data & 0xef); + } + return 0; +} + +/* Added for EKF Intel i960 serial boards */ +static int __devinit +pci_inteli960ni_fn(struct pci_dev *dev, + struct pci_board *board, + int enable) +{ + unsigned long oldval; + + if (!(pci_get_subdevice(dev) & 0x1000)) + return(-1); + + if (!enable) /* is there something to deinit? */ + return(0); + + /* is firmware started? */ + pci_read_config_dword(dev, 0x44, (void*) &oldval); + if (oldval == 0x00001000L) { /* RESET value */ + printk(KERN_DEBUG "Local i960 firmware missing"); + return(-1); + } + return(0); +} + +/* + * Timedia has an explosion of boards, and to avoid the PCI table from + * growing *huge*, we use this function to collapse some 70 entries + * in the PCI table into one, for sanity's and compactness's sake. + */ +static unsigned short timedia_single_port[] = { + 0x4025, 0x4027, 0x4028, 0x5025, 0x5027, 0 }; +static unsigned short timedia_dual_port[] = { + 0x0002, 0x4036, 0x4037, 0x4038, 0x4078, 0x4079, 0x4085, + 0x4088, 0x4089, 0x5037, 0x5078, 0x5079, 0x5085, 0x6079, + 0x7079, 0x8079, 0x8137, 0x8138, 0x8237, 0x8238, 0x9079, + 0x9137, 0x9138, 0x9237, 0x9238, 0xA079, 0xB079, 0xC079, + 0xD079, 0 }; +static unsigned short timedia_quad_port[] = { + 0x4055, 0x4056, 0x4095, 0x4096, 0x5056, 0x8156, 0x8157, + 0x8256, 0x8257, 0x9056, 0x9156, 0x9157, 0x9158, 0x9159, + 0x9256, 0x9257, 0xA056, 0xA157, 0xA158, 0xA159, 0xB056, + 0xB157, 0 }; +static unsigned short timedia_eight_port[] = { + 0x4065, 0x4066, 0x5065, 0x5066, 0x8166, 0x9066, 0x9166, + 0x9167, 0x9168, 0xA066, 0xA167, 0xA168, 0 }; +static struct timedia_struct { + int num; + unsigned short *ids; +} timedia_data[] = { + { 1, timedia_single_port }, + { 2, timedia_dual_port }, + { 4, timedia_quad_port }, + { 8, timedia_eight_port }, + { 0, 0 } +}; + +static int __devinit +pci_timedia_fn(struct pci_dev *dev, struct pci_board *board, int enable) +{ + int i, j; + unsigned short *ids; + + if (!enable) + return 0; + + for (i=0; timedia_data[i].num; i++) { + ids = timedia_data[i].ids; + for (j=0; ids[j]; j++) { + if (pci_get_subdevice(dev) == ids[j]) { + board->num_ports = timedia_data[i].num; + return 0; + } + } + } + return 0; +} + +static int __devinit +pci_xircom_fn(struct pci_dev *dev, struct pci_board *board, int enable) +{ + __set_current_state(TASK_UNINTERRUPTIBLE); + schedule_timeout(HZ/10); + return 0; +} + +/* + * This is the configuration table for all of the PCI serial boards + * which we support. It is directly indexed by the pci_board_num_t enum + * value, which is encoded in the pci_device_id PCI probe table's + * driver_data member. + */ +enum pci_board_num_t { + pbn_b0_1_115200, + pbn_default = 0, + + pbn_b0_2_115200, + pbn_b0_4_115200, + + pbn_b0_1_921600, + pbn_b0_2_921600, + pbn_b0_4_921600, + + pbn_b0_bt_1_115200, + pbn_b0_bt_2_115200, + pbn_b0_bt_1_460800, + pbn_b0_bt_2_460800, + + pbn_b1_1_115200, + pbn_b1_2_115200, + pbn_b1_4_115200, + pbn_b1_8_115200, + + pbn_b1_2_921600, + pbn_b1_4_921600, + pbn_b1_8_921600, + + pbn_b1_2_1382400, + pbn_b1_4_1382400, + pbn_b1_8_1382400, + + pbn_b2_8_115200, + pbn_b2_4_460800, + pbn_b2_8_460800, + pbn_b2_16_460800, + pbn_b2_4_921600, + pbn_b2_8_921600, + + pbn_b2_bt_1_115200, + pbn_b2_bt_2_115200, + pbn_b2_bt_4_115200, + pbn_b2_bt_2_921600, + + pbn_panacom, + pbn_panacom2, + pbn_panacom4, + pbn_plx_romulus, + pbn_oxsemi, + pbn_timedia, + pbn_intel_i960, + pbn_sgi_ioc3, +#ifdef CONFIG_DDB5074 + pbn_nec_nile4, +#endif +#if 0 + pbn_dci_pccom8, +#endif + pbn_xircom_combo, + + pbn_siig10x_0, + pbn_siig10x_1, + pbn_siig10x_2, + pbn_siig10x_4, + pbn_siig20x_0, + pbn_siig20x_2, + pbn_siig20x_4, + + pbn_computone_4, + pbn_computone_6, + pbn_computone_8, +}; + +static struct pci_board pci_boards[] __devinitdata = { + /* + * PCI Flags, Number of Ports, Base (Maximum) Baud Rate, + * Offset to get to next UART's registers, + * Register shift to use for memory-mapped I/O, + * Initialization function, first UART offset + */ + + /* Generic serial board, pbn_b0_1_115200, pbn_default */ + { SPCI_FL_BASE0, 1, 115200 }, /* pbn_b0_1_115200, + pbn_default */ + + { SPCI_FL_BASE0, 2, 115200 }, /* pbn_b0_2_115200 */ + { SPCI_FL_BASE0, 4, 115200 }, /* pbn_b0_4_115200 */ + + { SPCI_FL_BASE0, 1, 921600 }, /* pbn_b0_1_921600 */ + { SPCI_FL_BASE0, 2, 921600 }, /* pbn_b0_2_921600 */ + { SPCI_FL_BASE0, 4, 921600 }, /* pbn_b0_4_921600 */ + + { SPCI_FL_BASE0 | SPCI_FL_BASE_TABLE, 1, 115200 }, /* pbn_b0_bt_1_115200 */ + { SPCI_FL_BASE0 | SPCI_FL_BASE_TABLE, 2, 115200 }, /* pbn_b0_bt_2_115200 */ + { SPCI_FL_BASE0 | SPCI_FL_BASE_TABLE, 1, 460800 }, /* pbn_b0_bt_1_460800 */ + { SPCI_FL_BASE0 | SPCI_FL_BASE_TABLE, 2, 460800 }, /* pbn_b0_bt_2_460800 */ + + { SPCI_FL_BASE1, 1, 115200 }, /* pbn_b1_1_115200 */ + { SPCI_FL_BASE1, 2, 115200 }, /* pbn_b1_2_115200 */ + { SPCI_FL_BASE1, 4, 115200 }, /* pbn_b1_4_115200 */ + { SPCI_FL_BASE1, 8, 115200 }, /* pbn_b1_8_115200 */ + + { SPCI_FL_BASE1, 2, 921600 }, /* pbn_b1_2_921600 */ + { SPCI_FL_BASE1, 4, 921600 }, /* pbn_b1_4_921600 */ + { SPCI_FL_BASE1, 8, 921600 }, /* pbn_b1_8_921600 */ + + { SPCI_FL_BASE1, 2, 1382400 }, /* pbn_b1_2_1382400 */ + { SPCI_FL_BASE1, 4, 1382400 }, /* pbn_b1_4_1382400 */ + { SPCI_FL_BASE1, 8, 1382400 }, /* pbn_b1_8_1382400 */ + + { SPCI_FL_BASE2, 8, 115200 }, /* pbn_b2_8_115200 */ + { SPCI_FL_BASE2, 4, 460800 }, /* pbn_b2_4_460800 */ + { SPCI_FL_BASE2, 8, 460800 }, /* pbn_b2_8_460800 */ + { SPCI_FL_BASE2, 16, 460800 }, /* pbn_b2_16_460800 */ + { SPCI_FL_BASE2, 4, 921600 }, /* pbn_b2_4_921600 */ + { SPCI_FL_BASE2, 8, 921600 }, /* pbn_b2_8_921600 */ + + { SPCI_FL_BASE2 | SPCI_FL_BASE_TABLE, 1, 115200 }, /* pbn_b2_bt_1_115200 */ + { SPCI_FL_BASE2 | SPCI_FL_BASE_TABLE, 2, 115200 }, /* pbn_b2_bt_2_115200 */ + { SPCI_FL_BASE2 | SPCI_FL_BASE_TABLE, 4, 115200 }, /* pbn_b2_bt_4_115200 */ + { SPCI_FL_BASE2 | SPCI_FL_BASE_TABLE, 2, 921600 }, /* pbn_b2_bt_2_921600 */ + + { SPCI_FL_BASE2, 2, 921600, /* IOMEM */ /* pbn_panacom */ + 0x400, 7, pci_plx9050_fn }, + { SPCI_FL_BASE2 | SPCI_FL_BASE_TABLE, 2, 921600, /* pbn_panacom2 */ + 0x400, 7, pci_plx9050_fn }, + { SPCI_FL_BASE2 | SPCI_FL_BASE_TABLE, 4, 921600, /* pbn_panacom4 */ + 0x400, 7, pci_plx9050_fn }, + { SPCI_FL_BASE2, 4, 921600, /* pbn_plx_romulus */ + 0x20, 2, pci_plx9050_fn, 0x03 }, + /* This board uses the size of PCI Base region 0 to + * signal now many ports are available */ + { SPCI_FL_BASE0 | SPCI_FL_REGION_SZ_CAP, 32, 115200 }, /* pbn_oxsemi */ + { SPCI_FL_BASE_TABLE, 1, 921600, /* pbn_timedia */ + 0, 0, pci_timedia_fn }, + /* EKF addition for i960 Boards form EKF with serial port */ + { SPCI_FL_BASE0, 32, 921600, /* max 256 ports */ /* pbn_intel_i960 */ + 8<<2, 2, pci_inteli960ni_fn, 0x10000}, + { SPCI_FL_BASE0 | SPCI_FL_IRQRESOURCE, /* pbn_sgi_ioc3 */ + 1, 458333, 0, 0, 0, 0x20178 }, +#ifdef CONFIG_DDB5074 + /* + * NEC Vrc-5074 (Nile 4) builtin UART. + * Conditionally compiled in since this is a motherboard device. + */ + { SPCI_FL_BASE0, 1, 520833, /* pbn_nec_nile4 */ + 64, 3, NULL, 0x300 }, +#endif +#if 0 /* PCI_DEVICE_ID_DCI_PCCOM8 ? */ /* pbn_dci_pccom8 */ + { SPCI_FL_BASE3, 8, 115200, 8 }, +#endif + { SPCI_FL_BASE0, 1, 115200, /* pbn_xircom_combo */ + 0, 0, pci_xircom_fn }, + + { SPCI_FL_BASE2, 1, 460800, /* pbn_siig10x_0 */ + 0, 0, pci_siig10x_fn }, + { SPCI_FL_BASE2, 1, 921600, /* pbn_siig10x_1 */ + 0, 0, pci_siig10x_fn }, + { SPCI_FL_BASE2 | SPCI_FL_BASE_TABLE, 2, 921600, /* pbn_siig10x_2 */ + 0, 0, pci_siig10x_fn }, + { SPCI_FL_BASE2 | SPCI_FL_BASE_TABLE, 4, 921600, /* pbn_siig10x_4 */ + 0, 0, pci_siig10x_fn }, + { SPCI_FL_BASE0, 1, 921600, /* pbn_siix20x_0 */ + 0, 0, pci_siig20x_fn }, + { SPCI_FL_BASE0 | SPCI_FL_BASE_TABLE, 2, 921600, /* pbn_siix20x_2 */ + 0, 0, pci_siig20x_fn }, + { SPCI_FL_BASE0 | SPCI_FL_BASE_TABLE, 4, 921600, /* pbn_siix20x_4 */ + 0, 0, pci_siig20x_fn }, + + { SPCI_FL_BASE0, 4, 921600, /* IOMEM */ /* pbn_computone_4 */ + 0x40, 2, NULL, 0x200 }, + { SPCI_FL_BASE0, 6, 921600, /* IOMEM */ /* pbn_computone_6 */ + 0x40, 2, NULL, 0x200 }, + { SPCI_FL_BASE0, 8, 921600, /* IOMEM */ /* pbn_computone_8 */ + 0x40, 2, NULL, 0x200 }, +}; + +/* + * Given a complete unknown PCI device, try to use some heuristics to + * guess what the configuration might be, based on the pitiful PCI + * serial specs. Returns 0 on success, 1 on failure. + */ +static int __devinit serial_pci_guess_board(struct pci_dev *dev, + struct pci_board *board) +{ + int num_iomem = 0, num_port = 0, first_port = -1; + int i; + + /* + * If it is not a communications device or the programming + * interface is greater than 6, give up. + * + * (Should we try to make guesses for multiport serial devices + * later?) + */ + if ((((dev->class >> 8) != PCI_CLASS_COMMUNICATION_SERIAL) && + ((dev->class >> 8) != PCI_CLASS_COMMUNICATION_MODEM)) || + (dev->class & 0xff) > 6) + return 1; + + for (i=0; i < 6; i++) { + if (IS_PCI_REGION_IOPORT(dev, i)) { + num_port++; + if (first_port == -1) + first_port = i; + } + if (IS_PCI_REGION_IOMEM(dev, i)) + num_iomem++; + } + + /* + * If there is 1 or 0 iomem regions, and exactly one port, use + * it. + */ + if (num_iomem <= 1 && num_port == 1) { + board->flags = first_port; + return 0; + } + return 1; +} + +/* + * return -1 to refuse + */ +static int pci_init_one(struct pci_dev *dev, const struct pci_device_id *ent) +{ + struct serial_private *priv; + struct pci_board *board, tmp; + struct serial_struct serial_req; + int base_baud, rc, k; + + board = &pci_boards[ent->driver_data]; + + rc = pci_enable_device(dev); + if (rc) + return rc; + + if (ent->driver_data == pbn_default && + serial_pci_guess_board(dev, board)) + return -ENODEV; + else if (serial_pci_guess_board(dev, &tmp) == 0) { + printk(KERN_INFO "Redundant entry in serial pci_table. " + "Please send the output of\n" + "lspci -vv, this message (%d,%d,%d,%d)\n" + "and the manufacturer and name of " + "serial board or modem board\n" + "to serial-pci-info@lists.sourceforge.net.\n", + dev->vendor, dev->device, + pci_get_subvendor(dev), pci_get_subdevice(dev)); + } + + + priv = kmalloc(sizeof(struct serial_private) + + sizeof(unsigned int) * board->num_ports, + GFP_KERNEL); + if (!priv) + return -ENOMEM; + + /* + * Run the initialization function, if any + */ + if (board->init_fn && ((board->init_fn)(dev, board, 1) != 0)) { + kfree(priv); + return -ENODEV; + } + + base_baud = board->base_baud; + if (!base_baud) + base_baud = BASE_BAUD; + memset(&serial_req, 0, sizeof(serial_req)); + for (k=0; k < board->num_ports; k++) { + serial_req.irq = get_pci_irq(dev, board, k); + if (get_pci_port(dev, board, &serial_req, k)) + break; +#ifdef SERIAL_DEBUG_PCI + printk("Setup PCI/PNP port: port %x, irq %d, type %d\n", + serial_req.port, serial_req.irq, serial_req.io_type); +#endif + serial_req.flags = ASYNC_SKIP_TEST | ASYNC_AUTOPROBE; + serial_req.baud_base = base_baud; + priv->line[k] = register_serial(&serial_req); + if (priv->line[k] < 0) + break; + } + + priv->board = board; + priv->nr = k; + + pci_set_drvdata(dev, priv); + + return 0; +} + +static void pci_remove_one(struct pci_dev *dev) +{ + struct serial_private *priv = pci_get_drvdata(dev); + int i; + + pci_set_drvdata(dev, NULL); + + for (i = 0; i < priv->nr; i++) + unregister_serial(priv->line[i]); + + priv->board->init_fn(dev, priv->board, 0); + + kfree(priv); +} + +static struct pci_device_id serial_pci_tbl[] __devinitdata = { + { PCI_VENDOR_ID_V3, PCI_DEVICE_ID_V3_V960, + PCI_SUBVENDOR_ID_CONNECT_TECH, + PCI_SUBDEVICE_ID_CONNECT_TECH_BH8_232, 0, 0, + pbn_b1_8_1382400 }, + { PCI_VENDOR_ID_V3, PCI_DEVICE_ID_V3_V960, + PCI_SUBVENDOR_ID_CONNECT_TECH, + PCI_SUBDEVICE_ID_CONNECT_TECH_BH4_232, 0, 0, + pbn_b1_4_1382400 }, + { PCI_VENDOR_ID_V3, PCI_DEVICE_ID_V3_V960, + PCI_SUBVENDOR_ID_CONNECT_TECH, + PCI_SUBDEVICE_ID_CONNECT_TECH_BH2_232, 0, 0, + pbn_b1_2_1382400 }, + { PCI_VENDOR_ID_V3, PCI_DEVICE_ID_V3_V351, + PCI_SUBVENDOR_ID_CONNECT_TECH, + PCI_SUBDEVICE_ID_CONNECT_TECH_BH8_232, 0, 0, + pbn_b1_8_1382400 }, + { PCI_VENDOR_ID_V3, PCI_DEVICE_ID_V3_V351, + PCI_SUBVENDOR_ID_CONNECT_TECH, + PCI_SUBDEVICE_ID_CONNECT_TECH_BH4_232, 0, 0, + pbn_b1_4_1382400 }, + { PCI_VENDOR_ID_V3, PCI_DEVICE_ID_V3_V351, + PCI_SUBVENDOR_ID_CONNECT_TECH, + PCI_SUBDEVICE_ID_CONNECT_TECH_BH2_232, 0, 0, + pbn_b1_2_1382400 }, + { PCI_VENDOR_ID_V3, PCI_DEVICE_ID_V3_V351, + PCI_SUBVENDOR_ID_CONNECT_TECH, + PCI_SUBDEVICE_ID_CONNECT_TECH_BH8_485, 0, 0, + pbn_b1_8_921600 }, + { PCI_VENDOR_ID_V3, PCI_DEVICE_ID_V3_V351, + PCI_SUBVENDOR_ID_CONNECT_TECH, + PCI_SUBDEVICE_ID_CONNECT_TECH_BH8_485_4_4, 0, 0, + pbn_b1_8_921600 }, + { PCI_VENDOR_ID_V3, PCI_DEVICE_ID_V3_V351, + PCI_SUBVENDOR_ID_CONNECT_TECH, + PCI_SUBDEVICE_ID_CONNECT_TECH_BH4_485, 0, 0, + pbn_b1_4_921600 }, + { PCI_VENDOR_ID_V3, PCI_DEVICE_ID_V3_V351, + PCI_SUBVENDOR_ID_CONNECT_TECH, + PCI_SUBDEVICE_ID_CONNECT_TECH_BH4_485_2_2, 0, 0, + pbn_b1_4_921600 }, + { PCI_VENDOR_ID_V3, PCI_DEVICE_ID_V3_V351, + PCI_SUBVENDOR_ID_CONNECT_TECH, + PCI_SUBDEVICE_ID_CONNECT_TECH_BH2_485, 0, 0, + pbn_b1_2_921600 }, + { PCI_VENDOR_ID_V3, PCI_DEVICE_ID_V3_V351, + PCI_SUBVENDOR_ID_CONNECT_TECH, + PCI_SUBDEVICE_ID_CONNECT_TECH_BH8_485_2_6, 0, 0, + pbn_b1_8_921600 }, + { PCI_VENDOR_ID_V3, PCI_DEVICE_ID_V3_V351, + PCI_SUBVENDOR_ID_CONNECT_TECH, + PCI_SUBDEVICE_ID_CONNECT_TECH_BH081101V1, 0, 0, + pbn_b1_8_921600 }, + { PCI_VENDOR_ID_V3, PCI_DEVICE_ID_V3_V351, + PCI_SUBVENDOR_ID_CONNECT_TECH, + PCI_SUBDEVICE_ID_CONNECT_TECH_BH041101V1, 0, 0, + pbn_b1_4_921600 }, + + { PCI_VENDOR_ID_SEALEVEL, PCI_DEVICE_ID_SEALEVEL_U530, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b2_bt_1_115200 }, + { PCI_VENDOR_ID_SEALEVEL, PCI_DEVICE_ID_SEALEVEL_UCOMM2, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b2_bt_2_115200 }, + { PCI_VENDOR_ID_SEALEVEL, PCI_DEVICE_ID_SEALEVEL_UCOMM422, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b2_bt_4_115200 }, + { PCI_VENDOR_ID_SEALEVEL, PCI_DEVICE_ID_SEALEVEL_UCOMM232, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b2_bt_2_115200 }, + { PCI_VENDOR_ID_SEALEVEL, PCI_DEVICE_ID_SEALEVEL_COMM4, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b2_bt_4_115200 }, + { PCI_VENDOR_ID_SEALEVEL, PCI_DEVICE_ID_SEALEVEL_COMM8, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b2_8_115200 }, + + { PCI_VENDOR_ID_PLX, PCI_DEVICE_ID_PLX_GTEK_SERIAL2, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b2_bt_2_115200 }, + { PCI_VENDOR_ID_PLX, PCI_DEVICE_ID_PLX_SPCOM200, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b2_bt_2_921600 }, + /* VScom SPCOM800, from sl@s.pl */ + { PCI_VENDOR_ID_PLX, PCI_DEVICE_ID_PLX_SPCOM800, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b2_8_921600 }, + { PCI_VENDOR_ID_PLX, PCI_DEVICE_ID_PLX_1077, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b2_4_921600 }, + { PCI_VENDOR_ID_PLX, PCI_DEVICE_ID_PLX_9050, + PCI_SUBVENDOR_ID_KEYSPAN, + PCI_SUBDEVICE_ID_KEYSPAN_SX2, 0, 0, + pbn_panacom }, + { PCI_VENDOR_ID_PANACOM, PCI_DEVICE_ID_PANACOM_QUADMODEM, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_panacom4 }, + { PCI_VENDOR_ID_PANACOM, PCI_DEVICE_ID_PANACOM_DUALMODEM, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_panacom2 }, + { PCI_VENDOR_ID_PLX, PCI_DEVICE_ID_PLX_9050, + PCI_SUBVENDOR_ID_CHASE_PCIFAST, + PCI_SUBDEVICE_ID_CHASE_PCIFAST4, 0, 0, + pbn_b2_4_460800 }, + { PCI_VENDOR_ID_PLX, PCI_DEVICE_ID_PLX_9050, + PCI_SUBVENDOR_ID_CHASE_PCIFAST, + PCI_SUBDEVICE_ID_CHASE_PCIFAST8, 0, 0, + pbn_b2_8_460800 }, + { PCI_VENDOR_ID_PLX, PCI_DEVICE_ID_PLX_9050, + PCI_SUBVENDOR_ID_CHASE_PCIFAST, + PCI_SUBDEVICE_ID_CHASE_PCIFAST16, 0, 0, + pbn_b2_16_460800 }, + { PCI_VENDOR_ID_PLX, PCI_DEVICE_ID_PLX_9050, + PCI_SUBVENDOR_ID_CHASE_PCIFAST, + PCI_SUBDEVICE_ID_CHASE_PCIFAST16FMC, 0, 0, + pbn_b2_16_460800 }, + { PCI_VENDOR_ID_PLX, PCI_DEVICE_ID_PLX_9050, + PCI_SUBVENDOR_ID_CHASE_PCIRAS, + PCI_SUBDEVICE_ID_CHASE_PCIRAS4, 0, 0, + pbn_b2_4_460800 }, + { PCI_VENDOR_ID_PLX, PCI_DEVICE_ID_PLX_9050, + PCI_SUBVENDOR_ID_CHASE_PCIRAS, + PCI_SUBDEVICE_ID_CHASE_PCIRAS8, 0, 0, + pbn_b2_8_460800 }, + /* Megawolf Romulus PCI Serial Card, from Mike Hudson */ + /* (Exoray@isys.ca) */ + { PCI_VENDOR_ID_PLX, PCI_DEVICE_ID_PLX_ROMULUS, + 0x10b5, 0x106a, 0, 0, + pbn_plx_romulus }, + { PCI_VENDOR_ID_QUATECH, PCI_DEVICE_ID_QUATECH_QSC100, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b1_4_115200 }, + { PCI_VENDOR_ID_QUATECH, PCI_DEVICE_ID_QUATECH_DSC100, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b1_2_115200 }, + { PCI_VENDOR_ID_QUATECH, PCI_DEVICE_ID_QUATECH_ESC100D, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b1_8_115200 }, + { PCI_VENDOR_ID_QUATECH, PCI_DEVICE_ID_QUATECH_ESC100M, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b1_8_115200 }, + { PCI_VENDOR_ID_SPECIALIX, PCI_DEVICE_ID_OXSEMI_16PCI954, + PCI_VENDOR_ID_SPECIALIX, PCI_SUBDEVICE_ID_SPECIALIX_SPEED4, 0, 0, + pbn_b0_4_921600 }, + { PCI_VENDOR_ID_OXSEMI, PCI_DEVICE_ID_OXSEMI_16PCI954, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b0_4_115200 }, + { PCI_VENDOR_ID_OXSEMI, PCI_DEVICE_ID_OXSEMI_16PCI952, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b0_2_115200 }, + + /* Digitan DS560-558, from jimd@esoft.com */ + { PCI_VENDOR_ID_ATT, PCI_DEVICE_ID_ATT_VENUS_MODEM, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b1_1_115200 }, + + /* 3Com US Robotics 56k Voice Internal PCI model 5610 */ + { PCI_VENDOR_ID_USR, 0x1008, + PCI_ANY_ID, PCI_ANY_ID, }, + + /* Titan Electronic cards */ + { PCI_VENDOR_ID_TITAN, PCI_DEVICE_ID_TITAN_100, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b0_1_921600 }, + { PCI_VENDOR_ID_TITAN, PCI_DEVICE_ID_TITAN_200, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b0_2_921600 }, + { PCI_VENDOR_ID_TITAN, PCI_DEVICE_ID_TITAN_400, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b0_4_921600 }, + { PCI_VENDOR_ID_TITAN, PCI_DEVICE_ID_TITAN_800B, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b0_4_921600 }, + { PCI_VENDOR_ID_TITAN, PCI_DEVICE_ID_TITAN_100L, + PCI_ANY_ID, PCI_ANY_ID, + SPCI_FL_BASE1, 1, 921600 }, + { PCI_VENDOR_ID_TITAN, PCI_DEVICE_ID_TITAN_200L, + PCI_ANY_ID, PCI_ANY_ID, + SPCI_FL_BASE1 | SPCI_FL_BASE_TABLE, 2, 921600 }, + /* The 400L and 800L have a custom hack in get_pci_port */ + { PCI_VENDOR_ID_TITAN, PCI_DEVICE_ID_TITAN_400L, + PCI_ANY_ID, PCI_ANY_ID, + SPCI_FL_BASE_TABLE, 4, 921600 }, + { PCI_VENDOR_ID_TITAN, PCI_DEVICE_ID_TITAN_800L, + PCI_ANY_ID, PCI_ANY_ID, + SPCI_FL_BASE_TABLE, 8, 921600 }, + + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_1S_10x_550, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig10x_0 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_1S_10x_650, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig10x_0 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_1S_10x_850, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig10x_0 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_1S1P_10x_550, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig10x_1 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_1S1P_10x_650, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig10x_1 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_1S1P_10x_850, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig10x_1 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_2S_10x_550, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig10x_2 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_2S_10x_650, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig10x_2 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_2S_10x_850, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig10x_2 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_2S1P_10x_550, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig10x_2 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_2S1P_10x_650, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig10x_2 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_2S1P_10x_850, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig10x_2 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_4S_10x_550, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig10x_4 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_4S_10x_650, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig10x_4 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_4S_10x_850, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig10x_4 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_1S_20x_550, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig20x_0 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_1S_20x_650, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig20x_0 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_1S_20x_850, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig20x_0 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_1S1P_20x_550, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig20x_0 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_1S1P_20x_650, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig20x_0 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_1S1P_20x_850, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig20x_0 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_2P1S_20x_550, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig20x_0 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_2P1S_20x_650, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig20x_0 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_2P1S_20x_850, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig20x_0 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_2S_20x_550, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig20x_2 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_2S_20x_650, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig20x_2 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_2S_20x_850, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig20x_2 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_2S1P_20x_550, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig20x_2 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_2S1P_20x_650, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig20x_2 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_2S1P_20x_850, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig20x_2 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_4S_20x_550, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig20x_4 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_4S_20x_650, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig20x_4 }, + { PCI_VENDOR_ID_SIIG, PCI_DEVICE_ID_SIIG_4S_20x_850, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_siig20x_4 }, + + /* Computone devices submitted by Doug McNash dmcnash@computone.com */ + { PCI_VENDOR_ID_COMPUTONE, PCI_DEVICE_ID_COMPUTONE_PG, + PCI_SUBVENDOR_ID_COMPUTONE, PCI_SUBDEVICE_ID_COMPUTONE_PG4, + 0, 0, pbn_computone_4 }, + { PCI_VENDOR_ID_COMPUTONE, PCI_DEVICE_ID_COMPUTONE_PG, + PCI_SUBVENDOR_ID_COMPUTONE, PCI_SUBDEVICE_ID_COMPUTONE_PG8, + 0, 0, pbn_computone_8 }, + { PCI_VENDOR_ID_COMPUTONE, PCI_DEVICE_ID_COMPUTONE_PG, + PCI_SUBVENDOR_ID_COMPUTONE, PCI_SUBDEVICE_ID_COMPUTONE_PG6, + 0, 0, pbn_computone_6 }, + + { PCI_VENDOR_ID_OXSEMI, PCI_DEVICE_ID_OXSEMI_16PCI95N, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, pbn_oxsemi }, + { PCI_VENDOR_ID_TIMEDIA, PCI_DEVICE_ID_TIMEDIA_1889, + PCI_VENDOR_ID_TIMEDIA, PCI_ANY_ID, 0, 0, pbn_timedia }, + + { PCI_VENDOR_ID_LAVA, PCI_DEVICE_ID_LAVA_DSERIAL, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b0_bt_2_115200 }, + { PCI_VENDOR_ID_LAVA, PCI_DEVICE_ID_LAVA_QUATRO_A, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b0_bt_2_115200 }, + { PCI_VENDOR_ID_LAVA, PCI_DEVICE_ID_LAVA_QUATRO_B, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b0_bt_2_115200 }, + { PCI_VENDOR_ID_LAVA, PCI_DEVICE_ID_LAVA_PORT_PLUS, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b0_bt_2_460800 }, + { PCI_VENDOR_ID_LAVA, PCI_DEVICE_ID_LAVA_QUAD_A, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b0_bt_2_460800 }, + { PCI_VENDOR_ID_LAVA, PCI_DEVICE_ID_LAVA_QUAD_B, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b0_bt_2_460800 }, + { PCI_VENDOR_ID_LAVA, PCI_DEVICE_ID_LAVA_SSERIAL, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b0_bt_1_115200 }, + { PCI_VENDOR_ID_LAVA, PCI_DEVICE_ID_LAVA_PORT_650, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b0_bt_1_460800 }, + + /* RAStel 2 port modem, gerg@moreton.com.au */ + { PCI_VENDOR_ID_MORETON, PCI_DEVICE_ID_RASTEL_2PORT, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_b2_bt_2_115200 }, + + /* EKF addition for i960 Boards form EKF with serial port */ + { PCI_VENDOR_ID_INTEL, 0x1960, + 0xE4BF, PCI_ANY_ID, 0, 0, + pbn_intel_i960 }, + + /* Xircom Cardbus/Ethernet combos */ + { PCI_VENDOR_ID_XIRCOM, PCI_DEVICE_ID_XIRCOM_X3201_MDM, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_xircom_combo }, + + /* + * Untested PCI modems, sent in from various folks... + */ + + /* Elsa Model 56K PCI Modem, from Andreas Rath */ + { PCI_VENDOR_ID_ROCKWELL, 0x1004, + 0x1048, 0x1500, 0, 0, + pbn_b1_1_115200 }, + + { PCI_VENDOR_ID_SGI, PCI_DEVICE_ID_SGI_IOC3, + 0xFF00, 0, 0, 0, + pbn_sgi_ioc3 }, + +#ifdef CONFIG_DDB5074 + /* + * NEC Vrc-5074 (Nile 4) builtin UART. + * Conditionally compiled in since this is a motherboard device. + */ + { PCI_VENDOR_ID_NEC, PCI_DEVICE_ID_NEC_NILE4, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_nec_nile4 }, +#endif + +#if 0 /* PCI_DEVICE_ID_DCI_PCCOM8 ? */ + { PCI_VENDOR_ID_DCI, PCI_DEVICE_ID_DCI_PCCOM8, + PCI_ANY_ID, PCI_ANY_ID, 0, 0, + pbn_dci_pccom8 }, +#endif + + { PCI_ANY_ID, PCI_ANY_ID, PCI_ANY_ID, PCI_ANY_ID, + PCI_CLASS_COMMUNICATION_SERIAL << 8, 0xffff00, }, + { PCI_ANY_ID, PCI_ANY_ID, PCI_ANY_ID, PCI_ANY_ID, + PCI_CLASS_COMMUNICATION_MODEM << 8, 0xffff00, }, + { PCI_ANY_ID, PCI_ANY_ID, PCI_ANY_ID, PCI_ANY_ID, + PCI_CLASS_COMMUNICATION_MULTISERIAL << 8, 0xffff00, }, + { 0, } +}; + +static struct pci_driver serial_pci_driver = { + name: "serial", + probe: pci_init_one, + remove: pci_remove_one, + id_table: serial_pci_tbl, +}; + +static int __init serial8250_pci_init(void) +{ + return pci_module_init(&serial_pci_driver); +} + +static void __exit serial8250_pci_exit(void) +{ + pci_unregister_driver(&serial_pci_driver); +} + +module_init(serial8250_pci_init); +module_exit(serial8250_pci_exit); + +EXPORT_NO_SYMBOLS; + +MODULE_LICENSE("GPL"); +MODULE_DESCRIPTION("Generic 8250/16x50 PCI serial probe module"); +MODULE_GENERIC_TABLE(pci, serial_pci_tbl); diff -urN orig/drivers/serial/8250_pnp.c linux/drivers/serial/8250_pnp.c --- orig/drivers/serial/8250_pnp.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/serial/8250_pnp.c Wed Mar 19 19:06:20 2003 @@ -0,0 +1,553 @@ +/* + * linux/drivers/char/serial_8250_pnp.c + * + * Probe module for 8250/16550-type ISAPNP serial ports. + * + * Based on drivers/char/serial.c, by Linus Torvalds, Theodore Ts'o. + * + * Copyright (C) 2001 Russell King, All Rights Reserved. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License. + * + * $Id: 8250_pnp.c,v 1.3.2.1 2002/10/24 09:53:25 rmk Exp $ + */ +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include + +#include "8250.h" + +static struct serial_state rs_table[] = { }; +#define NR_PORTS 0 + +struct pnpbios_device_id +{ + char id[8]; + unsigned long driver_data; +}; + +static const struct pnpbios_device_id pnp_dev_table[] = { + /* Archtek America Corp. */ + /* Archtek SmartLink Modem 3334BT Plug & Play */ + { "AAC000F", 0 }, + /* Anchor Datacomm BV */ + /* SXPro 144 External Data Fax Modem Plug & Play */ + { "ADC0001", 0 }, + /* SXPro 288 External Data Fax Modem Plug & Play */ + { "ADC0002", 0 }, + /* Rockwell 56K ACF II Fax+Data+Voice Modem */ + { "AKY1021", SPCI_FL_NO_SHIRQ }, + /* AZT3005 PnP SOUND DEVICE */ + { "AZT4001", 0 }, + /* Best Data Products Inc. Smart One 336F PnP Modem */ + { "BDP3336", 0 }, + /* Boca Research */ + /* Boca Complete Ofc Communicator 14.4 Data-FAX */ + { "BRI0A49", 0 }, + /* Boca Research 33,600 ACF Modem */ + { "BRI1400", 0 }, + /* Boca 33.6 Kbps Internal FD34FSVD */ + { "BRI3400", 0 }, + /* Boca 33.6 Kbps Internal FD34FSVD */ + { "BRI0A49", 0 }, + /* Best Data Products Inc. Smart One 336F PnP Modem */ + { "BDP3336", 0 }, + /* Computer Peripherals Inc */ + /* EuroViVa CommCenter-33.6 SP PnP */ + { "CPI4050", 0 }, + /* Creative Labs */ + /* Creative Labs Phone Blaster 28.8 DSVD PnP Voice */ + { "CTL3001", 0 }, + /* Creative Labs Modem Blaster 28.8 DSVD PnP Voice */ + { "CTL3011", 0 }, + /* Creative */ + /* Creative Modem Blaster Flash56 DI5601-1 */ + { "DMB1032", 0 }, + /* Creative Modem Blaster V.90 DI5660 */ + { "DMB2001", 0 }, + /* FUJITSU */ + /* Fujitsu 33600 PnP-I2 R Plug & Play */ + { "FUJ0202", 0 }, + /* Fujitsu FMV-FX431 Plug & Play */ + { "FUJ0205", 0 }, + /* Fujitsu 33600 PnP-I4 R Plug & Play */ + { "FUJ0206", 0 }, + /* Fujitsu Fax Voice 33600 PNP-I5 R Plug & Play */ + { "FUJ0209", 0 }, + /* Archtek America Corp. */ + /* Archtek SmartLink Modem 3334BT Plug & Play */ + { "GVC000F", 0 }, + /* Hayes */ + /* Hayes Optima 288 V.34-V.FC + FAX + Voice Plug & Play */ + { "HAY0001", 0 }, + /* Hayes Optima 336 V.34 + FAX + Voice PnP */ + { "HAY000C", 0 }, + /* Hayes Optima 336B V.34 + FAX + Voice PnP */ + { "HAY000D", 0 }, + /* Hayes Accura 56K Ext Fax Modem PnP */ + { "HAY5670", 0 }, + /* Hayes Accura 56K Ext Fax Modem PnP */ + { "HAY5674", 0 }, + /* Hayes Accura 56K Fax Modem PnP */ + { "HAY5675", 0 }, + /* Hayes 288, V.34 + FAX */ + { "HAYF000", 0 }, + /* Hayes Optima 288 V.34 + FAX + Voice, Plug & Play */ + { "HAYF001", 0 }, + /* IBM */ + /* IBM Thinkpad 701 Internal Modem Voice */ + { "IBM0033", 0 }, + /* Intertex */ + /* Intertex 28k8 33k6 Voice EXT PnP */ + { "IXDC801", 0 }, + /* Intertex 33k6 56k Voice EXT PnP */ + { "IXDC901", 0 }, + /* Intertex 28k8 33k6 Voice SP EXT PnP */ + { "IXDD801", 0 }, + /* Intertex 33k6 56k Voice SP EXT PnP */ + { "IXDD901", 0 }, + /* Intertex 28k8 33k6 Voice SP INT PnP */ + { "IXDF401", 0 }, + /* Intertex 28k8 33k6 Voice SP EXT PnP */ + { "IXDF801", 0 }, + /* Intertex 33k6 56k Voice SP EXT PnP */ + { "IXDF901", 0 }, + /* Kortex International */ + /* KORTEX 28800 Externe PnP */ + { "KOR4522", 0 }, + /* KXPro 33.6 Vocal ASVD PnP */ + { "KORF661", 0 }, + /* Lasat */ + /* LASAT Internet 33600 PnP */ + { "LAS4040", 0 }, + /* Lasat Safire 560 PnP */ + { "LAS4540", 0 }, + /* Lasat Safire 336 PnP */ + { "LAS5440", 0 }, + /* Microcom, Inc. */ + /* Microcom TravelPorte FAST V.34 Plug & Play */ + { "MNP0281", 0 }, + /* Microcom DeskPorte V.34 FAST or FAST+ Plug & Play */ + { "MNP0336", 0 }, + /* Microcom DeskPorte FAST EP 28.8 Plug & Play */ + { "MNP0339", 0 }, + /* Microcom DeskPorte 28.8P Plug & Play */ + { "MNP0342", 0 }, + /* Microcom DeskPorte FAST ES 28.8 Plug & Play */ + { "MNP0500", 0 }, + /* Microcom DeskPorte FAST ES 28.8 Plug & Play */ + { "MNP0501", 0 }, + /* Microcom DeskPorte 28.8S Internal Plug & Play */ + { "MNP0502", 0 }, + /* Motorola */ + /* Motorola BitSURFR Plug & Play */ + { "MOT1105", 0 }, + /* Motorola TA210 Plug & Play */ + { "MOT1111", 0 }, + /* Motorola HMTA 200 (ISDN) Plug & Play */ + { "MOT1114", 0 }, + /* Motorola BitSURFR Plug & Play */ + { "MOT1115", 0 }, + /* Motorola Lifestyle 28.8 Internal */ + { "MOT1190", 0 }, + /* Motorola V.3400 Plug & Play */ + { "MOT1501", 0 }, + /* Motorola Lifestyle 28.8 V.34 Plug & Play */ + { "MOT1502", 0 }, + /* Motorola Power 28.8 V.34 Plug & Play */ + { "MOT1505", 0 }, + /* Motorola ModemSURFR External 28.8 Plug & Play */ + { "MOT1509", 0 }, + /* Motorola Premier 33.6 Desktop Plug & Play */ + { "MOT150A", 0 }, + /* Motorola VoiceSURFR 56K External PnP */ + { "MOT150F", 0 }, + /* Motorola ModemSURFR 56K External PnP */ + { "MOT1510", 0 }, + /* Motorola ModemSURFR 56K Internal PnP */ + { "MOT1550", 0 }, + /* Motorola ModemSURFR Internal 28.8 Plug & Play */ + { "MOT1560", 0 }, + /* Motorola Premier 33.6 Internal Plug & Play */ + { "MOT1580", 0 }, + /* Motorola OnlineSURFR 28.8 Internal Plug & Play */ + { "MOT15B0", 0 }, + /* Motorola VoiceSURFR 56K Internal PnP */ + { "MOT15F0", 0 }, + /* Com 1 */ + /* Deskline K56 Phone System PnP */ + { "MVX00A1", 0 }, + /* PC Rider K56 Phone System PnP */ + { "MVX00F2", 0 }, + /* Pace 56 Voice Internal Plug & Play Modem */ + { "PMC2430", 0 }, + /* Generic */ + /* Generic standard PC COM port */ + { "PNP0500", 0 }, + /* Generic 16550A-compatible COM port */ + { "PNP0501", 0 }, + /* Compaq 14400 Modem */ + { "PNPC000", 0 }, + /* Compaq 2400/9600 Modem */ + { "PNPC001", 0 }, + /* Dial-Up Networking Serial Cable between 2 PCs */ + { "PNPC031", 0 }, + /* Dial-Up Networking Parallel Cable between 2 PCs */ + { "PNPC032", 0 }, + /* Standard 9600 bps Modem */ + { "PNPC100", 0 }, + /* Standard 14400 bps Modem */ + { "PNPC101", 0 }, + /* Standard 28800 bps Modem*/ + { "PNPC102", 0 }, + /* Standard Modem*/ + { "PNPC103", 0 }, + /* Standard 9600 bps Modem*/ + { "PNPC104", 0 }, + /* Standard 14400 bps Modem*/ + { "PNPC105", 0 }, + /* Standard 28800 bps Modem*/ + { "PNPC106", 0 }, + /* Standard Modem */ + { "PNPC107", 0 }, + /* Standard 9600 bps Modem */ + { "PNPC108", 0 }, + /* Standard 14400 bps Modem */ + { "PNPC109", 0 }, + /* Standard 28800 bps Modem */ + { "PNPC10A", 0 }, + /* Standard Modem */ + { "PNPC10B", 0 }, + /* Standard 9600 bps Modem */ + { "PNPC10C", 0 }, + /* Standard 14400 bps Modem */ + { "PNPC10D", 0 }, + /* Standard 28800 bps Modem */ + { "PNPC10E", 0 }, + /* Standard Modem */ + { "PNPC10F", 0 }, + /* Standard PCMCIA Card Modem */ + { "PNP2000", 0 }, + /* Rockwell */ + /* Modular Technology */ + /* Rockwell 33.6 DPF Internal PnP */ + /* Modular Technology 33.6 Internal PnP */ + { "ROK0030", 0 }, + /* Kortex International */ + /* KORTEX 14400 Externe PnP */ + { "ROK0100", 0 }, + /* Viking Components, Inc */ + /* Viking 28.8 INTERNAL Fax+Data+Voice PnP */ + { "ROK4920", 0 }, + /* Rockwell */ + /* British Telecom */ + /* Modular Technology */ + /* Rockwell 33.6 DPF External PnP */ + /* BT Prologue 33.6 External PnP */ + /* Modular Technology 33.6 External PnP */ + { "RSS00A0", 0 }, + /* Viking 56K FAX INT */ + { "RSS0262", 0 }, + /* SupraExpress 28.8 Data/Fax PnP modem */ + { "SUP1310", 0 }, + /* SupraExpress 33.6 Data/Fax PnP modem */ + { "SUP1421", 0 }, + /* SupraExpress 33.6 Data/Fax PnP modem */ + { "SUP1590", 0 }, + /* SupraExpress 33.6 Data/Fax PnP modem */ + { "SUP1760", 0 }, + /* Phoebe Micro */ + /* Phoebe Micro 33.6 Data Fax 1433VQH Plug & Play */ + { "TEX0011", 0 }, + /* Archtek America Corp. */ + /* Archtek SmartLink Modem 3334BT Plug & Play */ + { "UAC000F", 0 }, + /* 3Com Corp. */ + /* Gateway Telepath IIvi 33.6 */ + { "USR0000", 0 }, + /* Sportster Vi 14.4 PnP FAX Voicemail */ + { "USR0004", 0 }, + /* U.S. Robotics 33.6K Voice INT PnP */ + { "USR0006", 0 }, + /* U.S. Robotics 33.6K Voice EXT PnP */ + { "USR0007", 0 }, + /* U.S. Robotics 33.6K Voice INT PnP */ + { "USR2002", 0 }, + /* U.S. Robotics 56K Voice INT PnP */ + { "USR2070", 0 }, + /* U.S. Robotics 56K Voice EXT PnP */ + { "USR2080", 0 }, + /* U.S. Robotics 56K FAX INT */ + { "USR3031", 0 }, + /* U.S. Robotics 56K Voice INT PnP */ + { "USR3070", 0 }, + /* U.S. Robotics 56K Voice EXT PnP */ + { "USR3080", 0 }, + /* U.S. Robotics 56K Voice INT PnP */ + { "USR3090", 0 }, + /* U.S. Robotics 56K Message */ + { "USR9100", 0 }, + /* U.S. Robotics 56K FAX EXT PnP*/ + { "USR9160", 0 }, + /* U.S. Robotics 56K FAX INT PnP*/ + { "USR9170", 0 }, + /* U.S. Robotics 56K Voice EXT PnP*/ + { "USR9180", 0 }, + /* U.S. Robotics 56K Voice INT PnP*/ + { "USR9190", 0 }, + { "", 0 } +}; + +static void inline avoid_irq_share(struct pci_dev *dev) +{ + int i, map = 0x1FF8; + struct serial_state *state = rs_table; + struct isapnp_irq *irq; + struct isapnp_resources *res = dev->sysdata; + + for (i = 0; i < NR_PORTS; i++) { + if (state->type != PORT_UNKNOWN) + clear_bit(state->irq, &map); + state++; + } + + for ( ; res; res = res->alt) + for(irq = res->irq; irq; irq = irq->next) + irq->map = map; +} + +static char *modem_names[] __devinitdata = { + "MODEM", "Modem", "modem", "FAX", "Fax", "fax", + "56K", "56k", "K56", "33.6", "28.8", "14.4", + "33,600", "28,800", "14,400", "33.600", "28.800", "14.400", + "33600", "28800", "14400", "V.90", "V.34", "V.32", 0 +}; + +static int __devinit check_name(char *name) +{ + char **tmp; + + for (tmp = modem_names; *tmp; tmp++) + if (strstr(name, *tmp)) + return 1; + + return 0; +} + +static int inline check_compatible_id(struct pci_dev *dev) +{ + int i; + for (i = 0; i < DEVICE_COUNT_COMPATIBLE; i++) + if ((dev->vendor_compatible[i] == + ISAPNP_VENDOR('P', 'N', 'P')) && + (swab16(dev->device_compatible[i]) >= 0xc000) && + (swab16(dev->device_compatible[i]) <= 0xdfff)) + return 0; + return 1; +} + +/* + * Given a complete unknown ISA PnP device, try to use some heuristics to + * detect modems. Currently use such heuristic set: + * - dev->name or dev->bus->name must contain "modem" substring; + * - device must have only one IO region (8 byte long) with base adress + * 0x2e8, 0x3e8, 0x2f8 or 0x3f8. + * + * Such detection looks very ugly, but can detect at least some of numerous + * ISA PnP modems, alternatively we must hardcode all modems in pnp_devices[] + * table. + */ +static int serial_pnp_guess_board(struct pci_dev *dev, int *flags) +{ + struct isapnp_resources *res = (struct isapnp_resources *)dev->sysdata; + struct isapnp_resources *resa; + + if (!(check_name(dev->name) || check_name(dev->bus->name)) && + !(check_compatible_id(dev))) + return -ENODEV; + + if (!res || res->next) + return -ENODEV; + + for (resa = res->alt; resa; resa = resa->alt) { + struct isapnp_port *port; + for (port = res->port; port; port = port->next) + if ((port->size == 8) && + ((port->min == 0x2f8) || + (port->min == 0x3f8) || + (port->min == 0x2e8) || + (port->min == 0x3e8))) + return 0; + } + + return -ENODEV; +} + +static int +pnp_init_one(struct pci_dev *dev, const struct pnpbios_device_id *ent, + char *slot_name) +{ + struct serial_struct serial_req; + int ret, line, flags = ent ? ent->driver_data : 0; + + if (!ent) { + ret = serial_pnp_guess_board(dev, &flags); + if (ret) + return ret; + } + + if (dev->prepare(dev) < 0) { + printk("serial: PNP device '%s' prepare failed\n", + slot_name); + return -ENODEV; + } + + if (dev->active) + return -ENODEV; + + if (flags & SPCI_FL_NO_SHIRQ) + avoid_irq_share(dev); + + if (dev->activate(dev) < 0) { + printk("serial: PNP device '%s' activate failed\n", + slot_name); + return -ENODEV; + } + + memset(&serial_req, 0, sizeof(serial_req)); + serial_req.irq = dev->irq_resource[0].start; + serial_req.port = pci_resource_start(dev, 0); + if (HIGH_BITS_OFFSET) + serial_req.port = pci_resource_start(dev, 0) >> HIGH_BITS_OFFSET; + +#ifdef SERIAL_DEBUG_PCI + printk("Setup PCI/PNP port: port %x, irq %d, type %d\n", + serial_req.port, serial_req.irq, serial_req.io_type); +#endif + + serial_req.flags = ASYNC_SKIP_TEST | ASYNC_AUTOPROBE; + serial_req.baud_base = 115200; + line = register_serial(&serial_req); + + if (line >= 0) { + pci_set_drvdata(dev, (void *)(line + 1)); + + /* + * Public health warning: remove this once the 2.5 + * pnpbios_module_init() stuff is incorporated. + */ + dev->driver = (void *)pnp_dev_table; + } else + dev->deactivate(dev); + + return line >= 0 ? 0 : -ENODEV; +} + +static void pnp_remove_one(struct pci_dev *dev) +{ + int line = (int)pci_get_drvdata(dev); + + if (line) { + pci_set_drvdata(dev, NULL); + + unregister_serial(line - 1); + + dev->deactivate(dev); + } +} + +static char hex[] = "0123456789ABCDEF"; + +/* + * This function should vanish when 2.5 comes around and + * we have pnpbios_module_init() + */ +static void pnp_init(void) +{ + const struct pnpbios_device_id *id; + struct pci_dev *dev = NULL; + +#ifdef SERIAL_DEBUG_PNP + printk("Entered probe_serial_pnp()\n"); +#endif + + isapnp_for_each_dev(dev) { + char slot_name[8]; + u32 pnpid; + + if (dev->active) + continue; + + pnpid = dev->vendor << 16 | dev->device; + pnpid = cpu_to_le32(pnpid); + +#define HEX(id,a) hex[((id)>>a) & 15] +#define CHAR(id,a) (0x40 + (((id)>>a) & 31)) + slot_name[0] = CHAR(pnpid, 26); + slot_name[1] = CHAR(pnpid, 21); + slot_name[2] = CHAR(pnpid, 16); + slot_name[3] = HEX(pnpid, 12); + slot_name[4] = HEX(pnpid, 8); + slot_name[5] = HEX(pnpid, 4); + slot_name[6] = HEX(pnpid, 0); + slot_name[7] = '\0'; + + for (id = pnp_dev_table; id->id[0]; id++) + if (memcmp(id->id, slot_name, 7) == 0) + break; + + if (id->id[0]) + pnp_init_one(dev, id, slot_name); + else + pnp_init_one(dev, NULL, slot_name); + } + +#ifdef SERIAL_DEBUG_PNP + printk("Leaving probe_serial_pnp() (probe finished)\n"); +#endif +} + +static int __init serial8250_pnp_init(void) +{ + if (!isapnp_present()) { +#ifdef SERIAL_DEBUG_PNP + printk("Leaving probe_serial_pnp() (no isapnp)\n"); +#endif + return -ENODEV; + } + pnp_init(); + return 0; +} + +static void __exit serial8250_pnp_exit(void) +{ + struct pci_dev *dev = NULL; + + isapnp_for_each_dev(dev) { + if (dev->driver != (void *)pnp_dev_table) + continue; + pnp_remove_one(dev); + } +} + +module_init(serial8250_pnp_init); +module_exit(serial8250_pnp_exit); + +EXPORT_NO_SYMBOLS; + +MODULE_LICENSE("GPL"); +MODULE_DESCRIPTION("Generic 8250/16x50 PNPBIOS serial probe module"); +MODULE_GENERIC_TABLE(pnp, pnp_dev_table); + diff -urN orig/drivers/serial/Config.in linux/drivers/serial/Config.in --- orig/drivers/serial/Config.in Thu Jan 1 01:00:00 1970 +++ linux/drivers/serial/Config.in Thu Feb 27 23:21:11 2003 @@ -0,0 +1,90 @@ +# +# Serial device configuration +# +# $Id: Config.in,v 1.4 2001/10/12 15:46:58 rmk Exp $ +# +mainmenu_option next_comment +comment 'Serial drivers' + +if [ "$CONFIG_ARM" = "y" ]; then + # I don't have this in my tree yet. + dep_bool 'Anakin serial port support' CONFIG_SERIAL_ANAKIN $CONFIG_ARCH_ANAKIN + dep_bool ' Console on Anakin serial port' CONFIG_SERIAL_ANAKIN_CONSOLE $CONFIG_SERIAL_ANAKIN + if [ "$CONFIG_SERIAL_ANAKIN" = "y" ]; then + int ' Default Anakin serial baudrate' CONFIG_ANAKIN_DEFAULT_BAUDRATE 9600 + fi + + dep_tristate 'ARM AMBA serial port support' CONFIG_SERIAL_AMBA $CONFIG_ARCH_INTEGRATOR + dep_bool ' Support for console on AMBA serial port' CONFIG_SERIAL_AMBA_CONSOLE $CONFIG_SERIAL_AMBA + if [ "$CONFIG_SERIAL_AMBA" = "y" ]; then + define_bool CONFIG_SERIAL_INTEGRATOR y + fi + + dep_tristate 'CLPS711X serial port support' CONFIG_SERIAL_CLPS711X $CONFIG_ARCH_CLPS711X + dep_bool ' Support for console on CLPS711X serial port' CONFIG_SERIAL_CLPS711X_CONSOLE $CONFIG_SERIAL_CLPS711X + + dep_bool 'DC21285 serial port support' CONFIG_SERIAL_21285 $CONFIG_FOOTBRIDGE + dep_bool ' Use /dev/ttyS0 device (OBSOLETE)' CONFIG_SERIAL_21285_OLD $CONFIG_SERIAL_21285 $CONFIG_OBSOLETE + dep_bool ' Console on DC21285 serial port' CONFIG_SERIAL_21285_CONSOLE $CONFIG_SERIAL_21285 + + dep_bool 'Excalibur serial port (uart00) support' CONFIG_SERIAL_UART00 $CONFIG_ARCH_CAMELOT + dep_bool ' Support for console on Excalibur serial port' CONFIG_SERIAL_UART00_CONSOLE $CONFIG_SERIAL_UART00 + + + dep_bool 'SA1100 serial port support' CONFIG_SERIAL_SA1100 $CONFIG_ARCH_SA1100 + dep_bool ' Console on SA1100 serial port' CONFIG_SERIAL_SA1100_CONSOLE $CONFIG_SERIAL_SA1100 + if [ "$CONFIG_SERIAL_SA1100" = "y" ]; then + int ' Default SA1100 serial baudrate' CONFIG_SA1100_DEFAULT_BAUDRATE 9600 + fi + + dep_tristate 'ARM Omaha serial port support' CONFIG_SERIAL_OMAHA $CONFIG_ARCH_OMAHA + dep_bool ' Support for console on Omaha serial port' CONFIG_SERIAL_OMAHA_CONSOLE $CONFIG_SERIAL_OMAHA + + dep_bool 'AT91RM9200 serial port support' CONFIG_SERIAL_AT91US3 $CONFIG_ARCH_AT91RM9200DK + dep_bool ' Support for console on AT91RM9200 serial port' CONFIG_SERIAL_AT91US3_CONSOLE $CONFIG_SERIAL_AT91US3 + +fi +# +# The new 8250/16550 serial drivers +dep_tristate '8250/16550 and compatible serial support (EXPERIMENTAL)' CONFIG_SERIAL_8250 $CONFIG_EXPERIMENTAL +dep_bool ' Console on 8250/16550 and compatible serial port (EXPERIMENTAL)' CONFIG_SERIAL_8250_CONSOLE $CONFIG_SERIAL_8250 $CONFIG_EXPERIMENTAL + +dep_mbool 'Extended 8250/16550 serial driver options' CONFIG_SERIAL_8250_EXTENDED $CONFIG_SERIAL_8250 +dep_bool ' Support more than 4 serial ports' CONFIG_SERIAL_8250_MANY_PORTS $CONFIG_SERIAL_8250_EXTENDED +dep_bool ' Support for sharing serial interrupts' CONFIG_SERIAL_8250_SHARE_IRQ $CONFIG_SERIAL_8250_EXTENDED +dep_bool ' Autodetect IRQ on standard ports (unsafe)' CONFIG_SERIAL_8250_DETECT_IRQ $CONFIG_SERIAL_8250_EXTENDED +dep_bool ' Support special multiport boards' CONFIG_SERIAL_8250_MULTIPORT $CONFIG_SERIAL_8250_EXTENDED +dep_bool ' Support Bell Technologies HUB6 card' CONFIG_SERIAL_8250_HUB6 $CONFIG_SERIAL_8250_EXTENDED + +if [ "$CONFIG_SERIAL_AMBA" = "y" -o \ + "$CONFIG_SERIAL_CLPS711X" = "y" -o \ + "$CONFIG_SERIAL_SA1100" = "y" -o \ + "$CONFIG_SERIAL_ANAKIN" = "y" -o \ + "$CONFIG_SERIAL_UART00" = "y" -o \ + "$CONFIG_SERIAL_8250" = "y" -o \ + "$CONFIG_SERIAL_OMAHA" = "y" -o \ + "$CONFIG_SERIAL_AT91US3" = "y" ]; then + define_bool CONFIG_SERIAL_CORE y +else + if [ "$CONFIG_SERIAL_AMBA" = "m" -o \ + "$CONFIG_SERIAL_CLPS711X" = "m" -o \ + "$CONFIG_SERIAL_SA1100" = "m" -o \ + "$CONFIG_SERIAL_ANAKIN" = "m" -o \ + "$CONFIG_SERIAL_UART00" = "m" -o \ + "$CONFIG_SERIAL_8250" = "m" -o \ + "$CONFIG_SERIAL_OMAHA" = "m" ]; then + define_bool CONFIG_SERIAL_CORE m + fi +fi +if [ "$CONFIG_SERIAL_AMBA_CONSOLE" = "y" -o \ + "$CONFIG_SERIAL_CLPS711X_CONSOLE" = "y" -o \ + "$CONFIG_SERIAL_SA1100_CONSOLE" = "y" -o \ + "$CONFIG_SERIAL_ANAKIN_CONSOLE" = "y" -o \ + "$CONFIG_SERIAL_UART00_CONSOLE" = "y" -o \ + "$CONFIG_SERIAL_8250_CONSOLE" = "y" -o \ + "$CONFIG_SERIAL_OMAHA" = "y" -o \ + "$CONFIG_SERIAL_AT91US3_CONSOLE" = "y" ]; then + define_bool CONFIG_SERIAL_CORE_CONSOLE y +fi + +endmenu diff -urN orig/drivers/serial/Makefile linux/drivers/serial/Makefile --- orig/drivers/serial/Makefile Thu Jan 1 01:00:00 1970 +++ linux/drivers/serial/Makefile Thu Feb 27 23:21:11 2003 @@ -0,0 +1,39 @@ +# +# Makefile for the kernel serial device drivers. +# +# Note! Dependencies are done automagically by 'make dep', which also +# removes any old dependencies. DON'T put your own dependencies here +# unless it's something special (ie not a .c file). +# +# Note 2! The CFLAGS definitions are now inherited from the +# parent makes.. +# +# $Id: Makefile,v 1.2 2001/10/12 15:46:58 rmk Exp $ +# + +O_TARGET := serial.o + +export-objs := core.o 8250.o +obj-y := +obj-m := +obj-n := +obj- := + +serial-8250-y := +serial-8250-$(CONFIG_PCI) += 8250_pci.o +serial-8250-$(CONFIG_ISAPNP) += 8250_pnp.o +obj-$(CONFIG_SERIAL_CORE) += core.o +obj-$(CONFIG_SERIAL_21285) += 21285.o +obj-$(CONFIG_SERIAL_8250) += 8250.o $(serial-8250-y) +obj-$(CONFIG_SERIAL_ANAKIN) += anakin.o +obj-$(CONFIG_SERIAL_AMBA) += amba.o +obj-$(CONFIG_SERIAL_CLPS711X) += clps711x.o +obj-$(CONFIG_SERIAL_SA1100) += sa1100.o +obj-$(CONFIG_SERIAL_UART00) += uart00.o +obj-$(CONFIG_SERIAL_OMAHA) += omaha.o +obj-$(CONFIG_SERIAL_AT91US3) += at91us3.o + +include $(TOPDIR)/Rules.make + +fastdep: + diff -urN orig/drivers/serial/amba.c linux/drivers/serial/amba.c --- orig/drivers/serial/amba.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/serial/amba.c Thu Oct 24 10:53:25 2002 @@ -0,0 +1,757 @@ +/* + * linux/drivers/char/serial_amba.c + * + * Driver for AMBA serial ports + * + * Based on drivers/char/serial.c, by Linus Torvalds, Theodore Ts'o. + * + * Copyright 1999 ARM Limited + * Copyright (C) 2000 Deep Blue Solutions Ltd. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + * + * $Id: amba.c,v 1.9.2.2 2002/10/24 09:53:25 rmk Exp $ + * + * This is a generic driver for ARM AMBA-type serial ports. They + * have a lot of 16550-like features, but are not register compatable. + * Note that although they do have CTS, DCD and DSR inputs, they do + * not have an RI input, nor do they have DTR or RTS outputs. If + * required, these have to be supplied via some other means (eg, GPIO) + * and hooked into this driver. + */ +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include + +#if defined(CONFIG_SERIAL_AMBA_CONSOLE) && defined(CONFIG_MAGIC_SYSRQ) +#define SUPPORT_SYSRQ +#endif + +#include + +#include + +#define UART_NR 2 + +#define SERIAL_AMBA_MAJOR 204 +#define SERIAL_AMBA_MINOR 16 +#define SERIAL_AMBA_NR UART_NR + +#define CALLOUT_AMBA_NAME "cuaam" +#define CALLOUT_AMBA_MAJOR 205 +#define CALLOUT_AMBA_MINOR 16 +#define CALLOUT_AMBA_NR UART_NR + +static struct tty_driver normal, callout; +static struct tty_struct *amba_table[UART_NR]; +static struct termios *amba_termios[UART_NR], *amba_termios_locked[UART_NR]; +#ifdef SUPPORT_SYSRQ +static struct console amba_console; +#endif + +#define AMBA_ISR_PASS_LIMIT 256 + +/* + * Access macros for the AMBA UARTs + */ +#define UART_GET_INT_STATUS(p) readb((p)->membase + AMBA_UARTIIR) +#define UART_PUT_ICR(p, c) writel((c), (p)->membase + AMBA_UARTICR) +#define UART_GET_FR(p) readb((p)->membase + AMBA_UARTFR) +#define UART_GET_CHAR(p) readb((p)->membase + AMBA_UARTDR) +#define UART_PUT_CHAR(p, c) writel((c), (p)->membase + AMBA_UARTDR) +#define UART_GET_RSR(p) readb((p)->membase + AMBA_UARTRSR) +#define UART_GET_CR(p) readb((p)->membase + AMBA_UARTCR) +#define UART_PUT_CR(p,c) writel((c), (p)->membase + AMBA_UARTCR) +#define UART_GET_LCRL(p) readb((p)->membase + AMBA_UARTLCR_L) +#define UART_PUT_LCRL(p,c) writel((c), (p)->membase + AMBA_UARTLCR_L) +#define UART_GET_LCRM(p) readb((p)->membase + AMBA_UARTLCR_M) +#define UART_PUT_LCRM(p,c) writel((c), (p)->membase + AMBA_UARTLCR_M) +#define UART_GET_LCRH(p) readb((p)->membase + AMBA_UARTLCR_H) +#define UART_PUT_LCRH(p,c) writel((c), (p)->membase + AMBA_UARTLCR_H) +#define UART_RX_DATA(s) (((s) & AMBA_UARTFR_RXFE) == 0) +#define UART_TX_READY(s) (((s) & AMBA_UARTFR_TXFF) == 0) +#define UART_TX_EMPTY(p) ((UART_GET_FR(p) & AMBA_UARTFR_TMSK) == 0) + +#define UART_DUMMY_RSR_RX 256 +#define UART_PORT_SIZE 64 + +/* + * On the Integrator platform, the port RTS and DTR are provided by + * bits in the following SC_CTRLS register bits: + * RTS DTR + * UART0 7 6 + * UART1 5 4 + * + * We encode this bit information into port->driver_priv using the + * following macros. + */ +//#define PORT_CTRLS(dtrbit,rtsbit) ((1 << dtrbit) | (1 << (16 + rtsbit))) +#define PORT_CTRLS_DTR(port) (1 << (port)->unused[1]) +#define PORT_CTRLS_RTS(port) (1 << (port)->unused[0]) + +#define SC_CTRLC (IO_ADDRESS(INTEGRATOR_SC_BASE) + INTEGRATOR_SC_CTRLC_OFFSET) +#define SC_CTRLS (IO_ADDRESS(INTEGRATOR_SC_BASE) + INTEGRATOR_SC_CTRLS_OFFSET) + +/* + * Our private driver data mappings. + */ +#define drv_old_status driver_priv + +static void ambauart_stop_tx(struct uart_port *port, u_int from_tty) +{ + unsigned int cr; + + cr = UART_GET_CR(port); + cr &= ~AMBA_UARTCR_TIE; + UART_PUT_CR(port, cr); +} + +static void ambauart_start_tx(struct uart_port *port, u_int nonempty, u_int from_tty) +{ + if (nonempty) { + unsigned int cr; + + cr = UART_GET_CR(port); + cr |= AMBA_UARTCR_TIE; + UART_PUT_CR(port, cr); + } +} + +static void ambauart_stop_rx(struct uart_port *port) +{ + unsigned int cr; + + cr = UART_GET_CR(port); + cr &= ~(AMBA_UARTCR_RIE | AMBA_UARTCR_RTIE); + UART_PUT_CR(port, cr); +} + +static void ambauart_enable_ms(struct uart_port *port) +{ + unsigned int cr; + + cr = UART_GET_CR(port); + cr |= AMBA_UARTCR_MSIE; + UART_PUT_CR(port, cr); +} + +static void +#ifdef SUPPORT_SYSRQ +ambauart_rx_chars(struct uart_info *info, struct pt_regs *regs) +#else +ambauart_rx_chars(struct uart_info *info) +#endif +{ + struct tty_struct *tty = info->tty; + unsigned int status, ch, rsr, max_count = 256; + struct uart_port *port = info->port; + + status = UART_GET_FR(port); + while (UART_RX_DATA(status) && max_count--) { + if (tty->flip.count >= TTY_FLIPBUF_SIZE) { + tty->flip.tqueue.routine((void *)tty); + if (tty->flip.count >= TTY_FLIPBUF_SIZE) { + printk(KERN_WARNING "TTY_DONT_FLIP set\n"); + return; + } + } + + ch = UART_GET_CHAR(port); + + *tty->flip.char_buf_ptr = ch; + *tty->flip.flag_buf_ptr = TTY_NORMAL; + port->icount.rx++; + + /* + * Note that the error handling code is + * out of the main execution path + */ + rsr = UART_GET_RSR(port) | UART_DUMMY_RSR_RX; + if (rsr & AMBA_UARTRSR_ANY) { + if (rsr & AMBA_UARTRSR_BE) { + rsr &= ~(AMBA_UARTRSR_FE | AMBA_UARTRSR_PE); + port->icount.brk++; + if (uart_handle_break(info, &amba_console)) + goto ignore_char; + } else if (rsr & AMBA_UARTRSR_PE) + port->icount.parity++; + else if (rsr & AMBA_UARTRSR_FE) + port->icount.frame++; + if (rsr & AMBA_UARTRSR_OE) + port->icount.overrun++; + + rsr &= port->read_status_mask; + + if (rsr & AMBA_UARTRSR_BE) + *tty->flip.flag_buf_ptr = TTY_BREAK; + else if (rsr & AMBA_UARTRSR_PE) + *tty->flip.flag_buf_ptr = TTY_PARITY; + else if (rsr & AMBA_UARTRSR_FE) + *tty->flip.flag_buf_ptr = TTY_FRAME; + } + + if (uart_handle_sysrq_char(info, ch, regs)) + goto ignore_char; + + if ((rsr & port->ignore_status_mask) == 0) { + tty->flip.flag_buf_ptr++; + tty->flip.char_buf_ptr++; + tty->flip.count++; + } + if ((rsr & AMBA_UARTRSR_OE) && + tty->flip.count < TTY_FLIPBUF_SIZE) { + /* + * Overrun is special, since it's reported + * immediately, and doesn't affect the current + * character + */ + *tty->flip.char_buf_ptr++ = 0; + *tty->flip.flag_buf_ptr++ = TTY_OVERRUN; + tty->flip.count++; + } + ignore_char: + status = UART_GET_FR(port); + } + tty_flip_buffer_push(tty); + return; +} + +static void ambauart_tx_chars(struct uart_info *info) +{ + struct uart_port *port = info->port; + int count; + + if (port->x_char) { + UART_PUT_CHAR(port, port->x_char); + port->icount.tx++; + port->x_char = 0; + return; + } + if (info->xmit.head == info->xmit.tail + || info->tty->stopped + || info->tty->hw_stopped) { + ambauart_stop_tx(port, 0); + return; + } + + count = port->fifosize >> 1; + do { + UART_PUT_CHAR(port, info->xmit.buf[info->xmit.tail]); + info->xmit.tail = (info->xmit.tail + 1) & (UART_XMIT_SIZE - 1); + port->icount.tx++; + if (info->xmit.head == info->xmit.tail) + break; + } while (--count > 0); + + if (CIRC_CNT(info->xmit.head, info->xmit.tail, UART_XMIT_SIZE) < + WAKEUP_CHARS) + uart_event(info, EVT_WRITE_WAKEUP); + + if (info->xmit.head == info->xmit.tail) + ambauart_stop_tx(info->port, 0); +} + +static void ambauart_modem_status(struct uart_info *info) +{ + struct uart_port *port = info->port; + unsigned int status, delta; + + UART_PUT_ICR(port, 0); + + status = UART_GET_FR(port) & AMBA_UARTFR_MODEM_ANY; + + delta = status ^ info->drv_old_status; + info->drv_old_status = status; + + if (!delta) + return; + + if (delta & AMBA_UARTFR_DCD) + uart_handle_dcd_change(info, status & AMBA_UARTFR_DCD); + + if (delta & AMBA_UARTFR_DSR) + port->icount.dsr++; + + if (delta & AMBA_UARTFR_CTS) + uart_handle_cts_change(info, status & AMBA_UARTFR_CTS); + + wake_up_interruptible(&info->delta_msr_wait); +} + +static void ambauart_int(int irq, void *dev_id, struct pt_regs *regs) +{ + struct uart_info *info = dev_id; + unsigned int status, pass_counter = AMBA_ISR_PASS_LIMIT; + + status = UART_GET_INT_STATUS(info->port); + do { + if (status & (AMBA_UARTIIR_RTIS | AMBA_UARTIIR_RIS)) +#ifdef SUPPORT_SYSRQ + ambauart_rx_chars(info, regs); +#else + ambauart_rx_chars(info); +#endif + if (status & AMBA_UARTIIR_TIS) + ambauart_tx_chars(info); + if (status & AMBA_UARTIIR_MIS) + ambauart_modem_status(info); + + if (pass_counter-- == 0) + break; + + status = UART_GET_INT_STATUS(info->port); + } while (status & (AMBA_UARTIIR_RTIS | AMBA_UARTIIR_RIS | + AMBA_UARTIIR_TIS)); +} + +static u_int ambauart_tx_empty(struct uart_port *port) +{ + return UART_GET_FR(port) & AMBA_UARTFR_BUSY ? 0 : TIOCSER_TEMT; +} + +static u_int ambauart_get_mctrl(struct uart_port *port) +{ + unsigned int result = 0; + unsigned int status; + + status = UART_GET_FR(port); + if (status & AMBA_UARTFR_DCD) + result |= TIOCM_CAR; + if (status & AMBA_UARTFR_DSR) + result |= TIOCM_DSR; + if (status & AMBA_UARTFR_CTS) + result |= TIOCM_CTS; + + return result; +} + +static void ambauart_set_mctrl(struct uart_port *port, u_int mctrl) +{ + u_int ctrls = 0, ctrlc = 0; + + if (mctrl & TIOCM_RTS) + ctrlc |= PORT_CTRLS_RTS(port); + else + ctrls |= PORT_CTRLS_RTS(port); + + if (mctrl & TIOCM_DTR) + ctrlc |= PORT_CTRLS_DTR(port); + else + ctrls |= PORT_CTRLS_DTR(port); + + __raw_writel(ctrls, SC_CTRLS); + __raw_writel(ctrlc, SC_CTRLC); +} + +static void ambauart_break_ctl(struct uart_port *port, int break_state) +{ + unsigned int lcr_h; + + lcr_h = UART_GET_LCRH(port); + if (break_state == -1) + lcr_h |= AMBA_UARTLCR_H_BRK; + else + lcr_h &= ~AMBA_UARTLCR_H_BRK; + UART_PUT_LCRH(port, lcr_h); +} + +static int ambauart_startup(struct uart_port *port, struct uart_info *info) +{ + int retval; + + /* + * Allocate the IRQ + */ + retval = request_irq(port->irq, ambauart_int, 0, "amba", info); + if (retval) + return retval; + + /* + * initialise the old status of the modem signals + */ + info->drv_old_status = UART_GET_FR(port) & AMBA_UARTFR_MODEM_ANY; + + /* + * Finally, enable interrupts + */ + UART_PUT_CR(port, AMBA_UARTCR_UARTEN | AMBA_UARTCR_RIE | + AMBA_UARTCR_RTIE); + + return 0; +} + +static void ambauart_shutdown(struct uart_port *port, struct uart_info *info) +{ + /* + * Free the interrupt + */ + free_irq(port->irq, info); + + /* + * disable all interrupts, disable the port + */ + UART_PUT_CR(port, 0); + + /* disable break condition and fifos */ + UART_PUT_LCRH(port, UART_GET_LCRH(port) & + ~(AMBA_UARTLCR_H_BRK | AMBA_UARTLCR_H_FEN)); +} + +static void ambauart_change_speed(struct uart_port *port, u_int cflag, u_int iflag, u_int quot) +{ + u_int lcr_h, old_cr; + unsigned long flags; + +#if DEBUG + printk("ambauart_set_cflag(0x%x) called\n", cflag); +#endif + /* byte size and parity */ + switch (cflag & CSIZE) { + case CS5: lcr_h = AMBA_UARTLCR_H_WLEN_5; break; + case CS6: lcr_h = AMBA_UARTLCR_H_WLEN_6; break; + case CS7: lcr_h = AMBA_UARTLCR_H_WLEN_7; break; + default: lcr_h = AMBA_UARTLCR_H_WLEN_8; break; // CS8 + } + if (cflag & CSTOPB) + lcr_h |= AMBA_UARTLCR_H_STP2; + if (cflag & PARENB) { + lcr_h |= AMBA_UARTLCR_H_PEN; + if (!(cflag & PARODD)) + lcr_h |= AMBA_UARTLCR_H_EPS; + } + if (port->fifosize > 1) + lcr_h |= AMBA_UARTLCR_H_FEN; + + port->read_status_mask = AMBA_UARTRSR_OE; + if (iflag & INPCK) + port->read_status_mask |= AMBA_UARTRSR_FE | AMBA_UARTRSR_PE; + if (iflag & (BRKINT | PARMRK)) + port->read_status_mask |= AMBA_UARTRSR_BE; + + /* + * Characters to ignore + */ + port->ignore_status_mask = 0; + if (iflag & IGNPAR) + port->ignore_status_mask |= AMBA_UARTRSR_FE | AMBA_UARTRSR_PE; + if (iflag & IGNBRK) { + port->ignore_status_mask |= AMBA_UARTRSR_BE; + /* + * If we're ignoring parity and break indicators, + * ignore overruns too (for real raw support). + */ + if (iflag & IGNPAR) + port->ignore_status_mask |= AMBA_UARTRSR_OE; + } + + /* + * Ignore all characters if CREAD is not set. + */ + if ((cflag & CREAD) == 0) + port->ignore_status_mask |= UART_DUMMY_RSR_RX; + + /* first, disable everything */ + save_flags(flags); cli(); + old_cr = UART_GET_CR(port) & ~AMBA_UARTCR_MSIE; + + if ((port->flags & ASYNC_HARDPPS_CD) || + (cflag & CRTSCTS) || !(cflag & CLOCAL)) + old_cr |= AMBA_UARTCR_MSIE; + + UART_PUT_CR(port, 0); + + /* Set baud rate */ + quot -= 1; + UART_PUT_LCRM(port, ((quot & 0xf00) >> 8)); + UART_PUT_LCRL(port, (quot & 0xff)); + + /* + * ----------v----------v----------v----------v----- + * NOTE: MUST BE WRITTEN AFTER UARTLCR_M & UARTLCR_L + * ----------^----------^----------^----------^----- + */ + UART_PUT_LCRH(port, lcr_h); + UART_PUT_CR(port, old_cr); + + restore_flags(flags); +} + +static const char *ambauart_type(struct uart_port *port) +{ + return port->type == PORT_AMBA ? "AMBA" : NULL; +} + +/* + * Release the memory region(s) being used by 'port' + */ +static void ambauart_release_port(struct uart_port *port) +{ + release_mem_region(port->mapbase, UART_PORT_SIZE); +} + +/* + * Request the memory region(s) being used by 'port' + */ +static int ambauart_request_port(struct uart_port *port) +{ + return request_mem_region(port->mapbase, UART_PORT_SIZE, "serial_amba") + != NULL ? 0 : -EBUSY; +} + +/* + * Configure/autoconfigure the port. + */ +static void ambauart_config_port(struct uart_port *port, int flags) +{ + if (flags & UART_CONFIG_TYPE) { + port->type = PORT_AMBA; + ambauart_request_port(port); + } +} + +/* + * verify the new serial_struct (for TIOCSSERIAL). + */ +static int ambauart_verify_port(struct uart_port *port, struct serial_struct *ser) +{ + int ret = 0; + if (ser->type != PORT_UNKNOWN && ser->type != PORT_AMBA) + ret = -EINVAL; + if (ser->irq < 0 || ser->irq >= NR_IRQS) + ret = -EINVAL; + if (ser->baud_base < 9600) + ret = -EINVAL; + return ret; +} + +static struct uart_ops amba_pops = { + tx_empty: ambauart_tx_empty, + set_mctrl: ambauart_set_mctrl, + get_mctrl: ambauart_get_mctrl, + stop_tx: ambauart_stop_tx, + start_tx: ambauart_start_tx, + stop_rx: ambauart_stop_rx, + enable_ms: ambauart_enable_ms, + break_ctl: ambauart_break_ctl, + startup: ambauart_startup, + shutdown: ambauart_shutdown, + change_speed: ambauart_change_speed, + type: ambauart_type, + release_port: ambauart_release_port, + request_port: ambauart_request_port, + config_port: ambauart_config_port, + verify_port: ambauart_verify_port, +}; + +static struct uart_port amba_ports[UART_NR] = { + { + membase: (void *)IO_ADDRESS(INTEGRATOR_UART0_BASE), + mapbase: INTEGRATOR_UART0_BASE, + iotype: SERIAL_IO_MEM, + irq: IRQ_UARTINT0, + uartclk: 14745600, + fifosize: 16, + unused: { 4, 5 }, /*driver_priv: PORT_CTRLS(5, 4), */ + ops: &amba_pops, + flags: ASYNC_BOOT_AUTOCONF, + }, + { + membase: (void *)IO_ADDRESS(INTEGRATOR_UART1_BASE), + mapbase: INTEGRATOR_UART1_BASE, + iotype: SERIAL_IO_MEM, + irq: IRQ_UARTINT1, + uartclk: 14745600, + fifosize: 16, + unused: { 6, 7 }, /*driver_priv: PORT_CTRLS(7, 6), */ + ops: &amba_pops, + flags: ASYNC_BOOT_AUTOCONF, + } +}; + +#ifdef CONFIG_SERIAL_AMBA_CONSOLE + +static void ambauart_console_write(struct console *co, const char *s, u_int count) +{ + struct uart_port *port = amba_ports + co->index; + unsigned int status, old_cr; + int i; + + /* + * First save the CR then disable the interrupts + */ + old_cr = UART_GET_CR(port); + UART_PUT_CR(port, AMBA_UARTCR_UARTEN); + + /* + * Now, do each character + */ + for (i = 0; i < count; i++) { + do { + status = UART_GET_FR(port); + } while (!UART_TX_READY(status)); + UART_PUT_CHAR(port, s[i]); + if (s[i] == '\n') { + do { + status = UART_GET_FR(port); + } while (!UART_TX_READY(status)); + UART_PUT_CHAR(port, '\r'); + } + } + + /* + * Finally, wait for transmitter to become empty + * and restore the TCR + */ + do { + status = UART_GET_FR(port); + } while (status & AMBA_UARTFR_BUSY); + UART_PUT_CR(port, old_cr); +} + +static kdev_t ambauart_console_device(struct console *co) +{ + return MKDEV(SERIAL_AMBA_MAJOR, SERIAL_AMBA_MINOR + co->index); +} + +static void __init +ambauart_console_get_options(struct uart_port *port, int *baud, int *parity, int *bits) +{ + if (UART_GET_CR(port) & AMBA_UARTCR_UARTEN) { + u_int lcr_h, quot; + lcr_h = UART_GET_LCRH(port); + + *parity = 'n'; + if (lcr_h & AMBA_UARTLCR_H_PEN) { + if (lcr_h & AMBA_UARTLCR_H_EPS) + *parity = 'e'; + else + *parity = 'o'; + } + + if ((lcr_h & 0x60) == AMBA_UARTLCR_H_WLEN_7) + *bits = 7; + else + *bits = 8; + + quot = UART_GET_LCRL(port) | UART_GET_LCRM(port) << 8; + *baud = port->uartclk / (16 * (quot + 1)); + } +} + +static int __init ambauart_console_setup(struct console *co, char *options) +{ + struct uart_port *port; + int baud = 38400; + int bits = 8; + int parity = 'n'; + int flow = 'n'; + + /* + * Check whether an invalid uart number has been specified, and + * if so, search for the first available port that does have + * console support. + */ + port = uart_get_console(amba_ports, UART_NR, co); + + if (options) + uart_parse_options(options, &baud, &parity, &bits, &flow); + else + ambauart_console_get_options(port, &baud, &parity, &bits); + + return uart_set_options(port, co, baud, parity, bits, flow); +} + +static struct console amba_console = { + name: "ttyAM", + write: ambauart_console_write, + device: ambauart_console_device, + setup: ambauart_console_setup, + flags: CON_PRINTBUFFER, + index: -1, +}; + +void __init ambauart_console_init(void) +{ + register_console(&amba_console); +} + +#define AMBA_CONSOLE &amba_console +#else +#define AMBA_CONSOLE NULL +#endif + +static struct uart_driver amba_reg = { + owner: THIS_MODULE, + normal_major: SERIAL_AMBA_MAJOR, +#ifdef CONFIG_DEVFS_FS + normal_name: "ttyAM%d", + callout_name: "cuaam%d", +#else + normal_name: "ttyAM", + callout_name: "cuaam", +#endif + normal_driver: &normal, + callout_major: CALLOUT_AMBA_MAJOR, + callout_driver: &callout, + table: amba_table, + termios: amba_termios, + termios_locked: amba_termios_locked, + minor: SERIAL_AMBA_MINOR, + nr: UART_NR, + port: amba_ports, + cons: AMBA_CONSOLE, +}; + +static int __init ambauart_init(void) +{ + return uart_register_driver(&amba_reg); +} + +static void __exit ambauart_exit(void) +{ + uart_unregister_driver(&amba_reg); +} + +module_init(ambauart_init); +module_exit(ambauart_exit); + +EXPORT_NO_SYMBOLS; + +MODULE_AUTHOR("ARM Ltd/Deep Blue Solutions Ltd"); +MODULE_DESCRIPTION("ARM AMBA serial port driver"); +MODULE_LICENSE("GPL"); diff -urN orig/drivers/serial/anakin.c linux/drivers/serial/anakin.c --- orig/drivers/serial/anakin.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/serial/anakin.c Thu Oct 24 10:53:25 2002 @@ -0,0 +1,545 @@ +/* + * linux/drivers/char/serial_anakin.c + * + * Based on driver for AMBA serial ports, by ARM Limited, + * Deep Blue Solutions Ltd., Linus Torvalds and Theodore Ts'o. + * + * Copyright (C) 2001 Aleph One Ltd. for Acunia N.V. + * + * Copyright (C) 2001 Blue Mug, Inc. for Acunia N.V. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + * + * Changelog: + * 20-Apr-2001 TTC Created + * 05-May-2001 W/TTC Updated for serial_core.c + * 27-Jun-2001 jonm Minor changes; add mctrl support, switch to + * SA_INTERRUPT. Works reliably now. No longer requires + * changes to the serial_core API. + * + * $Id: anakin.c,v 1.5.2.2 2002/10/24 09:53:25 rmk Exp $ + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include + +#include + +#include + +#define UART_NR 5 + +#define SERIAL_ANAKIN_NAME "ttyAN" +#define SERIAL_ANAKIN_MAJOR 204 +#define SERIAL_ANAKIN_MINOR 32 + +#define CALLOUT_ANAKIN_NAME "cuaan" +#define CALLOUT_ANAKIN_MAJOR 205 +#define CALLOUT_ANAKIN_MINOR 32 + +static struct tty_driver normal, callout; +static struct tty_struct *anakin_table[UART_NR]; +static struct termios *anakin_termios[UART_NR], *anakin_termios_locked[UART_NR]; +static struct uart_state anakin_state[UART_NR]; +static u_int txenable[NR_IRQS]; /* Software interrupt register */ + +static inline unsigned int +anakin_in(struct uart_port *port, u_int offset) +{ + return __raw_readl(port->base + offset); +} + +static inline void +anakin_out(struct uart_port *port, u_int offset, unsigned int value) +{ + __raw_writel(value, port->base + offset); +} + +static void +anakin_stop_tx(struct uart_port *port, u_int from_tty) +{ + txenable[port->irq] = 0; +} + +static inline void +anakin_transmit_buffer(struct uart_info *info) +{ + struct uart_port *port = info->port; + + while (!(anakin_in(port, 0x10) & TXEMPTY)); + anakin_out(port, 0x14, info->xmit.buf[info->xmit.tail]); + anakin_out(port, 0x18, anakin_in(port, 0x18) | SENDREQUEST); + info->xmit.tail = (info->xmit.tail + 1) & (UART_XMIT_SIZE-1); + info->state->icount.tx++; + + if (info->xmit.head == info->xmit.tail) + anakin_stop_tx(port, 0); +} + +static inline void +anakin_transmit_x_char(struct uart_info *info) +{ + struct uart_port *port = info->port; + + anakin_out(port, 0x14, info->x_char); + anakin_out(port, 0x18, anakin_in(port, 0x18) | SENDREQUEST); + info->state->icount.tx++; + info->x_char = 0; +} + +static void +anakin_start_tx(struct uart_port *port, u_int nonempty, u_int from_tty) +{ + unsigned int flags; + + save_flags_cli(flags); + + // is it this... or below: if (nonempty + if (!txenable[port->irq]) { + txenable[port->irq] = TXENABLE; + + if ((anakin_in(port, 0x10) & TXEMPTY) && nonempty) { + anakin_transmit_buffer((struct uart_info*)port->unused); + } + } + + restore_flags(flags); +} + +static void +anakin_stop_rx(struct uart_port *port) +{ + unsigned long flags; + + save_flags_cli(flags); + while (anakin_in(port, 0x10) & RXRELEASE) + anakin_in(port, 0x14); + anakin_out(port, 0x18, anakin_in(port, 0x18) | BLOCKRX); + restore_flags(flags); +} + +static void +anakin_enable_ms(struct uart_port *port) +{ +} + +static inline void +anakin_rx_chars(struct uart_info *info) +{ + unsigned int ch; + struct tty_struct *tty = info->tty; + + if (!(anakin_in(info->port, 0x10) & RXRELEASE)) + return; + + ch = anakin_in(info->port, 0x14) & 0xff; + + if (tty->flip.count < TTY_FLIPBUF_SIZE) { + *tty->flip.char_buf_ptr++ = ch; + *tty->flip.flag_buf_ptr++ = TTY_NORMAL; + info->state->icount.rx++; + tty->flip.count++; + } + tty_flip_buffer_push(tty); +} + +static inline void +anakin_overrun_chars(struct uart_info *info) +{ + unsigned int ch; + + ch = anakin_in(info->port, 0x14); + info->state->icount.overrun++; +} + +static inline void +anakin_tx_chars(struct uart_info *info) +{ + if (info->x_char) { + anakin_transmit_x_char(info); + return; + } + + if (info->xmit.head == info->xmit.tail + || info->tty->stopped + || info->tty->hw_stopped) { + anakin_stop_tx(info->port, 0); + return; + } + + anakin_transmit_buffer(info); + + if (CIRC_CNT(info->xmit.head, + info->xmit.tail, + UART_XMIT_SIZE) < WAKEUP_CHARS) + uart_event(info, EVT_WRITE_WAKEUP); +} + +static void +anakin_int(int irq, void *dev_id, struct pt_regs *regs) +{ + unsigned int status; + struct uart_info *info = dev_id; + + status = anakin_in(info->port, 0x1c); + + if (status & RX) + anakin_rx_chars(info); + + if (status & OVERRUN) + anakin_overrun_chars(info); + + if (txenable[info->port->irq] && (status & TX)) + anakin_tx_chars(info); +} + +static u_int +anakin_tx_empty(struct uart_port *port) +{ + return anakin_in(port, 0x10) & TXEMPTY ? TIOCSER_TEMT : 0; +} + +static u_int +anakin_get_mctrl(struct uart_port *port) +{ + unsigned int status = 0; + + status |= (anakin_in(port, 0x10) & CTS ? TIOCM_CTS : 0); + status |= (anakin_in(port, 0x18) & DCD ? TIOCM_CAR : 0); + status |= (anakin_in(port, 0x18) & DTR ? TIOCM_DTR : 0); + status |= (anakin_in(port, 0x18) & RTS ? TIOCM_RTS : 0); + + return status; +} + +static void +anakin_set_mctrl(struct uart_port *port, u_int mctrl) +{ + unsigned int status; + + status = anakin_in(port, 0x18); + + if (mctrl & TIOCM_RTS) + status |= RTS; + else + status &= ~RTS; + + if (mctrl & TIOCM_CAR) + status |= DCD; + else + status &= ~DCD; + + anakin_out(port, 0x18, status); +} + +static void +anakin_break_ctl(struct uart_port *port, int break_state) +{ + unsigned int status; + + status = anakin_in(port, 0x20); + + if (break_state == -1) + status |= SETBREAK; + else + status &= ~SETBREAK; + + anakin_out(port, 0x20, status); +} + +static int +anakin_startup(struct uart_port *port, struct uart_info *info) +{ + int retval; + unsigned int read,write; + + /* + * Allocate the IRQ + */ + retval = request_irq(port->irq, anakin_int, SA_INTERRUPT, "serial_anakin", info); + if (retval) + return retval; + + port->ops->set_mctrl(port, info->mctrl); + + /* + * initialise the old status of the modem signals + */ + port->old_status = 0; + + /* + * Finally, disable IRQ and softIRQs for first byte) + */ + txenable[port->irq] = 0; + read = anakin_in(port, 0x18); + write = (read & ~(RTS | DTR | BLOCKRX)) | IRQENABLE; + anakin_out(port, 0x18, write); + + /* Store the uart_info pointer so we can reference it in + * anakin_start_tx() */ + port->unused = (u_int)info; + + return 0; +} + +static void +anakin_shutdown(struct uart_port *port, struct uart_info *info) +{ + /* + * Free the interrupt + */ + free_irq(port->irq, info); + + /* + * disable all interrupts, disable the port + */ + anakin_out(port, 0x18, anakin_in(port, 0x18) & ~IRQENABLE); +} + +static void +anakin_change_speed(struct uart_port *port, u_int cflag, u_int iflag, u_int quot) +{ + unsigned int flags; + + save_flags_cli(flags); + while (!(anakin_in(port, 0x10) & TXEMPTY)); + anakin_out(port, 0x10, (anakin_in(port, 0x10) & ~PRESCALER) + | (quot << 3)); + + //parity always set to none + anakin_out(port, 0x18, anakin_in(port, 0x18) & ~PARITY); + restore_flags(flags); +} + +static const char *anakin_type(struct port *port) +{ + return port->type == PORT_ANAKIN ? "ANAKIN" : NULL; +} + +static struct uart_ops anakin_pops = { + tx_empty: anakin_tx_empty, + set_mctrl: anakin_set_mctrl, + get_mctrl: anakin_get_mctrl, + stop_tx: anakin_stop_tx, + start_tx: anakin_start_tx, + stop_rx: anakin_stop_rx, + enable_ms: anakin_enable_ms, + break_ctl: anakin_break_ctl, + startup: anakin_startup, + shutdown: anakin_shutdown, + change_speed: anakin_change_speed, + type: anakin_type, +}; + +static struct uart_port anakin_ports[UART_NR] = { + { + base: IO_BASE + UART0, + irq: IRQ_UART0, + uartclk: 3686400, + fifosize: 0, + ops: &anakin_pops, + }, + { + base: IO_BASE + UART1, + irq: IRQ_UART1, + uartclk: 3686400, + fifosize: 0, + ops: &anakin_pops, + }, + { + base: IO_BASE + UART2, + irq: IRQ_UART2, + uartclk: 3686400, + fifosize: 0, + ops: &anakin_pops, + }, + { + base: IO_BASE + UART3, + irq: IRQ_UART3, + uartclk: 3686400, + fifosize: 0, + ops: &anakin_pops, + }, + { + base: IO_BASE + UART4, + irq: IRQ_UART4, + uartclk: 3686400, + fifosize: 0, + ops: &anakin_pops, + }, +}; + + +#ifdef CONFIG_SERIAL_ANAKIN_CONSOLE + +static void +anakin_console_write(struct console *co, const char *s, u_int count) +{ + struct uart_port *port = anakin_ports + co->index; + unsigned int flags, status, i; + + /* + * First save the status then disable the interrupts + */ + save_flags_cli(flags); + status = anakin_in(port, 0x18); + anakin_out(port, 0x18, status & ~IRQENABLE); + restore_flags(flags); + + /* + * Now, do each character + */ + for (i = 0; i < count; i++, s++) { + while (!(anakin_in(port, 0x10) & TXEMPTY)); + + /* + * Send the character out. + * If a LF, also do CR... + */ + anakin_out(port, 0x14, *s); + anakin_out(port, 0x18, anakin_in(port, 0x18) | SENDREQUEST); + + if (*s == 10) { + while (!(anakin_in(port, 0x10) & TXEMPTY)); + anakin_out(port, 0x14, 13); + anakin_out(port, 0x18, anakin_in(port, 0x18) + | SENDREQUEST); + } + } + + /* + * Finally, wait for transmitter to become empty + * and restore the interrupts + */ + while (!(anakin_in(port, 0x10) & TXEMPTY)); + + if (status & IRQENABLE) + save_flags_cli(flags); + anakin_out(port, 0x18, anakin_in(port, 0x18) | IRQENABLE); + restore_flags(flags); +} + +static kdev_t +anakin_console_device(struct console *co) +{ + return MKDEV(SERIAL_ANAKIN_MAJOR, SERIAL_ANAKIN_MINOR + co->index); +} + +/* + * Read the current UART setup. + */ +static void __init +anakin_console_get_options(struct uart_port *port, int *baud, int *parity, int *bits) +{ + int paritycode; + + *baud = GETBAUD (anakin_in(port, 0x10) & PRESCALER); + paritycode = GETPARITY(anakin_in(port, 0x18) & PARITY); + switch (paritycode) { + case NONEPARITY: *parity = 'n'; break; + case ODDPARITY: *parity = 'o'; break; + case EVENPARITY: *parity = 'e'; break; + } + *bits = 8; +} + +static int __init +anakin_console_setup(struct console *co, char *options) +{ + struct uart_port *port; + int baud = CONFIG_ANAKIN_DEFAULT_BAUDRATE; + int bits = 8; + int parity = 'n'; + + /* + * Check whether an invalid uart number has been specified, and + * if so, search for the first available port that does have + * console support. + */ + port = uart_get_console(anakin_ports, UART_NR, co); + + if (options) + uart_parse_options(options, &baud, &parity, &bits); + else + anakin_console_get_options(port, &baud, &parity, &bits); + + return uart_set_options(port, co, baud, parity, bits); +} + +static struct console anakin_console = { + name: SERIAL_ANAKIN_NAME, + write: anakin_console_write, + device: anakin_console_device, + setup: anakin_console_setup, + flags: CON_PRINTBUFFER, + index: -1, +}; + +void __init +anakin_console_init(void) +{ + register_console(&anakin_console); +} + +#define ANAKIN_CONSOLE &anakin_console +#else +#define ANAKIN_CONSOLE NULL +#endif + +static struct uart_register anakin_reg = { + normal_major: SERIAL_ANAKIN_MAJOR, + normal_name: SERIAL_ANAKIN_NAME, + normal_driver: &normal, + callout_major: CALLOUT_ANAKIN_MAJOR, + callout_name: CALLOUT_ANAKIN_NAME, + callout_driver: &callout, + table: anakin_table, + termios: anakin_termios, + termios_locked: anakin_termios_locked, + minor: SERIAL_ANAKIN_MINOR, + nr: UART_NR, + state: anakin_state, + port: anakin_ports, + cons: ANAKIN_CONSOLE, +}; + +static int __init +anakin_init(void) +{ + return uart_register_port(&anakin_reg); +} + +__initcall(anakin_init); + +MODULE_DESCRIPTION("Anakin serial driver"); +MODULE_AUTHOR("Tak-Shing Chan "); +MODULE_SUPPORTED_DEVICE("ttyAN"); +MODULE_LICENSE("GPL"); + +EXPORT_NO_SYMBOLS; diff -urN orig/drivers/serial/at91us3.c linux/drivers/serial/at91us3.c --- orig/drivers/serial/at91us3.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/serial/at91us3.c Thu Feb 27 23:21:11 2003 @@ -0,0 +1,871 @@ +/* + * linux/drivers/char/at91rm9200us3.c + * + * Driver for at91rm9200 serial port + * + * Based on drivers/char/serial.c, by Linus Torvalds, Theodore Ts'o. + * + * Copyright (C) 2002 Atmel Rousset + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + * + */ +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include + +#include + +#if defined(CONFIG_SERIAL_AT91US3_CONSOLE) && defined(CONFIG_MAGIC_SYSRQ) +#define SUPPORT_SYSRQ +#endif + +#include + +#include + +#define SERIAL_AT91US3_MAJOR TTY_MAJOR +#define SERIAL_AT91US3_MINOR 64 + + +#define CALLOUT_AT91US3_MAJOR TTYAUX_MAJOR +#define CALLOUT_AT91US3_MINOR 64 + + +/* + * Access macros for the AT91US3DK USART3s + */ + +#define AT91C_PDC_TXTEN (1 << 8) +#define UART_GET_UTCR(p) readl(&(((AT91PS_USART)(p)->membase)->US_CR)) +#define UART_GET_UTMR(p) readl(&(((AT91PS_USART)(p)->membase)->US_MR)) +#define UART_GET_UTCSR(p) readl(&(((AT91PS_USART)(p)->membase)->US_CSR)) +#define UART_GET_CHAR(p) readl(&(((AT91PS_USART)(p)->membase)->US_RHR)) +#define UART_GET_UTIMR(p) readl(&(((AT91PS_USART)(p)->membase)->US_IMR)) +#define UART_GET_UTBRGR(p) readl(&(((AT91PS_USART)(p)->membase)->US_BRGR)) +#define UART_GET_TPR(p) readl(&(((AT91PS_USART)(p)->membase)->US_TPR)) +#define UART_GET_PTSR(p) readl(&(((AT91PS_USART)(p)->membase)->US_PTSR)) + +#define UART_PUT_UTCR(p,v) writel(v,&(((AT91PS_USART)(p)->membase)->US_CR)) +#define UART_PUT_UTMR(p,v) writel(v,&(((AT91PS_USART)(p)->membase)->US_MR)) +#define UART_PUT_CHAR(p,v) writel(v,&(((AT91PS_USART)(p)->membase)->US_THR)) +#define UART_PUT_UTIER(p,v) writel(v,&(((AT91PS_USART)(p)->membase)->US_IER)) +#define UART_PUT_UTIDR(p,v) writel(v,&(((AT91PS_USART)(p)->membase)->US_IDR)) +#define UART_PUT_UTBRGR(p,v) writel(v,&(((AT91PS_USART)(p)->membase)->US_BRGR)) +#define UART_PUT_PDC_PTCR(p, v) writel(v,&(((AT91PS_USART)(p)->membase)->US_PTCR)) +#define UART_SET_PDC(p,v,l) {writel(v,&(((AT91PS_USART)(p)->membase)->US_TPR));\ + writel(l,&(((AT91PS_USART)(p)->membase)->US_TCR));} +#define UART_SET_NPDC(p,v,l) {writel(v,&(((AT91PS_USART)(p)->membase)->US_TNPR));\ + writel(l,&(((AT91PS_USART)(p)->membase)->US_TNCR));} + +#define UART_PORT_SIZE SZ_16K +#define UART_PORT_DBGU_SIZE 512 + +#define AT91US3_ISR_PASS_LIMIT 256 + +static int at91us3_verify_port(struct uart_port *port, struct serial_struct *ser); +static const char *at91us3_type(struct uart_port *port); +static void at91us3_config_port(struct uart_port *port, int flags); +static int at91us3_request_port(struct uart_port *port); +static void at91us3_release_port(struct uart_port *port); +static void at91us3_change_speed(struct uart_port *port, u_int cflag, u_int iflag, u_int quot); +static void at91us3_set_mctrl(struct uart_port *port, u_int mctrl); +static u_int at91us3_get_mctrl(struct uart_port *port); +static void at91us3_set_mctrl(struct uart_port *port, u_int mctrl); +static u_int at91us3_tx_empty(struct uart_port *port); +static u_int at91us3_tx_empty(struct uart_port *port); +static void at91us3_int(int irq, void *dev_id, struct pt_regs *regs); +static void at91us3_tx_chars(struct uart_info *info); +static void at91us3_enable_ms(struct uart_port *port); +static void at91us3_break_ctl(struct uart_port *port, int break_state); +static void at91us3_rx_chars(struct uart_info *info, struct pt_regs *regs); +static void at91us3_stop_rx(struct uart_port *port); +static void at91us3_start_tx(struct uart_port *port, u_int nonempty, u_int from_tty); +static void at91us3_stop_tx(struct uart_port *port, u_int from_tty); +static int at91us3_startup(struct uart_port *port, struct uart_info *info); +static void at91us3_shutdown(struct uart_port *port, struct uart_info *info); + +static struct uart_ops at91us3_pops = { + tx_empty: at91us3_tx_empty, + set_mctrl: at91us3_set_mctrl, + get_mctrl: at91us3_get_mctrl, + stop_tx: at91us3_stop_tx, + start_tx: at91us3_start_tx, + stop_rx: at91us3_stop_rx, + enable_ms: at91us3_enable_ms, + break_ctl: at91us3_break_ctl, + startup: at91us3_startup, + shutdown: at91us3_shutdown, + change_speed: at91us3_change_speed, + type: at91us3_type, + release_port: at91us3_release_port, + request_port: at91us3_request_port, + config_port: at91us3_config_port, + verify_port: at91us3_verify_port, +}; + +#define UART_NR 5 /* 4 USART3's and one debug port */ +struct uart_port at91us3_ports[UART_NR]; + +static struct tty_driver normal, callout; +static struct tty_struct *at91us3_table[UART_NR]; +static struct termios *at91us3_termios[UART_NR], *at91us3_termios_locked[UART_NR]; +#ifdef SUPPORT_SYSRQ +static struct console at91us3_console; +#endif + + +// ============================================================ +// Private definitions for AT91RM9200DK board +// ============================================================ +#ifdef CONFIG_ARCH_AT91RM9200DK + +#define PORT_AT91US3 PORT_AT91RM9200 + +void __init at91rm9200dk_register_us0(void) +{ + ((AT91PS_SYS)AT91C_VA_BASE_SYS)->PIOA_PDR = AT91C_PIO_PA17 | AT91C_PIO_PA18 | + AT91C_PIO_PA19 | AT91C_PIO_PA20 | + AT91C_PIO_PA21; + at91us3_ports[0].membase = (AT91PS_USART)AT91C_VA_BASE_US0; + at91us3_ports[0].mapbase = (int) AT91C_BASE_US0; + at91us3_ports[0].irq = AT91C_ID_US0; + at91us3_ports[0].iotype = SERIAL_IO_MEM; + at91us3_ports[0].flags = ASYNC_BOOT_AUTOCONF; + at91us3_ports[0].uartclk = AT91C_MASTER_CLK; + at91us3_ports[0].ops = &at91us3_pops; + at91us3_ports[0].fifosize = 0; +} + +void __init at91rm9200dk_register_us1(void) +{ + ((AT91PS_SYS)AT91C_VA_BASE_SYS)->PIOB_PDR = AT91C_PIO_PB18 | AT91C_PIO_PB19 | + AT91C_PIO_PB20 | AT91C_PIO_PB21 | + AT91C_PIO_PB23 | AT91C_PIO_PB24 | + AT91C_PIO_PB25 | AT91C_PIO_PB26; + at91us3_ports[1].membase = (AT91PS_USART)AT91C_VA_BASE_US1; + at91us3_ports[1].mapbase = (int) AT91C_BASE_US1; + at91us3_ports[1].irq = AT91C_ID_US1; + at91us3_ports[1].iotype = SERIAL_IO_MEM; + at91us3_ports[1].flags = ASYNC_BOOT_AUTOCONF; + at91us3_ports[1].uartclk = AT91C_MASTER_CLK; + at91us3_ports[1].ops = &at91us3_pops; + at91us3_ports[1].fifosize = 0; + + at91us3_enable_ms(&at91us3_ports[1]); +} + +void __init at91rm9200dk_register_us2(void) +{ + ((AT91PS_SYS)AT91C_VA_BASE_SYS)->PIOA_PDR = AT91C_PIO_PA22 | AT91C_PIO_PA23; + at91us3_ports[2].membase = (AT91PS_USART)AT91C_VA_BASE_US2; + at91us3_ports[2].mapbase = (int) AT91C_BASE_US2; + at91us3_ports[2].irq = AT91C_ID_US2; + at91us3_ports[2].iotype = SERIAL_IO_MEM; + at91us3_ports[2].flags = ASYNC_BOOT_AUTOCONF; + at91us3_ports[2].uartclk = AT91C_MASTER_CLK; + at91us3_ports[2].ops = &at91us3_pops; + at91us3_ports[2].fifosize = 0; +} + +void __init at91rm9200dk_register_us3(void) +{ + ((AT91PS_SYS)AT91C_VA_BASE_SYS)->PIOA_PDR = AT91C_PIO_PA5 | AT91C_PIO_PA6 ; + ((AT91PS_SYS)AT91C_VA_BASE_SYS)->PIOA_BSR = AT91C_PIO_PA5 | AT91C_PIO_PA6; + ((AT91PS_SYS)AT91C_VA_BASE_SYS)->PIOB_PDR = AT91C_PIO_PA1 | AT91C_PIO_PA2 ; + ((AT91PS_SYS)AT91C_VA_BASE_SYS)->PIOB_BSR = AT91C_PIO_PA1 | AT91C_PIO_PA2; + at91us3_ports[3].membase = (AT91PS_USART)AT91C_VA_BASE_US3; + at91us3_ports[3].mapbase = (int) AT91C_BASE_US3; + at91us3_ports[3].irq = AT91C_ID_US3; + at91us3_ports[3].iotype = SERIAL_IO_MEM; + at91us3_ports[3].flags = ASYNC_BOOT_AUTOCONF; + at91us3_ports[3].uartclk = AT91C_MASTER_CLK; + at91us3_ports[3].ops = &at91us3_pops; + at91us3_ports[3].fifosize = 0; +} + +void __init at91rm9200dk_register_dbgu(void) +{ + ((AT91PS_SYS)AT91C_VA_BASE_SYS)->PIOA_PDR = AT91C_PIO_PA30 | AT91C_PIO_PA31; + at91us3_ports[4].membase = (AT91PS_USART)AT91C_VA_BASE_DBGU; + at91us3_ports[4].mapbase = (int) AT91C_BASE_DBGU; + at91us3_ports[4].irq = AT91C_ID_SYS; + at91us3_ports[4].iotype = SERIAL_IO_MEM; + at91us3_ports[4].flags = ASYNC_BOOT_AUTOCONF; + at91us3_ports[4].uartclk = AT91C_MASTER_CLK; + at91us3_ports[4].ops = &at91us3_pops; + at91us3_ports[4].fifosize = 0; +} + +#endif + + + + +/* + * interrupts disabled on entry + */ +static void at91us3_stop_tx(struct uart_port *port, u_int from_tty) +{ + UART_PUT_UTIDR(port, AT91C_US_TXEMPTY); + port->read_status_mask &= ~AT91C_US_TXEMPTY; +} + +/* + * interrupts may not be disabled on entry + */ +static void at91us3_start_tx(struct uart_port *port, u_int nonempty, u_int from_tty) +{ + + if (nonempty) { + unsigned long flags; + + local_irq_save(flags); + port->read_status_mask |= AT91C_US_TXEMPTY; + UART_PUT_UTIER(port, AT91C_US_TXEMPTY); + local_irq_restore(flags); + } +} + +/* + * Interrupts enabled + */ +static void at91us3_stop_rx(struct uart_port *port) +{ + UART_PUT_UTIDR(port, AT91C_US_RXRDY); +} + +static void at91us3_enable_ms(struct uart_port *port) +{ + unsigned int mode; + mode = UART_GET_UTMR(port); + mode &= ~AT91C_US_USMODE; + mode |= AT91C_US_USMODE_MODEM ; + UART_PUT_UTMR(port,mode); + UART_PUT_UTIER(port, AT91C_US_RIIC | AT91C_US_DSRIC | AT91C_US_DCDIC | AT91C_US_CTSIC ); + +} + +static void +at91us3_rx_chars(struct uart_info *info, struct pt_regs *regs) +{ + struct tty_struct *tty = info->tty; + unsigned int status, ch, flg, ignored = 0; + struct uart_port *port = info->port; + + status = UART_GET_UTCSR(port); + while (status & (AT91C_US_RXRDY)) { + ch = UART_GET_CHAR(port); + if (tty->flip.count >= TTY_FLIPBUF_SIZE) + goto ignore_char; + port->icount.rx++; + + flg = TTY_NORMAL; + + /* + * note that the error handling code is + * out of the main execution path + */ + if (status & (AT91C_US_PARE | AT91C_US_FRAME | AT91C_US_OVRE)) + goto handle_error; + + if (uart_handle_sysrq_char(info, ch, regs)) + goto ignore_char; + + error_return: + *tty->flip.flag_buf_ptr++ = flg; + *tty->flip.char_buf_ptr++ = ch; + tty->flip.count++; + ignore_char: + status = UART_GET_UTCSR(port); + } + out: + tty_flip_buffer_push(tty); + return; + + handle_error: + if (status & (AT91C_US_PARE | AT91C_US_FRAME | AT91C_US_OVRE)) + UART_PUT_UTCR(port, AT91C_US_RSTSTA); /* clear error */ + if (status & (AT91C_US_PARE)) + port->icount.parity++; + else if (status & (AT91C_US_FRAME)) + port->icount.frame++; + if (status & (AT91C_US_OVRE)) + port->icount.overrun++; + + if (status & port->ignore_status_mask) { + if (++ignored > 100) + goto out; + goto ignore_char; + } + + status &= port->read_status_mask; + + UART_PUT_UTCR(port, AT91C_US_RSTSTA); /* clear error */ + if (status & AT91C_US_PARE) + flg = TTY_PARITY; + else if (status & AT91C_US_FRAME) + flg = TTY_FRAME; + + if (status & AT91C_US_OVRE) { + /* + * overrun does *not* affect the character + * we read from the FIFO + */ + *tty->flip.flag_buf_ptr++ = flg; + *tty->flip.char_buf_ptr++ = ch; + tty->flip.count++; + if (tty->flip.count >= TTY_FLIPBUF_SIZE) + goto ignore_char; + ch = 0; + flg = TTY_OVERRUN; + } +#ifdef SUPPORT_SYSRQ + info->sysrq = 0; +#endif + goto error_return; +} + +static void at91us3_tx_chars(struct uart_info *info) +{ + struct uart_port *port = info->port; + u_int bytes_to_send; + u_int circ_cnt_to_end; + u_int xmit_pt; + + // Update the tail pointer + if (UART_GET_UTCSR(port) & AT91C_US_TXEMPTY) { + xmit_pt = (bus_to_virt((u_int)UART_GET_TPR(port)) - (u_int)info->xmit.buf); + if (xmit_pt > info->xmit.tail) + port->icount.tx += (xmit_pt - info->xmit.tail); + else + port->icount.tx += (xmit_pt + (UART_XMIT_SIZE -1 - info->xmit.tail)); + + info->xmit.tail = xmit_pt; + } + + if (port->x_char) { + UART_PUT_CHAR(port, port->x_char); + port->icount.tx++; + port->x_char = 0; + return; + } + + bytes_to_send = CIRC_CNT(info->xmit.head, info->xmit.tail, UART_XMIT_SIZE); + + if (bytes_to_send <= WAKEUP_CHARS) + uart_event(info, EVT_WRITE_WAKEUP); + else + bytes_to_send -= WAKEUP_CHARS; + + if (!bytes_to_send || info->tty->stopped || info->tty->hw_stopped) { + at91us3_stop_tx(info->port, 0); + return; + } + + // Init PDC and next PDC + xmit_pt = info->xmit.tail + bytes_to_send; + if (xmit_pt < UART_XMIT_SIZE) { + UART_SET_PDC (port, virt_to_bus(info->xmit.buf + info->xmit.tail), bytes_to_send); + UART_SET_NPDC(port, virt_to_bus(info->xmit.buf + xmit_pt), 0); + } + else { + circ_cnt_to_end = CIRC_CNT_TO_END(info->xmit.head, info->xmit.tail, UART_XMIT_SIZE); + UART_SET_PDC(port, virt_to_bus(info->xmit.buf + info->xmit.tail), circ_cnt_to_end); + UART_SET_NPDC(port, virt_to_bus(info->xmit.buf), bytes_to_send - circ_cnt_to_end); + } +} + +static void at91us3_int(int irq, void *dev_id, struct pt_regs *regs) +{ + struct uart_info *info = dev_id; + struct uart_port *port = info->port; + unsigned int status, unmask_status, pass_counter = 0; + + + unmask_status = UART_GET_UTCSR(port); + status = unmask_status & UART_GET_UTIMR(port); + if (status) { + status &= port->read_status_mask | ~AT91C_US_TXEMPTY; + do { + if (status & AT91C_US_RXRDY) { + at91us3_rx_chars(info, regs); + } + + /* Clear the relevent break bits */ + if (status & AT91C_US_RXBRK){ + UART_PUT_UTCR(port, status & AT91C_US_RSTSTA); + + port->icount.brk++; + +#ifdef SUPPORT_SYSRQ + if (port->line == at91us3_console.index && + !info->sysrq) { + info->sysrq = jiffies + HZ*5; + } +#endif + } + if (status & AT91C_US_TXEMPTY){ + at91us3_tx_chars(info); + } + if (status & AT91C_US_DCDIC){ + uart_handle_dcd_change(info, unmask_status & AT91C_US_DCD); + } + if (unmask_status & AT91C_US_CTSIC){ + uart_handle_cts_change(info, unmask_status & AT91C_US_CTS); + } + if (unmask_status & AT91C_US_DSRIC){ + port->icount.dsr++; + } + if (status & AT91C_US_RIIC){ + port->icount.rng++; + } + if (status & ( AT91C_US_DCDIC | AT91C_US_CTSIC | AT91C_US_DSRIC | AT91C_US_RIIC) ){ + wake_up_interruptible(&info->delta_msr_wait); + } + if (pass_counter++ > AT91US3_ISR_PASS_LIMIT) + break; + status = UART_GET_UTCSR(port); + status &= port->read_status_mask | ~AT91C_US_TXEMPTY; + } + while (status & (AT91C_US_TXEMPTY | AT91C_US_RXRDY)); + } +} + +/* + * Return TIOCSER_TEMT when transmitter is not busy. + */ +static u_int at91us3_tx_empty(struct uart_port *port) +{ + return UART_GET_UTCSR(port) & AT91C_US_TXEMPTY ? TIOCSER_TEMT : 0; +} + +static u_int at91us3_get_mctrl(struct uart_port *port) +{ + u_int ret = 0; + unsigned int status,mode; + + mode = UART_GET_UTMR(port); + if ( ( mode & AT91C_US_USMODE ) & AT91C_US_USMODE_MODEM ) { + status=UART_GET_UTCSR(port); + if (status & AT91C_US_DCD) + ret |= TIOCM_CD; + if (status & AT91C_US_CTS) + ret |= TIOCM_CTS; + if (status & AT91C_US_DSR) + ret |= TIOCM_DSR; + if (status & AT91C_US_RI) + ret |= TIOCM_RI; + } else { + ret = TIOCM_CD | TIOCM_CTS | TIOCM_DSR; + + } + + return ret; +} + +static void at91us3_set_mctrl(struct uart_port *port, u_int mctrl) +{ + unsigned int control,mode; + mode = UART_GET_UTMR(port); + control = 0; + + if ( ( mode & AT91C_US_USMODE ) & AT91C_US_USMODE_MODEM ) { + if (mctrl & TIOCM_RTS) + control |= AT91C_US_RTSEN; + else + control |= AT91C_US_RTSDIS; + + if (mctrl & TIOCM_DTR) + control |= AT91C_US_DTREN; + + else + control |= AT91C_US_DTRDIS; + UART_PUT_UTCR(port,control); + } +} + +/* + * Interrupts always disabled. + */ +static void at91us3_break_ctl(struct uart_port *port, int break_state) +{ + UART_PUT_UTCR(port, AT91C_US_STPBRK); +} + +static int at91us3_startup(struct uart_port *port, struct uart_info *info) +{ + /* Allocate the IRQ */ + int retval; + retval = request_irq(port->irq, at91us3_int, SA_SHIRQ, "serial_at91us3", info); + if (retval) { + printk("***at91us3_startup2 Error: can't get irq\n"); + return retval; + } + + /* Enable peripheral clock if required */ + if (port->irq != AT91C_ID_SYS) + ((AT91PS_SYS)AT91C_VA_BASE_SYS)->PMC_PCER = (1 << port->irq); + + port->read_status_mask = AT91C_US_RXRDY | AT91C_US_TXEMPTY | AT91C_US_OVRE | + AT91C_US_FRAME | AT91C_US_PARE | AT91C_US_RXBRK; + + UART_SET_PDC(port, virt_to_bus(info->xmit.buf + info->xmit.tail), 0); + UART_SET_NPDC(port, virt_to_bus(info->xmit.buf + info->xmit.tail), 0); + + /* Finally, clear and enable interrupts */ + UART_PUT_UTIDR(port, -1); + UART_PUT_UTCR(port, AT91C_US_TXEN | AT91C_US_RXEN); /* enable xmit & rcvr */ + UART_PUT_PDC_PTCR(port, AT91C_PDC_TXTEN); /* enable xmit & rcvr */ + UART_PUT_UTIER(port, AT91C_US_RXRDY ); /* do receive only */ + return 0; +} + +static void at91us3_shutdown(struct uart_port *port, struct uart_info *info) +{ + /* Disable peripheral clock if required */ + if (port->irq != AT91C_ID_SYS) + ((AT91PS_SYS)AT91C_VA_BASE_SYS)->PMC_PCDR = (1 << port->irq); + + /* Free the interrupt */ + free_irq(port->irq, info); + + /* Disable all interrupts, port and break condition. */ + UART_PUT_UTCR(port, AT91C_US_RSTSTA); + UART_PUT_UTIDR(port, -1); +} + +static void at91us3_change_speed(struct uart_port *port, u_int cflag, u_int iflag, u_int quot) +{ + unsigned long flags; + u_int utmr; + int imr; + + /* byte size and parity */ + switch (cflag & CSIZE) { + case CS7: utmr = 0; break; + default: utmr = AT91C_US_CHRL_8_BITS; break; + } + if (cflag & CSTOPB) + utmr |= AT91C_US_NBSTOP_1_BIT; + if (cflag & PARENB) { + if (!(cflag & PARODD)) + utmr |= AT91C_US_PAR_ODD; + else + utmr |= AT91C_US_PAR_EVEN; + } + else + utmr |= AT91C_US_PAR_NONE; + + port->read_status_mask |= AT91C_US_OVRE; + if (iflag & INPCK) + port->read_status_mask |= AT91C_US_FRAME | AT91C_US_PARE; + if (iflag & (BRKINT | PARMRK)) + port->read_status_mask |= AT91C_US_RXBRK; + + /* + * Characters to ignore + */ + port->ignore_status_mask = 0; + if (iflag & IGNPAR) + port->ignore_status_mask |= (AT91C_US_FRAME | AT91C_US_PARE); + if (iflag & IGNBRK) { + port->ignore_status_mask |= AT91C_US_RXBRK; + /* + * If we're ignoring parity and break indicators, + * ignore overruns too (for real raw support). + */ + if (iflag & IGNPAR) + port->ignore_status_mask |= AT91C_US_OVRE; + } + + /* first, disable interrupts and drain transmitter */ + local_irq_save(flags); + imr = UART_GET_UTIMR(port); /* get interrupt mask */ + UART_PUT_UTIDR(port, -1); + local_irq_restore(flags); + while (!(UART_GET_UTCSR(port) & AT91C_US_TXEMPTY)); + + /* then, disable everything */ + UART_PUT_UTCR(port, AT91C_US_TXDIS | AT91C_US_RXDIS); + + /* set the parity, stop bits and data size */ + UART_PUT_UTMR(port, utmr); + /* set the baud rate */ + UART_PUT_UTBRGR(port, quot); + + UART_PUT_UTCR(port, AT91C_US_TXEN | AT91C_US_RXEN); + + UART_PUT_UTIER(port, imr); /* set them back the way they were */ +} + +static const char *at91us3_type(struct uart_port *port) +{ + return port->type == PORT_AT91US3 ? "AT91US3" : NULL; +} + +/* + * Release the memory region(s) being used by 'port'. + */ +static void at91us3_release_port(struct uart_port *port) +{ + release_mem_region(port->mapbase, UART_PORT_SIZE); +} + +/* + * Request the memory region(s) being used by 'port'. + */ +static int at91us3_request_port(struct uart_port *port) +{ + return request_mem_region(port->mapbase, + port->membase == (void *) &(((AT91PS_SYS)AT91C_VA_BASE_SYS)->DBGU_CR) ? UART_PORT_DBGU_SIZE : UART_PORT_SIZE, + "serial_at91us3") != NULL ? 0 : -EBUSY; +} + +/* + * Configure/autoconfigure the port. + */ +static void at91us3_config_port(struct uart_port *port, int flags) +{ + if (!(flags & UART_CONFIG_TYPE)) + printk("at91us3_config_port Error, wrong type\n"); + else + if (at91us3_request_port(port) == 0) + port->type = PORT_AT91US3; + else + printk("at91us3_config_port Error, can't get port \n"); +} + +/* + * Verify the new serial_struct (for TIOCSSERIAL). + * The only change we allow are to the flags and type, and + * even then only between PORT_AT91US3 and PORT_UNKNOWN + */ +static int at91us3_verify_port(struct uart_port *port, struct serial_struct *ser) +{ + int ret = 0; + if (ser->type != PORT_UNKNOWN && ser->type != PORT_AT91US3) + ret = -EINVAL; + if (port->irq != ser->irq) + ret = -EINVAL; + if (ser->io_type != SERIAL_IO_MEM) + ret = -EINVAL; + if (port->uartclk / 16 != ser->baud_base) + ret = -EINVAL; + if ((void *)port->mapbase != ser->iomem_base) + ret = -EINVAL; + if (port->iobase != ser->port) + ret = -EINVAL; + if (ser->hub6 != 0) + ret = -EINVAL; + return ret; +} + +#ifdef CONFIG_SERIAL_AT91US3_CONSOLE + +/* + * Interrupts are disabled on entering + */ +static void at91us3_console_write(struct console *co, const char *s, u_int count) +{ + struct uart_port *port = at91us3_ports + co->index; + u_int status, i; + int imr; + + /* + * First, save UTCR3 and then disable interrupts + */ + imr = UART_GET_UTIMR(port); /* get interrupt mask */ + UART_PUT_UTIDR(port, AT91C_US_RXRDY | AT91C_US_TXEMPTY); + + /* + * Now, do each character + */ + + do { + status = UART_GET_UTCSR(port); + } while (!(status & AT91C_US_TXEMPTY)); + + for (i = 0; i < count; i++) { + do { + status = UART_GET_UTCSR(port); + } while (!(status & AT91C_US_TXRDY)); + UART_PUT_CHAR(port, s[i]); + if (s[i] == '\n') { + do { + status = UART_GET_UTCSR(port); + } while (!(status & AT91C_US_TXRDY)); + UART_PUT_CHAR(port, '\r'); + } + } + + /* + * Finally, wait for transmitter to become empty + * and restore UTCR3 + */ + do { + status = UART_GET_UTCSR(port); + } while (!(status & AT91C_US_TXRDY)); + UART_PUT_UTIER(port, imr); /* set them back the way they were */ +} + +static kdev_t at91us3_console_device(struct console *co) +{ + return MKDEV(SERIAL_AT91US3_MAJOR,SERIAL_AT91US3_MINOR + co->index); +} + +/* + * If the port was already initialised (eg, by a boot loader), try to determine + * the current setup. + */ +static void __init +at91us3_console_get_options(struct uart_port *port, int *baud, int *parity, int *bits) +{ + unsigned int utcr3, utmr, quot; + + utcr3 = UART_GET_UTCR(port) & (AT91C_US_RXEN | AT91C_US_TXEN); + if (utcr3 == (AT91C_US_RXEN | AT91C_US_TXEN)) { + /* ok, the port was enabled */ + + utmr = UART_GET_UTMR(port) & AT91C_US_PAR; + + *parity = 'n'; + if (utmr == AT91C_US_PAR_EVEN) + *parity = 'e'; + else + if (utmr == AT91C_US_PAR_ODD) + *parity = 'o'; + } + + utmr = UART_GET_UTMR(port) & AT91C_US_CHRL; + if (utmr == AT91C_US_CHRL_8_BITS) + *bits = 8; + else + *bits = 7; + + quot = UART_GET_UTBRGR(port); + *baud = port->uartclk / (16 * (quot)); +} + +#ifndef CONFIG_AT91US3_DEFAULT_BAUDRATE +#define CONFIG_AT91US3_DEFAULT_BAUDRATE 115200 +#endif + +static int __init +at91us3_console_setup(struct console *co, char *options) +{ + struct uart_port *port; + int baud = CONFIG_AT91US3_DEFAULT_BAUDRATE; + int bits = 8; + int parity = 'n'; + int flow = 'n'; + /* + * Check whether an invalid uart number has been specified, and + * if so, search for the first available port that does have + * console support. + */ + port = uart_get_console(at91us3_ports, UART_NR, co); + + if (options) + uart_parse_options(options, &baud, &parity, &bits, &flow); + else + at91us3_console_get_options(port, &baud, &parity, &bits); + + UART_PUT_UTCR(port, AT91C_US_TXEN | AT91C_US_RXEN); /* enable xmit & rcvr */ + return uart_set_options(port, co, baud, parity, bits, flow); +} + +static struct console at91us3_console = { + name: "ttyS", + write: at91us3_console_write, + device: at91us3_console_device, + setup: at91us3_console_setup, + flags: CON_PRINTBUFFER, + index: -1, +}; + +void __init at91us3_rs_console_init(void) +{ + register_console(&at91us3_console); +} + +#define AT91US3_CONSOLE &at91us3_console +#else +#define AT91US3_CONSOLE NULL +#endif + +static struct uart_driver at91us3_reg = { + owner: THIS_MODULE, + normal_major: SERIAL_AT91US3_MAJOR, +#ifdef CONFIG_DEVFS_FS + normal_name: "ttyS%d", + callout_name: "cua%d", +#else + normal_name: "ttyS", + callout_name: "cua", +#endif + normal_driver: &normal, + callout_major: CALLOUT_AT91US3_MAJOR, + callout_driver: &callout, + table: at91us3_table, + termios: at91us3_termios, + termios_locked: at91us3_termios_locked, + minor: SERIAL_AT91US3_MINOR, + nr: UART_NR, + port: at91us3_ports, + cons: AT91US3_CONSOLE, +}; + +static int __init at91us3_serial_init(void) +{ + return uart_register_driver(&at91us3_reg); +} + +static void __exit at91us3_serial_exit(void) +{ + uart_unregister_driver(&at91us3_reg); +} + +module_init(at91us3_serial_init); +module_exit(at91us3_serial_exit); + +EXPORT_NO_SYMBOLS; + +MODULE_LICENSE("GPL"); +MODULE_AUTHOR("Atmel Rousset"); +MODULE_DESCRIPTION("AT91US3 USART3 serial driver"); diff -urN orig/drivers/serial/clps711x.c linux/drivers/serial/clps711x.c --- orig/drivers/serial/clps711x.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/serial/clps711x.c Mon Jan 27 21:37:37 2003 @@ -0,0 +1,641 @@ +/* + * linux/drivers/char/serial_clps711x.c + * + * Driver for CLPS711x serial ports + * + * Based on drivers/char/serial.c, by Linus Torvalds, Theodore Ts'o. + * + * Copyright 1999 ARM Limited + * Copyright (C) 2000 Deep Blue Solutions Ltd. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + * + * $Id: clps711x.c,v 1.12.2.2 2002/10/24 09:53:25 rmk Exp $ + * + */ +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include + +#include + +#include + +#define UART_NR 2 + +#define SERIAL_CLPS711X_NAME "ttyAM" +#define SERIAL_CLPS711X_MAJOR 204 +#define SERIAL_CLPS711X_MINOR 16 +#define SERIAL_CLPS711X_NR UART_NR + +#define CALLOUT_CLPS711X_NAME "cuaam" +#define CALLOUT_CLPS711X_MAJOR 205 +#define CALLOUT_CLPS711X_MINOR 16 +#define CALLOUT_CLPS711X_NR UART_NR + +static struct tty_driver normal, callout; +static struct tty_struct *clps711x_table[UART_NR]; +static struct termios *clps711x_termios[UART_NR], *clps711x_termios_locked[UART_NR]; + +/* + * We use the relevant SYSCON register as a base address for these ports. + */ +#define UBRLCR(port) ((port)->iobase + UBRLCR1 - SYSCON1) +#define UARTDR(port) ((port)->iobase + UARTDR1 - SYSCON1) +#define SYSFLG(port) ((port)->iobase + SYSFLG1 - SYSCON1) +#define SYSCON(port) ((port)->iobase + SYSCON1 - SYSCON1) + +#define TX_IRQ(port) ((port)->irq) +#define RX_IRQ(port) ((port)->irq + 1) + +#define tx_enabled(port) ((port)->unused[0]) + +#define UART_ANY_ERR (UARTDR_FRMERR | UARTDR_PARERR | UARTDR_OVERR) + +static void clps711xuart_stop_tx(struct uart_port *port, u_int from_tty) +{ + if (tx_enabled(port)) { + disable_irq(TX_IRQ(port)); + tx_enabled(port) = 0; + } +} + +static void clps711xuart_start_tx(struct uart_port *port, u_int nonempty, u_int from_tty) +{ + if (nonempty && !tx_enabled(port)) { + enable_irq(TX_IRQ(port)); + tx_enabled(port) = 1; + } +} + +static void clps711xuart_stop_rx(struct uart_port *port) +{ + disable_irq(RX_IRQ(port)); +} + +static void clps711xuart_enable_ms(struct uart_port *port) +{ +} + +#if 0 +static void ambauart_modem_status(struct uart_info *info) +{ + unsigned int status, delta; + struct uart_icount *icount = &info->port->icount; + + UART_PUT_ICR(info->port, 0); + + status = UART_GET_FR(info->port) & AMBA_UARTFR_MODEM_ANY; + + delta = status ^ info->port->old_status; + info->port->old_status = status; + + if (!delta) + return; + + if (delta & AMBA_UARTFR_DCD) + uart_handle_dcd_change(info, status & AMBA_UARTFR_DCD); + + if (delta & AMBA_UARTFR_DSR) + icount->dsr++; + + if (delta & AMBA_UARTFR_CTS) + uart_handle_cts_change(info, status & AMBA_UARTFR_CTS); + + wake_up_interruptible(&info->delta_msr_wait); +} +#endif + +static void clps711xuart_int_rx(int irq, void *dev_id, struct pt_regs *regs) +{ + struct uart_info *info = dev_id; + struct tty_struct *tty = info->tty; + unsigned int status, ch, flg, ignored = 0; + struct uart_port *port = info->port; + + status = clps_readl(SYSFLG(port)); + while (!(status & SYSFLG_URXFE)) { + ch = clps_readl(UARTDR(port)); + + if (tty->flip.count >= TTY_FLIPBUF_SIZE) + goto ignore_char; + port->icount.rx++; + + flg = TTY_NORMAL; + + /* + * Note that the error handling code is + * out of the main execution path + */ + if (ch & UART_ANY_ERR) + goto handle_error; + + if (uart_handle_sysrq_char(info, ch, regs)) + goto ignore_char; + + error_return: + *tty->flip.flag_buf_ptr++ = flg; + *tty->flip.char_buf_ptr++ = ch; + tty->flip.count++; + ignore_char: + status = clps_readl(SYSFLG(port)); + } +out: + tty_flip_buffer_push(tty); + return; + +handle_error: + if (ch & UARTDR_PARERR) + port->icount.parity++; + else if (ch & UARTDR_FRMERR) + port->icount.frame++; + if (ch & UARTDR_OVERR) + port->icount.overrun++; + + if (ch & port->ignore_status_mask) { + if (++ignored > 100) + goto out; + goto ignore_char; + } + ch &= port->read_status_mask; + + if (ch & UARTDR_PARERR) + flg = TTY_PARITY; + else if (ch & UARTDR_FRMERR) + flg = TTY_FRAME; + + if (ch & UARTDR_OVERR) { + /* + * CHECK: does overrun affect the current character? + * ASSUMPTION: it does not. + */ + *tty->flip.flag_buf_ptr++ = flg; + *tty->flip.char_buf_ptr++ = ch; + tty->flip.count++; + if (tty->flip.count >= TTY_FLIPBUF_SIZE) + goto ignore_char; + ch = 0; + flg = TTY_OVERRUN; + } +#ifdef SUPPORT_SYSRQ + info->sysrq = 0; +#endif + goto error_return; +} + +static void clps711xuart_int_tx(int irq, void *dev_id, struct pt_regs *regs) +{ + struct uart_info *info = dev_id; + struct uart_port *port = info->port; + int count; + + if (port->x_char) { + clps_writel(port->x_char, UARTDR(port)); + port->icount.tx++; + port->x_char = 0; + return; + } + if (info->xmit.head == info->xmit.tail + || info->tty->stopped + || info->tty->hw_stopped) { + clps711xuart_stop_tx(info->port, 0); + return; + } + + count = port->fifosize >> 1; + do { + clps_writel(info->xmit.buf[info->xmit.tail], UARTDR(port)); + info->xmit.tail = (info->xmit.tail + 1) & (UART_XMIT_SIZE - 1); + port->icount.tx++; + if (info->xmit.head == info->xmit.tail) + break; + } while (--count > 0); + + if (CIRC_CNT(info->xmit.head, + info->xmit.tail, + UART_XMIT_SIZE) < WAKEUP_CHARS) + uart_event(info, EVT_WRITE_WAKEUP); + + if (info->xmit.head == info->xmit.tail) + clps711xuart_stop_tx(info->port, 0); +} + +static u_int clps711xuart_tx_empty(struct uart_port *port) +{ + u_int status = clps_readl(SYSFLG(port)); + return status & SYSFLG_UBUSY ? 0 : TIOCSER_TEMT; +} + +static u_int clps711xuart_get_mctrl(struct uart_port *port) +{ + unsigned int port_addr; + unsigned int result = 0; + unsigned int status; + + port_addr = SYSFLG(port); + if (port_addr == SYSFLG1) { + status = clps_readl(SYSFLG1); + if (status & SYSFLG1_DCD) + result |= TIOCM_CAR; + if (status & SYSFLG1_DSR) + result |= TIOCM_DSR; + if (status & SYSFLG1_CTS) + result |= TIOCM_CTS; + } + + return result; +} + +static void clps711xuart_set_mctrl_null(struct uart_port *port, u_int mctrl) +{ +} + +static void clps711xuart_break_ctl(struct uart_port *port, int break_state) +{ + unsigned int ubrlcr; + + ubrlcr = clps_readl(UBRLCR(port)); + if (break_state == -1) + ubrlcr |= UBRLCR_BREAK; + else + ubrlcr &= ~UBRLCR_BREAK; + clps_writel(ubrlcr, UBRLCR(port)); +} + +static int clps711xuart_startup(struct uart_port *port, struct uart_info *info) +{ + u_int syscon; + int retval; + + tx_enabled(port) = 1; + + /* + * Allocate the IRQs + */ + retval = request_irq(TX_IRQ(port), clps711xuart_int_tx, 0, + "clps711xuart_tx", info); + if (retval) + return retval; + + retval = request_irq(RX_IRQ(port), clps711xuart_int_rx, 0, + "clps711xuart_rx", info); + if (retval) { + free_irq(TX_IRQ(port), info); + return retval; + } + + port->ops->set_mctrl(port, info->mctrl); + + /* + * enable the port + */ + syscon = clps_readl(SYSCON(port)); + syscon |= SYSCON_UARTEN; + clps_writel(syscon, SYSCON(port)); + + return 0; +} + +static void clps711xuart_shutdown(struct uart_port *port, struct uart_info *info) +{ + u_int ubrlcr, syscon; + + /* + * Free the interrupt + */ + free_irq(TX_IRQ(port), info); /* TX interrupt */ + free_irq(RX_IRQ(port), info); /* RX interrupt */ + + /* + * disable the port + */ + syscon = clps_readl(SYSCON(port)); + syscon &= ~SYSCON_UARTEN; + clps_writel(syscon, SYSCON(port)); + + /* + * disable break condition and fifos + */ + ubrlcr = clps_readl(UBRLCR(port)); + ubrlcr &= ~(UBRLCR_FIFOEN | UBRLCR_BREAK); + clps_writel(ubrlcr, UBRLCR(port)); +} + +static void clps711xuart_change_speed(struct uart_port *port, u_int cflag, u_int iflag, u_int quot) +{ + u_int ubrlcr; + unsigned long flags; + +#if DEBUG + printk("clps711xuart_change_speed(cflag=0x%x, iflag=0x%x, quot=%d) called\n", + cflag, iflag, quot); +#endif + /* byte size and parity */ + switch (cflag & CSIZE) { + case CS5: ubrlcr = UBRLCR_WRDLEN5; break; + case CS6: ubrlcr = UBRLCR_WRDLEN6; break; + case CS7: ubrlcr = UBRLCR_WRDLEN7; break; + default: ubrlcr = UBRLCR_WRDLEN8; break; // CS8 + } + if (cflag & CSTOPB) + ubrlcr |= UBRLCR_XSTOP; + if (cflag & PARENB) { + ubrlcr |= UBRLCR_PRTEN; + if (!(cflag & PARODD)) + ubrlcr |= UBRLCR_EVENPRT; + } + if (port->fifosize > 1) + ubrlcr |= UBRLCR_FIFOEN; + + port->read_status_mask = UARTDR_OVERR; + if (iflag & INPCK) + port->read_status_mask |= UARTDR_PARERR | UARTDR_FRMERR; +// if (iflag & (BRKINT | PARMRK)) +// port->read_status_mask |= AMBA_UARTRSR_BE; + + /* + * Characters to ignore + */ + port->ignore_status_mask = 0; + if (iflag & IGNPAR) + port->ignore_status_mask |= UARTDR_FRMERR | UARTDR_PARERR; + if (iflag & IGNBRK) { +// port->ignore_status_mask |= AMBA_UARTRSR_BE; + /* + * If we're ignoring parity and break indicators, + * ignore overruns to (for real raw support). + */ + if (iflag & IGNPAR) + port->ignore_status_mask |= UARTDR_OVERR; + } + + quot -= 1; + + /* first, disable everything */ + save_flags(flags); cli(); + + clps_writel(ubrlcr | quot, UBRLCR(port)); + + restore_flags(flags); +} + +static const char *clps711xuart_type(struct uart_port *port) +{ + return port->type == PORT_CLPS711X ? "CLPS711x" : NULL; +} + +/* + * Configure/autoconfigure the port. + */ +static void clps711xuart_config_port(struct uart_port *port, int flags) +{ + if (flags & UART_CONFIG_TYPE) + port->type = PORT_CLPS711X; +} + +static void clps711xuart_release_port(struct uart_port *port) +{ +} + +static int clps711xuart_request_port(struct uart_port *port) +{ + return 0; +} + +static struct uart_ops clps711x_pops = { + tx_empty: clps711xuart_tx_empty, + set_mctrl: clps711xuart_set_mctrl_null, + get_mctrl: clps711xuart_get_mctrl, + stop_tx: clps711xuart_stop_tx, + start_tx: clps711xuart_start_tx, + stop_rx: clps711xuart_stop_rx, + enable_ms: clps711xuart_enable_ms, + break_ctl: clps711xuart_break_ctl, + startup: clps711xuart_startup, + shutdown: clps711xuart_shutdown, + change_speed: clps711xuart_change_speed, + type: clps711xuart_type, + config_port: clps711xuart_config_port, + release_port: clps711xuart_release_port, + request_port: clps711xuart_request_port, +}; + +static struct uart_port clps711x_ports[UART_NR] = { + { + iobase: SYSCON1, + irq: IRQ_UTXINT1, /* IRQ_URXINT1, IRQ_UMSINT */ + uartclk: 3686400, + fifosize: 16, + ops: &clps711x_pops, + flags: ASYNC_BOOT_AUTOCONF, + }, + { + iobase: SYSCON2, + irq: IRQ_UTXINT2, /* IRQ_URXINT2 */ + uartclk: 3686400, + fifosize: 16, + ops: &clps711x_pops, + flags: ASYNC_BOOT_AUTOCONF, + } +}; + +#ifdef CONFIG_SERIAL_CLPS711X_CONSOLE +/* + * Print a string to the serial port trying not to disturb + * any possible real use of the port... + * + * The console_lock must be held when we get here. + * + * Note that this is called with interrupts already disabled + */ +static void clps711xuart_console_write(struct console *co, const char *s, u_int count) +{ + struct uart_port *port = clps711x_ports + co->index; + unsigned int status, syscon; + int i; + + /* + * Ensure that the port is enabled. + */ + syscon = clps_readl(SYSCON(port)); + clps_writel(syscon | SYSCON_UARTEN, SYSCON(port)); + + /* + * Now, do each character + */ + for (i = 0; i < count; i++) { + do { + status = clps_readl(SYSFLG(port)); + } while (status & SYSFLG_UTXFF); + clps_writel(s[i], UARTDR(port)); + if (s[i] == '\n') { + do { + status = clps_readl(SYSFLG(port)); + } while (status & SYSFLG_UTXFF); + clps_writel('\r', UARTDR(port)); + } + } + + /* + * Finally, wait for transmitter to become empty + * and restore the uart state. + */ + do { + status = clps_readl(SYSFLG(port)); + } while (status & SYSFLG_UBUSY); + + clps_writel(syscon, SYSCON(port)); +} + +static kdev_t clps711xuart_console_device(struct console *co) +{ + return MKDEV(SERIAL_CLPS711X_MAJOR, SERIAL_CLPS711X_MINOR + co->index); +} + +static void __init +clps711xuart_console_get_options(struct uart_port *port, int *baud, int *parity, int *bits) +{ + if (clps_readl(SYSCON(port)) & SYSCON_UARTEN) { + u_int ubrlcr, quot; + + ubrlcr = clps_readl(UBRLCR(port)); + + *parity = 'n'; + if (ubrlcr & UBRLCR_PRTEN) { + if (ubrlcr & UBRLCR_EVENPRT) + *parity = 'e'; + else + *parity = 'o'; + } + + if ((ubrlcr & UBRLCR_WRDLEN_MASK) == UBRLCR_WRDLEN7) + *bits = 7; + else + *bits = 8; + + quot = ubrlcr & UBRLCR_BAUD_MASK; + *baud = port->uartclk / (16 * (quot + 1)); + } +} + +static int __init clps711xuart_console_setup(struct console *co, char *options) +{ + struct uart_port *port; + int baud = 38400; + int bits = 8; + int parity = 'n'; + int flow = 'n'; + + /* + * Check whether an invalid uart number has been specified, and + * if so, search for the first available port that does have + * console support. + */ + port = uart_get_console(clps711x_ports, UART_NR, co); + + if (options) + uart_parse_options(options, &baud, &parity, &bits, &flow); + else + clps711xuart_console_get_options(port, &baud, &parity, &bits); + + return uart_set_options(port, co, baud, parity, bits, flow); +} + +static struct console clps711x_console = { + name: SERIAL_CLPS711X_NAME, + write: clps711xuart_console_write, + device: clps711xuart_console_device, + setup: clps711xuart_console_setup, + flags: CON_PRINTBUFFER, + index: -1, +}; + +void __init clps711xuart_console_init(void) +{ + register_console(&clps711x_console); +} + +#define CLPS711X_CONSOLE &clps711x_console +#else +#define CLPS711X_CONSOLE NULL +#endif + +static struct uart_driver clps711x_reg = { +#ifdef CONFIG_DEVFS_FS + normal_name: SERIAL_CLPS711X_NAME, + callout_name: CALLOUT_CLPS711X_NAME, +#else + normal_name: SERIAL_CLPS711X_NAME, + callout_name: CALLOUT_CLPS711X_NAME, +#endif + + normal_major: SERIAL_CLPS711X_MAJOR, + normal_driver: &normal, + callout_major: CALLOUT_CLPS711X_MAJOR, + callout_driver: &callout, + + table: clps711x_table, + termios: clps711x_termios, + termios_locked: clps711x_termios_locked, + + minor: SERIAL_CLPS711X_MINOR, + nr: UART_NR, + + port: clps711x_ports, + cons: CLPS711X_CONSOLE, +}; + +static int __init clps711xuart_init(void) +{ + return uart_register_driver(&clps711x_reg); +} + +static void __exit clps711xuart_exit(void) +{ + uart_unregister_driver(&clps711x_reg); +} + +module_init(clps711xuart_init); +module_exit(clps711xuart_exit); + +EXPORT_NO_SYMBOLS; + +MODULE_AUTHOR("Deep Blue Solutions Ltd"); +MODULE_DESCRIPTION("CLPS-711x generic serial driver"); +MODULE_LICENSE("GPL"); diff -urN orig/drivers/serial/core.c linux/drivers/serial/core.c --- orig/drivers/serial/core.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/serial/core.c Thu Feb 27 23:21:11 2003 @@ -0,0 +1,2248 @@ +/* + * linux/drivers/char/serial_core.c + * + * Driver core for serial ports + * + * Based on drivers/char/serial.c, by Linus Torvalds, Theodore Ts'o. + * + * Copyright 1999 ARM Limited + * Copyright (C) 2000-2001 Deep Blue Solutions Ltd. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + * + * $Id: core.c,v 1.20.2.5 2002/03/13 15:22:26 rmk Exp $ + * + */ +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include + +#undef DEBUG + +#ifndef CONFIG_PM +#define pm_access(pm) do { } while (0) +#define pm_unregister(pm) do { } while (0) +#endif + +/* + * tmp_buf is used as a temporary buffer by serial_write. We need to + * lock it in case the copy_from_user blocks while swapping in a page, + * and some other program tries to do a serial write at the same time. + * Since the lock will only come under contention when the system is + * swapping and available memory is low, it makes sense to share one + * buffer across all the serial ports, since it significantly saves + * memory if large numbers of serial ports are open. + */ +static u_char *tmp_buf; +static DECLARE_MUTEX(tmp_buf_sem); + +/* + * This is used to lock changes in serial line configuration. + */ +static DECLARE_MUTEX(port_sem); + +#define HIGH_BITS_OFFSET ((sizeof(long)-sizeof(int))*8) + +static void uart_change_speed(struct uart_info *info, struct termios *old_termios); +static void uart_wait_until_sent(struct tty_struct *tty, int timeout); + +/* + * This routine is used by the interrupt handler to schedule processing in + * the software interrupt portion of the driver. It is expected that + * interrupts will be disabled (and so the tasklet will be prevented + * from running (CHECK)). + */ +void uart_event(struct uart_info *info, int event) +{ + info->event |= 1 << event; + tasklet_schedule(&info->tlet); +} + +static void uart_stop(struct tty_struct *tty) +{ + struct uart_info *info = tty->driver_data; + unsigned long flags; + + spin_lock_irqsave(&info->lock, flags); + info->ops->stop_tx(info->port, 1); + spin_unlock_irqrestore(&info->lock, flags); +} + +static void __uart_start(struct tty_struct *tty) +{ + struct uart_info *info = tty->driver_data; + if (info->xmit.head != info->xmit.tail && info->xmit.buf && + !tty->stopped && !tty->hw_stopped) + info->ops->start_tx(info->port, 1, 1); +} + +static void uart_start(struct tty_struct *tty) +{ + struct uart_info *info = tty->driver_data; + unsigned long flags; + + pm_access(info->state->pm); + + spin_lock_irqsave(&info->lock, flags); + __uart_start(tty); + spin_unlock_irqrestore(&info->lock, flags); +} + +static void uart_tasklet_action(unsigned long data) +{ + struct uart_info *info = (struct uart_info *)data; + struct tty_struct *tty; + + tty = info->tty; + if (!tty || !test_and_clear_bit(EVT_WRITE_WAKEUP, &info->event)) + return; + + if ((tty->flags & (1 << TTY_DO_WRITE_WAKEUP)) && + tty->ldisc.write_wakeup) + (tty->ldisc.write_wakeup)(tty); + wake_up_interruptible(&tty->write_wait); +} + +static inline void uart_update_altspeed(struct uart_info *info) +{ + if ((info->flags & ASYNC_SPD_MASK) == ASYNC_SPD_HI) + info->tty->alt_speed = 57600; + if ((info->flags & ASYNC_SPD_MASK) == ASYNC_SPD_VHI) + info->tty->alt_speed = 115200; + if ((info->flags & ASYNC_SPD_MASK) == ASYNC_SPD_SHI) + info->tty->alt_speed = 230400; + if ((info->flags & ASYNC_SPD_MASK) == ASYNC_SPD_WARP) + info->tty->alt_speed = 460800; +} + +static int uart_startup(struct uart_info *info) +{ + unsigned long flags; + unsigned long page; + int retval = 0; + + page = get_zeroed_page(GFP_KERNEL); + if (!page) + return -ENOMEM; + + save_flags(flags); cli(); + + if (info->flags & ASYNC_INITIALIZED) { + free_page(page); + goto errout; + } + + if (info->port->type == PORT_UNKNOWN) { + if (info->tty) + set_bit(TTY_IO_ERROR, &info->tty->flags); + free_page(page); + goto errout; + } + + if (info->xmit.buf) + free_page(page); + else + info->xmit.buf = (unsigned char *) page; + + info->mctrl = 0; + + retval = info->ops->startup(info->port, info); + if (retval) { + if (capable(CAP_SYS_ADMIN)) { + if (info->tty) + set_bit(TTY_IO_ERROR, &info->tty->flags); + retval = 0; + } + goto errout; + } + + if (info->tty) + clear_bit(TTY_IO_ERROR, &info->tty->flags); + info->xmit.head = info->xmit.tail = 0; + + /* + * Set up the tty->alt_speed kludge + */ + if (info->tty) + uart_update_altspeed(info); + + /* + * and set the speed of the serial port + */ + uart_change_speed(info, NULL); + + /* + * Setup the RTS and DTR signals once the port + * is open and ready to respond. + */ + if (info->tty->termios->c_cflag & CBAUD) + info->mctrl |= TIOCM_RTS | TIOCM_DTR; + info->ops->set_mctrl(info->port, info->mctrl); + + info->flags |= ASYNC_INITIALIZED; + retval = 0; + +errout: + restore_flags(flags); + return retval; +} + +/* + * This routine will shutdown a serial port; interrupts are disabled, and + * DTR is dropped if the hangup on close termio flag is on. + */ +static void uart_shutdown(struct uart_info *info) +{ + unsigned long flags; + + if (!(info->flags & ASYNC_INITIALIZED)) + return; + + save_flags(flags); cli(); /* Disable interrupts */ + + /* + * clear delta_msr_wait queue to avoid mem leaks: we may free the irq + * here so the queue might never be woken up + */ + wake_up_interruptible(&info->delta_msr_wait); + + /* + * Free the IRQ and disable the port + */ + info->ops->shutdown(info->port, info); + + if (info->xmit.buf) { + unsigned long pg = (unsigned long) info->xmit.buf; + info->xmit.buf = NULL; + free_page(pg); + } + + if (!info->tty || (info->tty->termios->c_cflag & HUPCL)) + info->mctrl &= ~(TIOCM_DTR|TIOCM_RTS); + info->ops->set_mctrl(info->port, info->mctrl); + + /* kill off our tasklet */ + tasklet_kill(&info->tlet); + if (info->tty) + set_bit(TTY_IO_ERROR, &info->tty->flags); + + info->flags &= ~ASYNC_INITIALIZED; + restore_flags(flags); +} + +static inline u_int uart_calculate_quot(struct uart_info *info, u_int baud) +{ + u_int quot; + + /* Special case: B0 rate */ + if (!baud) + baud = 9600; + + /* Old HI/VHI/custom speed handling */ + if (baud == 38400 && + ((info->flags & ASYNC_SPD_MASK) == ASYNC_SPD_CUST)) + quot = info->state->custom_divisor; + else + quot = info->port->uartclk / (16 * baud); + + return quot; +} + +static void uart_change_speed(struct uart_info *info, struct termios *old_termios) +{ + struct uart_port *port = info->port; + u_int quot, baud, cflag, bits, try; + + /* + * If we have no tty, termios, or the port does not exist, + * then we can't set the parameters for this port. + */ + if (!info->tty || !info->tty->termios || + info->port->type == PORT_UNKNOWN) + return; + + cflag = info->tty->termios->c_cflag; + + /* byte size and parity */ + switch (cflag & CSIZE) { + case CS5: bits = 7; break; + case CS6: bits = 8; break; + case CS7: bits = 9; break; + default: bits = 10; break; // CS8 + } + + if (cflag & CSTOPB) + bits++; + if (cflag & PARENB) + bits++; + + for (try = 0; try < 2; try ++) { + /* Determine divisor based on baud rate */ + baud = tty_get_baud_rate(info->tty); + quot = uart_calculate_quot(info, baud); + if (quot) + break; + + /* + * Oops, the quotient was zero. Try again with + * the old baud rate if possible. + */ + info->tty->termios->c_cflag &= ~CBAUD; + if (old_termios) { + info->tty->termios->c_cflag |= + (old_termios->c_cflag & CBAUD); + old_termios = NULL; + continue; + } + + /* + * As a last resort, if the quotient is zero, + * default to 9600 bps + */ + info->tty->termios->c_cflag |= B9600; + } + + info->timeout = (port->fifosize * HZ * bits * quot) / + (port->uartclk / 16); + info->timeout += HZ/50; /* Add .02 seconds of slop */ + + if (cflag & CRTSCTS) + info->flags |= ASYNC_CTS_FLOW; + else + info->flags &= ~ASYNC_CTS_FLOW; + if (cflag & CLOCAL) + info->flags &= ~ASYNC_CHECK_CD; + else + info->flags |= ASYNC_CHECK_CD; + + /* + * Set up parity check flag + */ +#define RELEVENT_IFLAG(iflag) ((iflag) & (IGNBRK|BRKINT|IGNPAR|PARMRK|INPCK)) + + pm_access(info->state->pm); + + info->ops->change_speed(port, cflag, info->tty->termios->c_iflag, quot); +} + +static void uart_put_char(struct tty_struct *tty, u_char ch) +{ + struct uart_info *info = tty->driver_data; + unsigned long flags; + + if (!tty || !info->xmit.buf) + return; + + spin_lock_irqsave(&info->lock, flags); + if (CIRC_SPACE(info->xmit.head, info->xmit.tail, UART_XMIT_SIZE) != 0) { + info->xmit.buf[info->xmit.head] = ch; + info->xmit.head = (info->xmit.head + 1) & (UART_XMIT_SIZE - 1); + } + spin_unlock_irqrestore(&info->lock, flags); +} + +static void uart_flush_chars(struct tty_struct *tty) +{ + uart_start(tty); +} + +static int uart_write(struct tty_struct *tty, int from_user, + const u_char * buf, int count) +{ + struct uart_info *info = tty->driver_data; + unsigned long flags; + int c, ret = 0; + + if (!tty || !info->xmit.buf || !tmp_buf) + return 0; + + if (from_user) { + down(&tmp_buf_sem); + while (1) { + int c1; + c = CIRC_SPACE_TO_END(info->xmit.head, + info->xmit.tail, + UART_XMIT_SIZE); + if (count < c) + c = count; + if (c <= 0) + break; + + c -= copy_from_user(tmp_buf, buf, c); + if (!c) { + if (!ret) + ret = -EFAULT; + break; + } + spin_lock_irqsave(&info->lock, flags); + c1 = CIRC_SPACE_TO_END(info->xmit.head, + info->xmit.tail, + UART_XMIT_SIZE); + if (c1 < c) + c = c1; + memcpy(info->xmit.buf + info->xmit.head, tmp_buf, c); + info->xmit.head = (info->xmit.head + c) & + (UART_XMIT_SIZE - 1); + spin_unlock_irqrestore(&info->lock, flags); + buf += c; + count -= c; + ret += c; + } + up(&tmp_buf_sem); + } else { + spin_lock_irqsave(&info->lock, flags); + while (1) { + c = CIRC_SPACE_TO_END(info->xmit.head, + info->xmit.tail, + UART_XMIT_SIZE); + if (count < c) + c = count; + if (c <= 0) + break; + memcpy(info->xmit.buf + info->xmit.head, buf, c); + info->xmit.head = (info->xmit.head + c) & + (UART_XMIT_SIZE - 1); + buf += c; + count -= c; + ret += c; + } + spin_unlock_irqrestore(&info->lock, flags); + } + + uart_start(tty); + return ret; +} + +static int uart_write_room(struct tty_struct *tty) +{ + struct uart_info *info = tty->driver_data; + + return CIRC_SPACE(info->xmit.head, info->xmit.tail, UART_XMIT_SIZE); +} + +static int uart_chars_in_buffer(struct tty_struct *tty) +{ + struct uart_info *info = tty->driver_data; + + return CIRC_CNT(info->xmit.head, info->xmit.tail, UART_XMIT_SIZE); +} + +static void uart_flush_buffer(struct tty_struct *tty) +{ + struct uart_info *info = tty->driver_data; + unsigned long flags; + +#ifdef DEBUG + printk("uart_flush_buffer(%d) called\n", + MINOR(tty->device) - tty->driver.minor_start); +#endif + spin_lock_irqsave(&info->lock, flags); + info->xmit.head = info->xmit.tail = 0; + spin_unlock_irqrestore(&info->lock, flags); + wake_up_interruptible(&tty->write_wait); + if ((tty->flags & (1 << TTY_DO_WRITE_WAKEUP)) && + tty->ldisc.write_wakeup) + (tty->ldisc.write_wakeup)(tty); +} + +/* + * This function is used to send a high-priority XON/XOFF character to + * the device + */ +static void uart_send_xchar(struct tty_struct *tty, char ch) +{ + struct uart_info *info = tty->driver_data; + + info->port->x_char = ch; + if (ch) + info->ops->start_tx(info->port, 1, 0); +} + +static void uart_throttle(struct tty_struct *tty) +{ + struct uart_info *info = tty->driver_data; + unsigned long flags; + + if (I_IXOFF(tty)) + uart_send_xchar(tty, STOP_CHAR(tty)); + + if (tty->termios->c_cflag & CRTSCTS) { + spin_lock_irqsave(&info->lock, flags); + info->mctrl &= ~TIOCM_RTS; + info->ops->set_mctrl(info->port, info->mctrl); + spin_unlock_irqrestore(&info->lock, flags); + } +} + +static void uart_unthrottle(struct tty_struct *tty) +{ + struct uart_info *info = (struct uart_info *) tty->driver_data; + unsigned long flags; + + if (I_IXOFF(tty)) { + if (info->port->x_char) + info->port->x_char = 0; + else + uart_send_xchar(tty, START_CHAR(tty)); + } + + if (tty->termios->c_cflag & CRTSCTS) { + spin_lock_irqsave(&info->lock, flags); + info->mctrl |= TIOCM_RTS; + info->ops->set_mctrl(info->port, info->mctrl); + spin_unlock_irqrestore(&info->lock, flags); + } +} + +static int uart_get_info(struct uart_info *info, struct serial_struct *retinfo) +{ + struct uart_state *state = info->state; + struct uart_port *port = info->port; + struct serial_struct tmp; + + memset(&tmp, 0, sizeof(tmp)); + tmp.type = port->type; + tmp.line = port->line; + tmp.port = port->iobase; + if (HIGH_BITS_OFFSET) + tmp.port_high = port->iobase >> HIGH_BITS_OFFSET; + tmp.irq = port->irq; + tmp.flags = port->flags; + tmp.xmit_fifo_size = port->fifosize; + tmp.baud_base = port->uartclk / 16; + tmp.close_delay = state->close_delay; + tmp.closing_wait = state->closing_wait; + tmp.custom_divisor = state->custom_divisor; + tmp.hub6 = port->hub6; + tmp.io_type = port->iotype; + tmp.iomem_reg_shift= port->regshift; + tmp.iomem_base = (void *)port->mapbase; + + if (copy_to_user(retinfo, &tmp, sizeof(*retinfo))) + return -EFAULT; + return 0; +} + +static int uart_set_info(struct uart_info *info, + struct serial_struct *newinfo) +{ + struct serial_struct new_serial; + struct uart_state *state = info->state; + struct uart_port *port = info->port; + unsigned long new_port; + unsigned int change_irq, change_port, old_flags; + unsigned int old_custom_divisor; + int retval = 0; + + if (copy_from_user(&new_serial, newinfo, sizeof(new_serial))) + return -EFAULT; + + new_port = new_serial.port; + if (HIGH_BITS_OFFSET) + new_port += (unsigned long) new_serial.port_high << HIGH_BITS_OFFSET; + + new_serial.irq = irq_cannonicalize(new_serial.irq); + + /* + * This semaphore protects state->count. It is also + * very useful to prevent opens. Also, take the + * port configuration semaphore to make sure that a + * module insertion/removal doesn't change anything + * under us. + */ + down(&port_sem); + down(&state->count_sem); + + change_irq = new_serial.irq != port->irq; + + /* + * Since changing the 'type' of the port changes its resource + * allocations, we should treat type changes the same as + * IO port changes. + */ + change_port = new_port != port->iobase || + (unsigned long)new_serial.iomem_base != port->mapbase || + new_serial.hub6 != port->hub6 || + new_serial.io_type != port->iotype || + new_serial.iomem_reg_shift != port->regshift || + new_serial.type != port->type; + + old_flags = port->flags; + old_custom_divisor = state->custom_divisor; + + if (!capable(CAP_SYS_ADMIN)) { + retval = -EPERM; + if (change_irq || change_port || + (new_serial.baud_base != port->uartclk / 16) || + (new_serial.close_delay != state->close_delay) || + (new_serial.closing_wait != state->closing_wait) || + (new_serial.xmit_fifo_size != port->fifosize) || + ((new_serial.flags & ~ASYNC_USR_MASK) != + (port->flags & ~ASYNC_USR_MASK))) + goto exit; + port->flags = ((port->flags & ~ASYNC_USR_MASK) | + (new_serial.flags & ASYNC_USR_MASK)); + info->flags = ((info->flags & ~ASYNC_USR_MASK) | + (new_serial.flags & ASYNC_USR_MASK)); + state->custom_divisor = new_serial.custom_divisor; + goto check_and_exit; + } + + /* + * Ask the low level driver to verify the settings. + */ + if (port->ops->verify_port) + retval = port->ops->verify_port(port, &new_serial); + + if ((new_serial.irq >= NR_IRQS) || (new_serial.irq < 0) || + (new_serial.baud_base < 9600)) + retval = -EINVAL; + + if (retval) + goto exit; + + if (change_port || change_irq) { + retval = -EBUSY; + + /* + * Make sure that we are the sole user of this port. + */ + if (state->count > 1) + goto exit; + + /* + * We need to shutdown the serial port at the old + * port/type/irq combination. + */ + uart_shutdown(info); + } + + if (change_port) { + unsigned long old_iobase, old_mapbase; + unsigned int old_type, old_iotype, old_hub6, old_shift; + + old_iobase = port->iobase; + old_mapbase = port->mapbase; + old_type = port->type; + old_hub6 = port->hub6; + old_iotype = port->iotype; + old_shift = port->regshift; + + /* + * Free and release old regions + */ + if (old_type != PORT_UNKNOWN) + port->ops->release_port(port); + + port->iobase = new_port; + port->type = new_serial.type; + port->hub6 = new_serial.hub6; + port->iotype = new_serial.io_type; + port->regshift = new_serial.iomem_reg_shift; + port->mapbase = (unsigned long)new_serial.iomem_base; + + /* + * Claim and map the new regions + */ + if (port->type != PORT_UNKNOWN) + retval = port->ops->request_port(port); + + /* + * If we fail to request resources for the + * new port, try to restore the old settings. + */ + if (retval && old_type != PORT_UNKNOWN) { + port->iobase = old_iobase; + port->type = old_type; + port->hub6 = old_hub6; + port->iotype = old_iotype; + port->regshift = old_shift; + port->mapbase = old_mapbase; + retval = port->ops->request_port(port); + /* + * If we failed to restore the old settings, + * we fail like this. + */ + if (retval) + port->type = PORT_UNKNOWN; + + /* + * We failed anyway. + */ + retval = -EBUSY; + } + } + + port->irq = new_serial.irq; + port->uartclk = new_serial.baud_base * 16; + port->flags = ((port->flags & ~ASYNC_FLAGS) | + (new_serial.flags & ASYNC_FLAGS)); + info->flags = ((port->flags & ~ASYNC_INTERNAL_FLAGS) | + (info->flags & ASYNC_INTERNAL_FLAGS)); + state->custom_divisor = new_serial.custom_divisor; + state->close_delay = new_serial.close_delay * HZ / 100; + state->closing_wait = new_serial.closing_wait * HZ / 100; + info->tty->low_latency = (info->flags & ASYNC_LOW_LATENCY) ? 1 : 0; + port->fifosize = new_serial.xmit_fifo_size; + +check_and_exit: + retval = 0; + if (port->type == PORT_UNKNOWN) + goto exit; + if (info->flags & ASYNC_INITIALIZED) { + if (((old_flags & info->flags) & ASYNC_SPD_MASK) || + old_custom_divisor != state->custom_divisor) { + uart_update_altspeed(info); + uart_change_speed(info, NULL); + } + } else + retval = uart_startup(info); +exit: + up(&state->count_sem); + up(&port_sem); + return retval; +} + + +/* + * uart_get_lsr_info - get line status register info + */ +static int uart_get_lsr_info(struct uart_info *info, unsigned int *value) +{ + u_int result; + unsigned long flags; + + spin_lock_irqsave(&info->lock, flags); + result = info->ops->tx_empty(info->port); + spin_unlock_irqrestore(&info->lock, flags); + + /* + * If we're about to load something into the transmit + * register, we'll pretend the transmitter isn't empty to + * avoid a race condition (depending on when the transmit + * interrupt happens). + */ + if (info->port->x_char || + ((CIRC_CNT(info->xmit.head, info->xmit.tail, + UART_XMIT_SIZE) > 0) && + !info->tty->stopped && !info->tty->hw_stopped)) + result &= ~TIOCSER_TEMT; + + return put_user(result, value); +} + +static int uart_get_modem_info(struct uart_info *info, unsigned int *value) +{ + unsigned int result = info->mctrl; + + result &= TIOCM_DTR|TIOCM_RTS|TIOCM_OUT1|TIOCM_OUT2; + result |= info->ops->get_mctrl(info->port); + + return put_user(result, value); +} + +static int uart_set_modem_info(struct uart_info *info, unsigned int cmd, + unsigned int *value) +{ + unsigned int arg, old; + int ret = 0; + + if (get_user(arg, value)) + return -EFAULT; + + spin_lock_irq(&info->lock); + old = info->mctrl; + switch (cmd) { + case TIOCMBIS: info->mctrl |= arg; break; + case TIOCMBIC: info->mctrl &= ~arg; break; + case TIOCMSET: info->mctrl = arg; break; + default: ret = -EINVAL; break; + } + if (old != info->mctrl) + info->ops->set_mctrl(info->port, info->mctrl); + spin_unlock_irq(&info->lock); + return ret; +} + +static void uart_break_ctl(struct tty_struct *tty, int break_state) +{ + struct uart_info *info = tty->driver_data; + unsigned long flags; + + if (info->port->type != PORT_UNKNOWN) { + spin_lock_irqsave(&info->lock, flags); + info->ops->break_ctl(info->port, break_state); + spin_unlock_irqrestore(&info->lock, flags); + } +} + +static int uart_do_autoconfig(struct uart_info *info) +{ + struct uart_port *port = info->port; + int flags, ret; + + if (!capable(CAP_SYS_ADMIN)) + return -EPERM; + + /* + * Take the 'count' lock. This prevents count + * from incrementing, and hence any extra opens + * of the port while we're auto-configging. + */ + down(&info->state->count_sem); + + ret = -EBUSY; + if (info->state->count == 1) { + uart_shutdown(info); + + /* + * If we already have a port type configured, + * we must release its resources. + */ + if (port->type != PORT_UNKNOWN) + port->ops->release_port(port); + + flags = UART_CONFIG_TYPE; + if (port->flags & ASYNC_AUTO_IRQ) + flags |= UART_CONFIG_IRQ; + + /* + * This will claim the ports resources if + * a port is found. + */ + port->ops->config_port(port, flags); + + ret = uart_startup(info); + } + up(&info->state->count_sem); + return ret; +} + +/* + * Called from userspace. We can use spin_lock_irq() here. + */ +static int uart_ioctl(struct tty_struct *tty, struct file *file, + unsigned int cmd, unsigned long arg) +{ + struct uart_info *info = tty->driver_data; + struct uart_icount cprev, cnow; + struct serial_icounter_struct icount; + int ret = -ENOIOCTLCMD; + + if ((cmd != TIOCGSERIAL) && (cmd != TIOCSSERIAL) && + (cmd != TIOCSERCONFIG) && (cmd != TIOCSERGSTRUCT) && + (cmd != TIOCMIWAIT) && (cmd != TIOCGICOUNT)) { + if (tty->flags & (1 << TTY_IO_ERROR)) + return -EIO; + } + + switch (cmd) { + case TIOCMGET: + ret = uart_get_modem_info(info, (unsigned int *)arg); + break; + + case TIOCMBIS: + case TIOCMBIC: + case TIOCMSET: + ret = uart_set_modem_info(info, cmd, + (unsigned int *)arg); + break; + + case TIOCGSERIAL: + ret = uart_get_info(info, (struct serial_struct *)arg); + break; + + case TIOCSSERIAL: + ret = uart_set_info(info, (struct serial_struct *)arg); + break; + + case TIOCSERCONFIG: + ret = uart_do_autoconfig(info); + break; + + case TIOCSERGETLSR: /* Get line status register */ + ret = uart_get_lsr_info(info, (unsigned int *)arg); + break; + + /* + * Wait for any of the 4 modem inputs (DCD,RI,DSR,CTS) to change + * - mask passed in arg for lines of interest + * (use |'ed TIOCM_RNG/DSR/CD/CTS for masking) + * Caller should use TIOCGICOUNT to see which one it was + */ + case TIOCMIWAIT: + spin_lock_irq(&info->lock); + /* note the counters on entry */ + cprev = info->port->icount; + /* Force modem status interrupts on */ + info->ops->enable_ms(info->port); + spin_unlock_irq(&info->lock); + while (1) { + interruptible_sleep_on(&info->delta_msr_wait); + /* see if a signal did it */ + if (signal_pending(current)) { + ret = -ERESTARTSYS; + break; + } + spin_lock_irq(&info->lock); + cnow = info->port->icount; /* atomic copy */ + spin_unlock_irq(&info->lock); + if (cnow.rng == cprev.rng && cnow.dsr == cprev.dsr && + cnow.dcd == cprev.dcd && cnow.cts == cprev.cts) { + ret = -EIO; /* no change => error */ + break; + } + if (((arg & TIOCM_RNG) && (cnow.rng != cprev.rng)) || + ((arg & TIOCM_DSR) && (cnow.dsr != cprev.dsr)) || + ((arg & TIOCM_CD) && (cnow.dcd != cprev.dcd)) || + ((arg & TIOCM_CTS) && (cnow.cts != cprev.cts))) { + ret = 0; + break; + } + cprev = cnow; + } + break; + + /* + * Get counter of input serial line interrupts (DCD,RI,DSR,CTS) + * Return: write counters to the user passed counter struct + * NB: both 1->0 and 0->1 transitions are counted except for + * RI where only 0->1 is counted. + */ + case TIOCGICOUNT: + spin_lock_irq(&info->lock); + cnow = info->port->icount; + spin_unlock_irq(&info->lock); + + icount.cts = cnow.cts; + icount.dsr = cnow.dsr; + icount.rng = cnow.rng; + icount.dcd = cnow.dcd; + icount.rx = cnow.rx; + icount.tx = cnow.tx; + icount.frame = cnow.frame; + icount.overrun = cnow.overrun; + icount.parity = cnow.parity; + icount.brk = cnow.brk; + icount.buf_overrun = cnow.buf_overrun; + + ret = copy_to_user((void *)arg, &icount, sizeof(icount)) + ? -EFAULT : 0; + break; + + case TIOCSERGWILD: /* obsolete */ + case TIOCSERSWILD: /* obsolete */ + ret = 0; + break; + + default: + if (info->ops->ioctl) + ret = info->ops->ioctl(info->port, cmd, arg); + break; + } + return ret; +} + +static void uart_set_termios(struct tty_struct *tty, struct termios *old_termios) +{ + struct uart_info *info = tty->driver_data; + unsigned long flags; + unsigned int cflag = tty->termios->c_cflag; + + if ((cflag ^ old_termios->c_cflag) == 0 && + RELEVENT_IFLAG(tty->termios->c_iflag ^ old_termios->c_iflag) == 0) + return; + + uart_change_speed(info, old_termios); + + spin_lock_irqsave(&info->lock, flags); + + /* Handle transition to B0 status */ + if ((old_termios->c_cflag & CBAUD) && !(cflag & CBAUD)) { + info->mctrl &= ~(TIOCM_RTS | TIOCM_DTR); + info->ops->set_mctrl(info->port, info->mctrl); + } + + /* Handle transition away from B0 status */ + if (!(old_termios->c_cflag & CBAUD) && (cflag & CBAUD)) { + info->mctrl |= TIOCM_DTR; + if (!(cflag & CRTSCTS) || + !test_bit(TTY_THROTTLED, &tty->flags)) + info->mctrl |= TIOCM_RTS; + info->ops->set_mctrl(info->port, info->mctrl); + } + + /* Handle turning off CRTSCTS */ + if ((old_termios->c_cflag & CRTSCTS) && !(cflag & CRTSCTS)) { + tty->hw_stopped = 0; + __uart_start(tty); + } + spin_unlock_irqrestore(&info->lock, flags); + +#if 0 + /* + * No need to wake up processes in open wait, since they + * sample the CLOCAL flag once, and don't recheck it. + * XXX It's not clear whether the current behavior is correct + * or not. Hence, this may change..... + */ + if (!(old_termios->c_cflag & CLOCAL) && + (tty->termios->c_cflag & CLOCAL)) + wake_up_interruptible(&info->open_wait); +#endif +} + +/* + * In 2.4.5, calls to this will be serialized via the BKL in + * linux/drivers/char/tty_io.c:tty_release() + * linux/drivers/char/tty_io.c:do_tty_handup() + */ +static void uart_close(struct tty_struct *tty, struct file *filp) +{ + struct uart_driver *drv = (struct uart_driver *)tty->driver.driver_state; + struct uart_info *info = tty->driver_data; + struct uart_state *state; + unsigned long flags; + + if (!info) + return; + + state = info->state; + +#ifdef DEBUG + printk("uart_close() called\n"); +#endif + + /* + * This is safe, as long as the BKL exists in + * do_tty_hangup(), and we're protected by the BKL. + */ + if (tty_hung_up_p(filp)) + goto done; + + down(&state->count_sem); + spin_lock_irqsave(&info->lock, flags); + if ((tty->count == 1) && (state->count != 1)) { + /* + * Uh, oh. tty->count is 1, which means that the tty + * structure will be freed. state->count should always + * be one in these conditions. If it's greater than + * one, we've got real problems, since it means the + * serial port won't be shutdown. + */ + printk("uart_close: bad serial port count; tty->count is 1, " + "state->count is %d\n", state->count); + state->count = 1; + } + if (--state->count < 0) { + printk("rs_close: bad serial port count for %s%d: %d\n", + tty->driver.name, info->port->line, state->count); + state->count = 0; + } + if (state->count) { + spin_unlock_irqrestore(&info->lock, flags); + up(&state->count_sem); + goto done; + } + info->flags |= ASYNC_CLOSING; + spin_unlock_irqrestore(&info->lock, flags); + up(&state->count_sem); + + /* + * Save the termios structure, since this port may have + * separate termios for callout and dialin. + */ + if (info->flags & ASYNC_NORMAL_ACTIVE) + info->state->normal_termios = *tty->termios; + if (info->flags & ASYNC_CALLOUT_ACTIVE) + info->state->callout_termios = *tty->termios; + /* + * Now we wait for the transmit buffer to clear; and we notify + * the line discipline to only process XON/XOFF characters. + */ + tty->closing = 1; + if (info->state->closing_wait != ASYNC_CLOSING_WAIT_NONE) + tty_wait_until_sent(tty, info->state->closing_wait); + /* + * At this point, we stop accepting input. To do this, we + * disable the receive line status interrupts. + */ + if (info->flags & ASYNC_INITIALIZED) { + info->ops->stop_rx(info->port); + /* + * Before we drop DTR, make sure the UART transmitter + * has completely drained; this is especially + * important if there is a transmit FIFO! + */ + uart_wait_until_sent(tty, info->timeout); + } + uart_shutdown(info); + if (tty->driver.flush_buffer) + tty->driver.flush_buffer(tty); + if (tty->ldisc.flush_buffer) + tty->ldisc.flush_buffer(tty); + tty->closing = 0; + info->event = 0; + info->tty = NULL; + if (info->blocked_open) { + if (info->state->close_delay) { + set_current_state(TASK_INTERRUPTIBLE); + schedule_timeout(info->state->close_delay); + set_current_state(TASK_RUNNING); + } + wake_up_interruptible(&info->open_wait); + } else { +#ifdef CONFIG_PM + /* + * Put device into D3 state. + */ + pm_send(info->state->pm, PM_SUSPEND, (void *)3); +#else + if (info->ops->pm) + info->ops->pm(info->port, 3, 0); +#endif + } + + info->flags &= ~(ASYNC_NORMAL_ACTIVE|ASYNC_CALLOUT_ACTIVE| + ASYNC_CLOSING); + wake_up_interruptible(&info->close_wait); + +done: + if (drv->owner) + __MOD_DEC_USE_COUNT(drv->owner); +} + +static void uart_wait_until_sent(struct tty_struct *tty, int timeout) +{ + struct uart_info *info = (struct uart_info *) tty->driver_data; + unsigned long char_time, expire; + + if (info->port->type == PORT_UNKNOWN || + info->port->fifosize == 0) + return; + + /* + * Set the check interval to be 1/5 of the estimated time to + * send a single character, and make it at least 1. The check + * interval should also be less than the timeout. + * + * Note: we have to use pretty tight timings here to satisfy + * the NIST-PCTS. + */ + char_time = (info->timeout - HZ/50) / info->port->fifosize; + char_time = char_time / 5; + if (char_time == 0) + char_time = 1; + if (timeout && timeout < char_time) + char_time = timeout; + /* + * If the transmitter hasn't cleared in twice the approximate + * amount of time to send the entire FIFO, it probably won't + * ever clear. This assumes the UART isn't doing flow + * control, which is currently the case. Hence, if it ever + * takes longer than info->timeout, this is probably due to a + * UART bug of some kind. So, we clamp the timeout parameter at + * 2*info->timeout. + */ + if (!timeout || timeout > 2 * info->timeout) + timeout = 2 * info->timeout; + + expire = jiffies + timeout; +#ifdef DEBUG + printk("uart_wait_until_sent(%d), jiff=%lu, expire=%lu...\n", + MINOR(tty->device) - tty->driver.minor_start, jiffies, + expire); +#endif + while (!info->ops->tx_empty(info->port)) { + set_current_state(TASK_INTERRUPTIBLE); + schedule_timeout(char_time); + if (signal_pending(current)) + break; + if (timeout && time_after(jiffies, expire)) + break; + } + set_current_state(TASK_RUNNING); /* might not be needed */ +} + +/* + * This is called with the BKL in effect + * linux/drivers/char/tty_io.c:do_tty_hangup() + */ +static void uart_hangup(struct tty_struct *tty) +{ + struct uart_info *info = tty->driver_data; + struct uart_state *state = info->state; + + uart_flush_buffer(tty); + if (info->flags & ASYNC_CLOSING) + return; + uart_shutdown(info); + info->event = 0; + state->count = 0; + info->flags &= ~(ASYNC_NORMAL_ACTIVE|ASYNC_CALLOUT_ACTIVE); + info->tty = NULL; + wake_up_interruptible(&info->open_wait); +} + +static int uart_block_til_ready(struct tty_struct *tty, struct file *filp, + struct uart_info *info) +{ + DECLARE_WAITQUEUE(wait, current); + struct uart_state *state = info->state; + unsigned long flags; + int do_clocal = 0, extra_count = 0, retval; + + /* + * If the device is in the middle of being closed, then block + * until it's done, and then try again. + */ + if (tty_hung_up_p(filp) || + (info->flags & ASYNC_CLOSING)) { + if (info->flags & ASYNC_CLOSING) + interruptible_sleep_on(&info->close_wait); + return (info->flags & ASYNC_HUP_NOTIFY) ? + -EAGAIN : -ERESTARTSYS; + } + + /* + * If this is a callout device, then just make sure the normal + * device isn't being used. + */ + if (tty->driver.subtype == SERIAL_TYPE_CALLOUT) { + if (info->flags & ASYNC_NORMAL_ACTIVE) + return -EBUSY; + if ((info->flags & ASYNC_CALLOUT_ACTIVE) && + (info->flags & ASYNC_SESSION_LOCKOUT) && + (info->session != current->session)) + return -EBUSY; + if ((info->flags & ASYNC_CALLOUT_ACTIVE) && + (info->flags & ASYNC_PGRP_LOCKOUT) && + (info->pgrp != current->pgrp)) + return -EBUSY; + info->flags |= ASYNC_CALLOUT_ACTIVE; + return 0; + } + + /* + * If non-blocking mode is set, or the port is not enabled, + * then make the check up front and then exit. Note that + * we have set TTY_IO_ERROR for a non-enabled port. + */ + if ((filp->f_flags & O_NONBLOCK) || + (tty->flags & (1 << TTY_IO_ERROR))) { + if (info->flags & ASYNC_CALLOUT_ACTIVE) + return -EBUSY; + info->flags |= ASYNC_NORMAL_ACTIVE; + return 0; + } + + if (info->flags & ASYNC_CALLOUT_ACTIVE) { + if (state->normal_termios.c_cflag & CLOCAL) + do_clocal = 1; + } else { + if (tty->termios->c_cflag & CLOCAL) + do_clocal = 1; + } + + /* + * Block waiting for the carrier detect and the line to become + * free (i.e., not in use by the callout). While we are in + * this loop, state->count is dropped by one, so that + * rs_close() knows when to free things. We restore it upon + * exit, either normal or abnormal. + */ + retval = 0; + add_wait_queue(&info->open_wait, &wait); + down(&state->count_sem); + spin_lock_irqsave(&info->lock, flags); + if (!tty_hung_up_p(filp)) { + extra_count = 1; + state->count--; + } + spin_unlock_irqrestore(&info->lock, flags); + info->blocked_open++; + up(&state->count_sem); + while (1) { + spin_lock_irqsave(&info->lock, flags); + if (!(info->flags & ASYNC_CALLOUT_ACTIVE) && + (tty->termios->c_cflag & CBAUD)) { + info->mctrl |= TIOCM_DTR | TIOCM_RTS; + info->ops->set_mctrl(info->port, info->mctrl); + } + spin_unlock_irqrestore(&info->lock, flags); + set_current_state(TASK_INTERRUPTIBLE); + if (tty_hung_up_p(filp) || + !(info->flags & ASYNC_INITIALIZED)) { + if (info->flags & ASYNC_HUP_NOTIFY) + retval = -EAGAIN; + else + retval = -ERESTARTSYS; + break; + } + if (!(info->flags & ASYNC_CALLOUT_ACTIVE) && + !(info->flags & ASYNC_CLOSING) && + (do_clocal || + (info->ops->get_mctrl(info->port) & TIOCM_CAR))) + break; + if (signal_pending(current)) { + retval = -ERESTARTSYS; + break; + } + schedule(); + } + set_current_state(TASK_RUNNING); + remove_wait_queue(&info->open_wait, &wait); + down(&state->count_sem); + if (extra_count) + state->count++; + info->blocked_open--; + up(&state->count_sem); + if (retval) + return retval; + info->flags |= ASYNC_NORMAL_ACTIVE; + return 0; +} + +static struct uart_info *uart_get(struct uart_driver *drv, int line) +{ + struct uart_state *state = drv->state + line; + struct uart_info *info; + + down(&state->count_sem); + state->count++; + if (state->info) + goto out; + + info = kmalloc(sizeof(struct uart_info), GFP_KERNEL); + if (info) { + memset(info, 0, sizeof(struct uart_info)); + init_waitqueue_head(&info->open_wait); + init_waitqueue_head(&info->close_wait); + init_waitqueue_head(&info->delta_msr_wait); + info->port = state->port; + info->flags = info->port->flags; + info->ops = info->port->ops; + info->state = state; + tasklet_init(&info->tlet, uart_tasklet_action, + (unsigned long)info); + } + if (state->info) + kfree(info); + else + state->info = info; +out: + up(&state->count_sem); + return state->info; +} + +/* + * Make sure we have the temporary buffer allocated. Note + * that we set retval appropriately above, and we rely on + * this. + */ +static inline int uart_alloc_tmpbuf(void) +{ + if (!tmp_buf) { + unsigned long buf = get_zeroed_page(GFP_KERNEL); + if (!tmp_buf) { + if (buf) + tmp_buf = (u_char *)buf; + else + return -ENOMEM; + } else + free_page(buf); + } + return 0; +} + +/* + * In 2.4.5, calls to uart_open are serialised by the BKL in + * linux/fs/devices.c:chrdev_open() + * Note that if this fails, then uart_close() _will_ be called. + */ +static int uart_open(struct tty_struct *tty, struct file *filp) +{ + struct uart_driver *drv = (struct uart_driver *)tty->driver.driver_state; + struct uart_info *info; + int retval, line = MINOR(tty->device) - tty->driver.minor_start; + +#ifdef DEBUG + printk("uart_open(%d) called\n", line); +#endif + + retval = -ENODEV; + if (line >= tty->driver.num) + goto fail; + + if (!try_inc_mod_count(drv->owner)) + goto fail; + + info = uart_get(drv, line); + retval = -ENOMEM; + if (!info) + goto out; + + /* + * Set the tty driver_data. If we fail from this point on, + * the generic tty layer will cause uart_close(), which will + * decrement the module use count. + */ + tty->driver_data = info; + info->tty = tty; + info->tty->low_latency = (info->flags & ASYNC_LOW_LATENCY) ? 1 : 0; + + if (uart_alloc_tmpbuf()) + goto fail; + + /* + * If the port is in the middle of closing, bail out now. + */ + if (tty_hung_up_p(filp) || + (info->flags & ASYNC_CLOSING)) { + if (info->flags & ASYNC_CLOSING) + interruptible_sleep_on(&info->close_wait); + retval = (info->flags & ASYNC_HUP_NOTIFY) ? + -EAGAIN : -ERESTARTSYS; + goto fail; + } + + /* + * Make sure the device is in D0 state. + */ + if (info->state->count == 1) +#ifdef CONFIG_PM + pm_send(info->state->pm, PM_RESUME, (void *)0); +#else + if (info->ops->pm) + info->ops->pm(info->port, 0, 3); +#endif + + /* + * Start up the serial port + */ + retval = uart_startup(info); + if (retval) + goto fail; + + retval = uart_block_til_ready(tty, filp, info); + if (retval) + goto fail; + + if (info->state->count == 1) { + int changed_termios = 0; + + if (info->flags & ASYNC_SPLIT_TERMIOS) { + if (tty->driver.subtype == SERIAL_TYPE_NORMAL) + *tty->termios = info->state->normal_termios; + else + *tty->termios = info->state->callout_termios; + changed_termios = 1; + } + +#ifdef CONFIG_SERIAL_CORE_CONSOLE + /* + * Copy across the serial console cflag setting + */ + { + struct console *c = drv->cons; + if (c && c->cflag && c->index == line) { + tty->termios->c_cflag = c->cflag; + c->cflag = 0; + changed_termios = 1; + } + } +#endif + if (changed_termios) + uart_change_speed(info, NULL); + } + + info->session = current->session; + info->pgrp = current->pgrp; + return 0; + +out: + if (drv->owner) + __MOD_DEC_USE_COUNT(drv->owner); +fail: + return retval; +} + +#ifdef CONFIG_PROC_FS + +static const char *uart_type(struct uart_port *port) +{ + const char *str = NULL; + + if (port->ops->type) + str = port->ops->type(port); + + if (!str) + str = "unknown"; + + return str; +} + +static int uart_line_info(char *buf, struct uart_driver *drv, int i) +{ + struct uart_state *state = drv->state + i; + struct uart_port *port = state->port; + char stat_buf[32]; + u_int status; + int ret; + + ret = sprintf(buf, "%d: uart:%s port:%08X irq:%d", + port->line, uart_type(port), + port->iobase, port->irq); + + if (port->type == PORT_UNKNOWN) { + strcat(buf, "\n"); + return ret + 1; + } + + status = port->ops->get_mctrl(port); + + ret += sprintf(buf + ret, " tx:%d rx:%d", + port->icount.tx, port->icount.rx); + if (port->icount.frame) + ret += sprintf(buf + ret, " fe:%d", + port->icount.frame); + if (port->icount.parity) + ret += sprintf(buf + ret, " pe:%d", + port->icount.parity); + if (port->icount.brk) + ret += sprintf(buf + ret, " brk:%d", + port->icount.brk); + if (port->icount.overrun) + ret += sprintf(buf + ret, " oe:%d", + port->icount.overrun); + +#define INFOBIT(bit,str) \ + if (state->info && state->info->mctrl & (bit)) \ + strcat(stat_buf, (str)) +#define STATBIT(bit,str) \ + if (status & (bit)) \ + strcat(stat_buf, (str)) + + stat_buf[0] = '\0'; + stat_buf[1] = '\0'; + INFOBIT(TIOCM_RTS, "|RTS"); + STATBIT(TIOCM_CTS, "|CTS"); + INFOBIT(TIOCM_DTR, "|DTR"); + STATBIT(TIOCM_DSR, "|DSR"); + STATBIT(TIOCM_CAR, "|CD"); + STATBIT(TIOCM_RNG, "|RI"); + if (stat_buf[0]) + stat_buf[0] = ' '; + strcat(stat_buf, "\n"); + + ret += sprintf(buf + ret, stat_buf); + return ret; +} + +static int uart_read_proc(char *page, char **start, off_t off, + int count, int *eof, void *data) +{ + struct tty_driver *ttydrv = data; + struct uart_driver *drv = ttydrv->driver_state; + int i, len = 0, l; + off_t begin = 0; + + len += sprintf(page, "serinfo:1.0 driver%s%s revision:%s\n", + "", "", ""); + for (i = 0; i < drv->nr && len < PAGE_SIZE - 96; i++) { + l = uart_line_info(page + len, drv, i); + len += l; + if (len + begin > off + count) + goto done; + if (len + begin < off) { + begin += len; + len = 0; + } + } + *eof = 1; +done: + if (off >= len + begin) + return 0; + *start = page + (off - begin); + return (count < begin + len - off) ? count : (begin + len - off); +} +#endif + +#ifdef CONFIG_SERIAL_CORE_CONSOLE +/* + * Check whether an invalid uart number has been specified, and + * if so, search for the first available port that does have + * console support. + */ +struct uart_port * __init +uart_get_console(struct uart_port *ports, int nr, struct console *co) +{ + int idx = co->index; + + if (idx < 0 || idx >= nr || (ports[idx].iobase == 0 && + ports[idx].membase == NULL)) + for (idx = 0; idx < nr; idx++) + if (ports[idx].iobase != 0 || + ports[idx].membase != NULL) + break; + + co->index = idx; + + return ports + idx; +} + +/** + * uart_parse_options - Parse serial port baud/parity/bits/flow contro. + * @options: pointer to option string + * @baud: pointer to an 'int' variable for the baud rate. + * @parity: pointer to an 'int' variable for the parity. + * @bits: pointer to an 'int' variable for the number of data bits. + * @flow: pointer to an 'int' variable for the flow control character. + * + * uart_parse_options decodes a string containing the serial console + * options. The format of the string is , + * eg: 115200n8r + */ +void __init +uart_parse_options(char *options, int *baud, int *parity, int *bits, int *flow) +{ + char *s = options; + + *baud = simple_strtoul(s, NULL, 10); + while (*s >= '0' && *s <= '9') + s++; + if (*s) + *parity = *s++; + if (*s) + *bits = *s++ - '0'; + if (*s) + *flow = *s; +} + +/** + * uart_set_options - setup the serial console parameters + * @port: pointer to the serial ports uart_port structure + * @co: console pointer + * @baud: baud rate + * @parity: parity character - 'n' (none), 'o' (odd), 'e' (even) + * @bits: number of data bits + * @flow: flow control character - 'r' (rts) + */ +int __init +uart_set_options(struct uart_port *port, struct console *co, + int baud, int parity, int bits, int flow) +{ + u_int cflag = CREAD | HUPCL | CLOCAL; + u_int quot; + + /* + * Construct a cflag setting. + */ + switch (baud) { + case 1200: cflag |= B1200; break; + case 2400: cflag |= B2400; break; + case 4800: cflag |= B4800; break; + case 9600: cflag |= B9600; break; + case 19200: cflag |= B19200; break; + default: cflag |= B38400; baud = 38400; break; + case 57600: cflag |= B57600; break; + case 115200: cflag |= B115200; break; + case 230400: cflag |= B230400; break; + case 460800: cflag |= B460800; break; + } + + if (bits == 7) + cflag |= CS7; + else + cflag |= CS8; + + switch (parity) { + case 'o': case 'O': + cflag |= PARODD; + /*fall through*/ + case 'e': case 'E': + cflag |= PARENB; + break; + } + + co->cflag = cflag; + quot = (port->uartclk / (16 * baud)); + port->ops->change_speed(port, cflag, 0, quot); + + return 0; +} + +extern void ambauart_console_init(void); +extern void anakin_console_init(void); +extern void clps711xuart_console_init(void); +extern void sa1100_rs_console_init(void); +extern void serial8250_console_init(void); +extern void at91us3_rs_console_init(void); + +/* + * Central "initialise all serial consoles" container. Needs to be killed. + */ +void __init uart_console_init(void) +{ +#ifdef CONFIG_SERIAL_AMBA_CONSOLE + ambauart_console_init(); +#endif +#ifdef CONFIG_SERIAL_ANAKIN_CONSOLE + anakin_console_init(); +#endif +#ifdef CONFIG_SERIAL_CLPS711X_CONSOLE + clps711xuart_console_init(); +#endif +#ifdef CONFIG_SERIAL_SA1100_CONSOLE + sa1100_rs_console_init(); +#endif +#ifdef CONFIG_SERIAL_8250_CONSOLE + serial8250_console_init(); +#endif +#ifdef CONFIG_SERIAL_UART00_CONSOLE + uart00_console_init(); +#endif +#ifdef CONFIG_SERIAL_AT91US3_CONSOLE + at91us3_rs_console_init(); +#endif +} +#endif /* CONFIG_SERIAL_CORE_CONSOLE */ + +#ifdef CONFIG_PM +/* + * Serial port power management. + * + * This is pretty coarse at the moment - either all on or all off. We + * should probably some day do finer power management here some day. + * + * We don't actually save any state; the serial driver has enough + * state held internally to re-setup the port when we come out of D3. + */ +static int uart_pm_set_state(struct uart_state *state, int pm_state, int oldstate) +{ + struct uart_port *port = state->port; + struct uart_ops *ops = port->ops; + int running = state->info && + state->info->flags & ASYNC_INITIALIZED; + + if (port->type == PORT_UNKNOWN) + return 0; + +//printk("pm: %08x: %d -> %d, %srunning\n", port->iobase, dev->state, pm_state, running ? "" : "not "); + if (pm_state == 0) { + if (ops->pm) + ops->pm(port, pm_state, oldstate); + if (running) { + ops->set_mctrl(port, 0); + ops->startup(port, state->info); + uart_change_speed(state->info, NULL); + ops->set_mctrl(port, state->info->mctrl); + ops->start_tx(port, 1, 0); + } + + /* + * Re-enable the console device after suspending. + */ + if (state->cons && state->cons->index == port->line) + state->cons->flags |= CON_ENABLED; + } else if (pm_state == 1) { + if (ops->pm) + ops->pm(port, pm_state, oldstate); + } else { + /* + * Disable the console device before suspending. + */ + if (state->cons && state->cons->index == port->line) + state->cons->flags &= ~CON_ENABLED; + + if (running) { + ops->stop_tx(port, 0); + ops->set_mctrl(port, 0); + ops->stop_rx(port); + ops->shutdown(port, state->info); + } + if (ops->pm) + ops->pm(port, pm_state, oldstate); + } + return 0; +} + +/* + * Wakeup support. + */ +static int uart_pm_set_wakeup(struct uart_state *state, int data) +{ + int err = 0; + + if (state->port->ops->set_wake) + err = state->port->ops->set_wake(state->port, data); + + return err; +} + +static int uart_pm(struct pm_dev *dev, pm_request_t rqst, void *data) +{ + struct uart_state *state = dev->data; + int err = 0; + + switch (rqst) { + case PM_SUSPEND: + case PM_RESUME: + err = uart_pm_set_state(state, (int)data, dev->state); + break; + + case PM_SET_WAKEUP: + err = uart_pm_set_wakeup(state, (int)data); + break; + } + return err; +} +#endif + +static inline void +uart_report_port(struct uart_driver *drv, struct uart_port *port) +{ + printk("%s%d at ", drv->normal_name, port->line); + switch (port->iotype) { + case SERIAL_IO_PORT: + printk("I/O 0x%x", port->iobase); + break; + case SERIAL_IO_HUB6: + printk("I/O 0x%x offset 0x%x", port->iobase, port->hub6); + break; + case SERIAL_IO_MEM: + printk("MEM 0x%lx", port->mapbase); + break; + } + printk(" (irq = %d) is a %s\n", port->irq, uart_type(port)); +} + +static void +uart_setup_port(struct uart_driver *drv, struct uart_state *state) +{ + struct uart_port *port = state->port; + int flags = UART_CONFIG_TYPE; + + init_MUTEX(&state->count_sem); + + state->close_delay = 5 * HZ / 10; + state->closing_wait = 30 * HZ; + + port->type = PORT_UNKNOWN; + +#ifdef CONFIG_PM + state->cons = drv->cons; + state->pm = pm_register(PM_SYS_DEV, PM_SYS_COM, uart_pm); + if (state->pm) + state->pm->data = state; +#endif + + /* + * If there isn't a port here, don't do anything further. + */ + if (!port->iobase && !port->mapbase) + return; + + /* + * Now do the auto configuration stuff. Note that config_port + * is expected to claim the resources and map the port for us. + */ + if (port->flags & ASYNC_AUTO_IRQ) + flags |= UART_CONFIG_IRQ; + if (port->flags & ASYNC_BOOT_AUTOCONF) + port->ops->config_port(port, flags); + + /* + * Only register this port if it is detected. + */ + if (port->type != PORT_UNKNOWN) { + tty_register_devfs(drv->normal_driver, 0, drv->minor + + state->port->line); + tty_register_devfs(drv->callout_driver, 0, drv->minor + + state->port->line); + uart_report_port(drv, port); + } + +#ifdef CONFIG_PM + /* + * Power down all ports by default, except the console if we have one. + */ + if (state->pm && (!drv->cons || port->line != drv->cons->index)) + pm_send(state->pm, PM_SUSPEND, (void *)3); +#endif +} + +/* + * Register a set of ports with the core driver. Note that we don't + * printk any information about the ports; that is up to the low level + * driver to do if they so wish. + */ +int uart_register_driver(struct uart_driver *drv) +{ + struct tty_driver *normal, *callout; + int i, retval; + + if (drv->state) + panic("drv->state already allocated\n"); + + /* + * Maybe we should be using a slab cache for this, especially if + * we have a large number of ports to handle. Note that we also + * allocate space for an integer for reference counting. + */ + drv->state = kmalloc(sizeof(struct uart_state) * drv->nr + + sizeof(int), GFP_KERNEL); + retval = -ENOMEM; + if (!drv->state) + goto out; + + memset(drv->state, 0, sizeof(struct uart_state) * drv->nr + + sizeof(int)); + + normal = drv->normal_driver; + callout = drv->callout_driver; + + normal->magic = TTY_DRIVER_MAGIC; + normal->driver_name = drv->normal_name; + normal->name = drv->normal_name; + normal->major = drv->normal_major; + normal->minor_start = drv->minor; + normal->num = drv->nr; + normal->type = TTY_DRIVER_TYPE_SERIAL; + normal->subtype = SERIAL_TYPE_NORMAL; + normal->init_termios = tty_std_termios; + normal->init_termios.c_cflag = B38400 | CS8 | CREAD | HUPCL | CLOCAL; + normal->flags = TTY_DRIVER_REAL_RAW | TTY_DRIVER_NO_DEVFS; + normal->refcount = (int *)(drv->state + drv->nr); + normal->table = drv->table; + normal->termios = drv->termios; + normal->termios_locked = drv->termios_locked; + normal->driver_state = drv; + + normal->open = uart_open; + normal->close = uart_close; + normal->write = uart_write; + normal->put_char = uart_put_char; + normal->flush_chars = uart_flush_chars; + normal->write_room = uart_write_room; + normal->chars_in_buffer = uart_chars_in_buffer; + normal->flush_buffer = uart_flush_buffer; + normal->ioctl = uart_ioctl; + normal->throttle = uart_throttle; + normal->unthrottle = uart_unthrottle; + normal->send_xchar = uart_send_xchar; + normal->set_termios = uart_set_termios; + normal->stop = uart_stop; + normal->start = uart_start; + normal->hangup = uart_hangup; + normal->break_ctl = uart_break_ctl; + normal->wait_until_sent = uart_wait_until_sent; +#ifdef CONFIG_PROC_FS + normal->read_proc = uart_read_proc; +#endif + + /* + * The callout device is just like the normal device except for + * the major number and the subtype code. + */ + *callout = *normal; + callout->name = drv->callout_name; + callout->major = drv->callout_major; + callout->subtype = SERIAL_TYPE_CALLOUT; + callout->read_proc = NULL; + callout->proc_entry = NULL; + + for (i = 0; i < drv->nr; i++) { + struct uart_state *state = drv->state + i; + + state->callout_termios = callout->init_termios; + state->normal_termios = normal->init_termios; + state->port = drv->port + i; + state->port->line = i; + + uart_setup_port(drv, state); + } + + retval = tty_register_driver(normal); + if (retval) + goto out; + + retval = tty_register_driver(callout); + if (retval) + tty_unregister_driver(normal); + +out: + if (retval && drv->state) + kfree(drv->state); + return retval; +} + +void uart_unregister_driver(struct uart_driver *drv) +{ + int i; + + for (i = 0; i < drv->nr; i++) { + struct uart_state *state = drv->state + i; + + if (state->info && state->info->tty) + tty_hangup(state->info->tty); + + pm_unregister(state->pm); + + if (state->port->type != PORT_UNKNOWN) + state->port->ops->release_port(state->port); + if (state->info) { + tasklet_kill(&state->info->tlet); + kfree(state->info); + } + } + + tty_unregister_driver(drv->normal_driver); + tty_unregister_driver(drv->callout_driver); + + kfree(drv->state); +} + +static int uart_match_port(struct uart_port *port1, struct uart_port *port2) +{ + if (port1->iotype != port2->iotype) + return 0; + switch (port1->iotype) { + case SERIAL_IO_PORT: return (port1->iobase == port2->iobase); + case SERIAL_IO_MEM: return (port1->membase == port2->membase); + } + return 0; +} + +/** + * uart_register_port: register a port with the generic uart driver + * @reg: pointer to the uart low level driver structure for this port + * @port: uart port structure describing the port + * + * Register a UART with the specified low level driver. Detect the + * type of the port if ASYNC_BOOT_AUTOCONF is set, and detect the IRQ + * if ASYNC_AUTO_IRQ is set. + * + * Returns negative error, or positive line number. + */ +int uart_register_port(struct uart_driver *drv, struct uart_port *port) +{ + struct uart_state *state = NULL; + int i, flags = UART_CONFIG_TYPE; + + /* + * First, find a port entry which matches. Note: if we do + * find a matching entry, and it has a non-zero use count, + * then we can't register the port. + */ + down(&port_sem); + for (i = 0; i < drv->nr; i++) { + if (uart_match_port(drv->state[i].port, port)) { + down(&drv->state[i].count_sem); + state = &drv->state[i]; + break; + } + } + + /* + * If we didn't find a matching entry, look for the first + * free entry. We look for one which hasn't been previously + * used (indicated by zero iobase). + */ + if (!state) { + for (i = 0; i < drv->nr; i++) { + if (drv->state[i].port->type == PORT_UNKNOWN && + drv->state[i].port->iobase == 0) { + down(&drv->state[i].count_sem); + if (drv->state[i].count == 0) { + state = &drv->state[i]; + break; + } + } + } + } + + /* + * Ok, that also failed. Find the first unused entry, which + * may be previously in use. + */ + if (!state) { + for (i = 0; i < drv->nr; i++) { + if (drv->state[i].port->type == PORT_UNKNOWN) { + down(&drv->state[i].count_sem); + if (drv->state[i].count == 0) { + state = &drv->state[i]; + break; + } + } + } + } + + up(&port_sem); + + if (!state) + return -ENOSPC; + + /* + * If we find a port that matches this one, and it appears to + * be in-use (even if it doesn't have a type) we shouldn't alter + * it underneath itself - the port may be open and trying to do + * useful work. + */ + if (state->count != 0 || + (state->info && state->info->blocked_open != 0)) { + up(&state->count_sem); + return -EBUSY; + } + + /* + * We're holding the lock for this port. Copy the relevant data + * into the port structure. + */ + state->port->iobase = port->iobase; + state->port->membase = port->membase; + state->port->irq = port->irq; + state->port->uartclk = port->uartclk; + state->port->fifosize = port->fifosize; + state->port->regshift = port->regshift; + state->port->iotype = port->iotype; + state->port->flags = port->flags; + +#if 0 //def CONFIG_PM + /* we have already registered the power management handlers */ + state->pm = pm_register(PM_SYS_DEV, PM_SYS_COM, uart_pm); + if (state->pm) { + state->pm->data = state; + + /* + * Power down all ports by default, except + * the console if we have one. + */ + if (!drv->cons || state->port->line != drv->cons->index) + pm_send(state->pm, PM_SUSPEND, (void *)3); + } +#endif + + if (state->port->flags & ASYNC_AUTO_IRQ) + flags |= UART_CONFIG_IRQ; + if (state->port->flags & ASYNC_BOOT_AUTOCONF) + state->port->ops->config_port(state->port, flags); + + tty_register_devfs(drv->normal_driver, 0, drv->minor + + state->port->line); + tty_register_devfs(drv->callout_driver, 0, drv->minor + + state->port->line); + + uart_report_port(drv, state->port); + + up(&state->count_sem); + return i; +} + +/* + * Unregister the specified port index on the specified driver. + */ +void uart_unregister_port(struct uart_driver *drv, int line) +{ + struct uart_state *state; + + if (line < 0 || line >= drv->nr) { + printk(KERN_ERR "Attempt to unregister %s%d\n", + drv->normal_name, line); + return; + } + + state = drv->state + line; + + down(&state->count_sem); + /* + * The port has already gone. We have to hang up the line + * to kill all usage of this port. + */ + if (state->info && state->info->tty) + tty_hangup(state->info->tty); + + /* + * Free the ports resources, if any. + */ + state->port->ops->release_port(state->port); + + /* + * Indicate that there isn't a port here anymore. + */ + state->port->type = PORT_UNKNOWN; + +#if 0 // not yet + /* + * No point in doing power management for hardware that + * isn't present. + */ + pm_unregister(state->pm); +#endif + + /* + * Remove the devices from devfs + */ + tty_unregister_devfs(drv->normal_driver, drv->minor + line); + tty_unregister_devfs(drv->callout_driver, drv->minor + line); + up(&state->count_sem); +} + +EXPORT_SYMBOL(uart_event); +EXPORT_SYMBOL(uart_register_driver); +EXPORT_SYMBOL(uart_unregister_driver); +EXPORT_SYMBOL(uart_register_port); +EXPORT_SYMBOL(uart_unregister_port); + +static int __init uart_init(void) +{ + return 0; +} + +static void __exit uart_exit(void) +{ + free_page((unsigned long)tmp_buf); + tmp_buf = NULL; +} + +module_init(uart_init); +module_exit(uart_exit); + +MODULE_DESCRIPTION("Serial driver core"); +MODULE_LICENSE("GPL"); diff -urN orig/drivers/serial/omaha.c linux/drivers/serial/omaha.c --- orig/drivers/serial/omaha.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/serial/omaha.c Thu Oct 24 14:58:03 2002 @@ -0,0 +1,584 @@ +/* + * linux/drivers/char/omaha.c + * + * Driver for Omaha serial port + * + * Based on drivers/char/serial.c, by Linus Torvalds, Theodore Ts'o. + * + * Copyright 1999-2002 ARM Limited + * Copyright (C) 2000 Deep Blue Solutions Ltd. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + * + * $Id: serial_amba.c,v 1.4 2001/07/17 20:34:27 rmk Exp $ + * + * This is a generic driver for ARM AMBA-type serial ports. They + * have a lot of 16550-like features, but are not register compatable. + * Note that although they do have CTS, DCD and DSR inputs, they do + * not have an RI input, nor do they have DTR or RTS outputs. If + * required, these have to be supplied via some other means (eg, GPIO) + * and hooked into this driver. + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include + +#if defined(CONFIG_SERIAL_OMAHA_CONSOLE) && defined(CONFIG_MAGIC_SYSRQ) +#define SUPPORT_SYSRQ +#endif + +#include + +#include + +#define UART_NR 1 + +#define SERIAL_OMAHA_MAJOR 204 +#define SERIAL_OMAHA_MINOR 32 +#define SERIAL_OMAHA_NR UART_NR + +#define CALLOUT_OMAHA_NAME "cuaom" +#define CALLOUT_OMAHA_MAJOR 205 +#define CALLOUT_OMAHA_MINOR 32 +#define CALLOUT_OMAHA_NR UART_NR + +static struct tty_driver normal, callout; +static struct tty_struct *omaha_table[UART_NR]; +static struct termios *omaha_termios[UART_NR], *omaha_termios_locked[UART_NR]; +#ifdef SUPPORT_SYSRQ +static struct console omaha_console; +#endif + +#define OMAHA_ISR_PASS_LIMIT 256 + +/* + * Access macros for the Omaha UARTs + */ + +#define UART_GET_FR(p) readb((p)->membase + OMAHA_UTRSTAT) +#define UART_GET_CHAR(p) readb((p)->membase + OMAHA_URXH) +#define UART_PUT_CHAR(p, c) writel((c), (p)->membase + OMAHA_UTXH) +#define UART_GET_RSR(p) readb((p)->membase + OMAHA_UERSTAT) +#define UART_FIFO_STATUS(p) (readl((p)->membase + OMAHA_UFSTAT)) +#define UART_RX_DATA(s) (((s) & OMAHA_RXFF_CNT) != 0) +#define UART_TX_DATA(s) (!((s) & OMAHA_TXFF)) +#define UART_TX_READY(s) (((s) & OMAHA_UTX_EMPTY)) +#define UART_TX_EMPTY(p) ((UART_GET_FR(p) & OMAHA_UTXEMPTY) != 0) + +#define UART_DUMMY_RSR_RX 256 +#define UART_PORT_SIZE 64 + +#define RX_IRQ(port) ((port)->irq) +#define TX_IRQ(port) ((port)->irq + 5) + +/* + * Our private driver data mappings. + */ +#define drv_old_status driver_priv + +static void omahauart_stop_tx(struct uart_port *port, u_int from_tty) +{ + disable_irq(TX_IRQ(port)); +} + +static void omahauart_start_tx(struct uart_port *port, u_int nonempty, u_int from_tty) +{ + if (nonempty) + enable_irq(TX_IRQ(port)); +} + +static void omahauart_stop_rx(struct uart_port *port) +{ + disable_irq(RX_IRQ(port)); +} + +static void omahauart_enable_ms(struct uart_port *port) +{ + // Do nothing... +} + +static void +#ifdef SUPPORT_SYSRQ +omahauart_rx_chars(struct uart_info *info, struct pt_regs *regs) +#else +omahauart_rx_chars(struct uart_info *info) +#endif +{ + struct tty_struct *tty = info->tty; + volatile unsigned int status, data, ch, rsr, max_count = 256; + struct uart_port *port = info->port; + + status = UART_FIFO_STATUS(port); + while (UART_RX_DATA(status) && max_count--) { + if (tty->flip.count >= TTY_FLIPBUF_SIZE) { + tty->flip.tqueue.routine((void *)tty); + if (tty->flip.count >= TTY_FLIPBUF_SIZE) { + printk(KERN_WARNING "TTY_DONT_FLIP set\n"); + return; + } + } + + ch = UART_GET_CHAR(port); + + *tty->flip.char_buf_ptr = ch; + *tty->flip.flag_buf_ptr = TTY_NORMAL; + port->icount.rx++; + + /* + * Note that the error handling code is + * out of the main execution path + */ + rsr = UART_GET_RSR(port) | UART_DUMMY_RSR_RX; + if (rsr & 0xf) { + if (rsr & OMAHA_UART_BREAK) { + rsr &= ~(OMAHA_UART_FRAME | OMAHA_UART_PARITY); + port->icount.brk++; + if (uart_handle_break(info, &omaha_console)) + goto ignore_char; + } else if (rsr & OMAHA_UART_PARITY) + port->icount.parity++; + else if (rsr & OMAHA_UART_FRAME) + port->icount.frame++; + if (rsr & OMAHA_UART_OVERRUN) + port->icount.overrun++; + + rsr &= port->read_status_mask; + + if (rsr & OMAHA_UART_BREAK) + *tty->flip.flag_buf_ptr = TTY_BREAK; + else if (rsr & OMAHA_UART_PARITY) + *tty->flip.flag_buf_ptr = TTY_PARITY; + else if (rsr & OMAHA_UART_FRAME) + *tty->flip.flag_buf_ptr = TTY_FRAME; + } + + if (uart_handle_sysrq_char(info, ch, regs)) + goto ignore_char; + + if ((rsr & port->ignore_status_mask) == 0) { + tty->flip.flag_buf_ptr++; + tty->flip.char_buf_ptr++; + tty->flip.count++; + } + if ((rsr & OMAHA_UART_OVERRUN) && + tty->flip.count < TTY_FLIPBUF_SIZE) { + /* + * Overrun is special, since it's reported + * immediately, and doesn't affect the current + * character + */ + *tty->flip.char_buf_ptr++ = 0; + *tty->flip.flag_buf_ptr++ = TTY_OVERRUN; + tty->flip.count++; + } + ignore_char: + status = UART_FIFO_STATUS(port); + } + tty_flip_buffer_push(tty); + return; +} + +static void omahauart_tx_chars(struct uart_info *info) +{ + struct uart_port *port = info->port; + volatile unsigned int status; + + if (port->x_char) { + UART_PUT_CHAR(port, port->x_char); + port->icount.tx++; + port->x_char = 0; + return; + } + if (info->xmit.head == info->xmit.tail + || info->tty->stopped + || info->tty->hw_stopped) { + omahauart_stop_tx(port, 0); + return; + } + + status = UART_FIFO_STATUS(info->port); + + // FIll FIFO as far as possible + while(UART_TX_DATA(UART_FIFO_STATUS(info->port))) + { + UART_PUT_CHAR(port, info->xmit.buf[info->xmit.tail]); + info->xmit.tail = (info->xmit.tail + 1) & (UART_XMIT_SIZE - 1); + port->icount.tx++; + if (info->xmit.head == info->xmit.tail) + break; + } + + if (CIRC_CNT(info->xmit.head, info->xmit.tail, UART_XMIT_SIZE) < + WAKEUP_CHARS) + uart_event(info, EVT_WRITE_WAKEUP); + + if (info->xmit.head == info->xmit.tail) + omahauart_stop_tx(info->port, 0); +} + +static void omahauart_int_tx(int irq, void *dev_id, struct pt_regs *regs) +{ + struct uart_info *info = dev_id; + volatile unsigned int status, pass_counter = OMAHA_ISR_PASS_LIMIT; + + status = UART_FIFO_STATUS(info->port); + do { + // TX if FIFO not full + if (UART_TX_DATA(status)) + omahauart_tx_chars(info); + + if (pass_counter-- == 0) + break; + + status = UART_FIFO_STATUS(info->port); + } while (UART_TX_DATA(status)); +} + +static void omahauart_int_rx(int irq, void *dev_id, struct pt_regs *regs) +{ + struct uart_info *info = dev_id; + volatile unsigned int status, pass_counter = OMAHA_ISR_PASS_LIMIT; + + status = UART_FIFO_STATUS(info->port); + do { + if (UART_RX_DATA(status)) +#ifdef SUPPORT_SYSRQ + omahauart_rx_chars(info, regs); +#else + omahauart_rx_chars(info); +#endif + + if (pass_counter-- == 0) + break; + + status = UART_FIFO_STATUS(info->port); + } while (UART_RX_DATA(status)); +} + +static u_int omahauart_tx_empty(struct uart_port *port) +{ + return UART_FIFO_STATUS(port) ? 0 : TIOCSER_TEMT; +} + +static int omahauart_get_mctrl(struct uart_port *port) +{ + // Report no errors. + + return 0; +} + +static void omahauart_set_mctrl(struct uart_port *port, u_int mctrl) +{ + // Do nothing. +} + +static void omahauart_break_ctl(struct uart_port *port, int break_state) +{ + // Do nothing. +} + +static int omahauart_startup(struct uart_port *port, struct uart_info *info) +{ + unsigned int tmp; + int retval; + + /* + * Allocate the IRQs + */ + retval = request_irq(TX_IRQ(port), omahauart_int_tx, 0, "omaha_uart_tx", info); + if (retval) + return retval; + + retval = request_irq(RX_IRQ(port), omahauart_int_rx, 0, "omaha_uart_rx", info); + + if (retval) + { + free_irq(TX_IRQ(port), info); + return retval; + } + + /* + * initialise the old status of the modem signals + */ + info->drv_old_status = 0; + + // Clear all errors + writel(0, port->membase + OMAHA_UERSTAT); + + // Enable FIFO, 16-byte watermark, also do reset (auto-clearing) + writel(0xF7, port->membase + OMAHA_UFCON); + + // Level driven TX/RX ints, with rx timeout enabled + tmp = readl(port->membase + OMAHA_UCON); + tmp |= 0x280; // rx is pulse driven... + writel(tmp, port->membase + OMAHA_UCON); + + return 0; +} + +static void omahauart_shutdown(struct uart_port *port, struct uart_info *info) +{ + /* + * Free the interrupt + */ + free_irq(TX_IRQ(port), info); /* TX interrupt */ + free_irq(RX_IRQ(port), info); /* RX interrupt */ + +} + +static void omahauart_change_speed(struct uart_port *port, u_int cflag, u_int iflag, u_int quot) +{ + // Do nothing. +} + +static const char *omahauart_type(struct uart_port *port) +{ + return port->type == PORT_OMAHA ? "OMAHA" : NULL; +} + +/* + * Release the memory region(s) being used by 'port' + */ +static void omahauart_release_port(struct uart_port *port) +{ + release_mem_region(port->mapbase, UART_PORT_SIZE); +} + +/* + * Request the memory region(s) being used by 'port' + */ +static int omahauart_request_port(struct uart_port *port) +{ + return request_mem_region(port->mapbase, UART_PORT_SIZE, "serial_omaha") + != NULL ? 0 : -EBUSY; +} + +/* + * Configure/autoconfigure the port. + */ +static void omahauart_config_port(struct uart_port *port, int flags) +{ + if (flags & UART_CONFIG_TYPE) { + port->type = PORT_OMAHA; + omahauart_request_port(port); + } +} + +/* + * verify the new serial_struct (for TIOCSSERIAL). + */ +static int omahauart_verify_port(struct uart_port *port, struct serial_struct *ser) +{ + int ret = 0; + if (ser->type != PORT_UNKNOWN && ser->type != PORT_OMAHA) + ret = -EINVAL; + if (ser->irq < 0 || ser->irq >= NR_IRQS) + ret = -EINVAL; + if (ser->baud_base < 9600) + ret = -EINVAL; + return ret; +} + +static struct uart_ops omaha_pops = { + .tx_empty = omahauart_tx_empty, + .set_mctrl = omahauart_set_mctrl, + .get_mctrl = omahauart_get_mctrl, + .stop_tx = omahauart_stop_tx, + .start_tx = omahauart_start_tx, + .stop_rx = omahauart_stop_rx, + .enable_ms = omahauart_enable_ms, + .break_ctl = omahauart_break_ctl, + .startup = omahauart_startup, + .shutdown = omahauart_shutdown, + .change_speed = omahauart_change_speed, + .type = omahauart_type, + .release_port = omahauart_release_port, + .request_port = omahauart_request_port, + .config_port = omahauart_config_port, + .verify_port = omahauart_verify_port, +}; + +static struct uart_port omaha_ports[UART_NR] = { + { + .membase = (void *)IO_ADDRESS(OMAHA_UART0_BASE), + .mapbase = OMAHA_UART0_BASE, + .iotype = SERIAL_IO_MEM, + .irq = OMAHA_INT_URXD0, + .uartclk = 10000000, + .fifosize = 8, + .unused = { 4, 5 }, /*Udriver_priv: PORT_CTRLS(5, 4), */ + .ops = &omaha_pops, + .flags = ASYNC_BOOT_AUTOCONF, + } +}; + +#ifdef CONFIG_SERIAL_OMAHA_CONSOLE +static void omahauart_console_write(struct console *co, const char *s, u_int count) +{ + struct uart_port *port = omaha_ports + co->index; + unsigned int status; + int i; + + /* + * First save the CR then disable the interrupts + */ + + /* + * Now, do each character + */ + for (i = 0; i < count; i++) { + do { + status = UART_GET_FR(port); + } while ((status & OMAHA_UTX_EMPTY) == 0); + UART_PUT_CHAR(port, s[i]); + if (s[i] == '\n') { + do { + status = UART_GET_FR(port); + } while ((status & OMAHA_UTX_EMPTY) == 0); + UART_PUT_CHAR(port, '\r'); + } + } + + /* + * Finally, wait for transmitter to become empty + * and restore the TCR + */ + do { + status = UART_GET_FR(port); + } while ((status & OMAHA_UTX_EMPTY) == 0); +} + +static kdev_t omahauart_console_device(struct console *co) +{ + return MKDEV(SERIAL_OMAHA_MAJOR, SERIAL_OMAHA_MINOR + co->index); +} + +static int omahauart_console_wait_key(struct console *co) +{ + struct uart_port *port = omaha_ports + co->index; + unsigned int status; + + do { + status = UART_FIFO_STATUS(port); + } while (!UART_RX_DATA(status)); + return UART_GET_CHAR(port); +} + +static void __init +omahauart_console_get_options(struct uart_port *port, int *baud, int *parity, int *bits) +{ + // Do nothing. +} + +static int __init omahauart_console_setup(struct console *co, char *options) +{ + struct uart_port *port; + int baud = 38400; + int bits = 8; + int parity = 'n'; + int flow = 'n'; + + /* + * Check whether an invalid uart number has been specified, and + * if so, search for the first available port that does have + * console support. + */ + port = uart_get_console(omaha_ports, UART_NR, co); + + if (options) + uart_parse_options(options, &baud, &parity, &bits, &flow); + else + omahauart_console_get_options(port, &baud, &parity, &bits); + + return uart_set_options(port, co, baud, parity, bits, flow); +} + +static struct console omaha_console = { + .write = omahauart_console_write, + .device = omahauart_console_device, + .wait_key = omahauart_console_wait_key, + .setup = omahauart_console_setup, + .flags = CON_PRINTBUFFER, + .index = -1, +}; + +void __init omahauart_console_init(void) +{ + register_console(&omaha_console); +} + +#define OMAHA_CONSOLE &omaha_console +#else +#define OMAHA_CONSOLE NULL +#endif + +static struct uart_driver omaha_reg = { + .owner = THIS_MODULE, + .normal_major = SERIAL_OMAHA_MAJOR, +#ifdef CONFIG_DEVFS_FS + .normal_name = "ttyOM%d", + .callout_name = "cuaom%d", +#else + .normal_name = "ttyOM", + .callout_name = "cuaom", +#endif + .normal_driver = &normal, + .callout_major = CALLOUT_OMAHA_MAJOR, + .callout_driver = &callout, + .table = omaha_table, + .termios = omaha_termios, + .termios_locked = omaha_termios_locked, + .minor = SERIAL_OMAHA_MINOR, + .nr = UART_NR, + .port = omaha_ports, + .cons = OMAHA_CONSOLE, +}; + +static int __init omahauart_init(void) +{ + return uart_register_driver(&omaha_reg); +} + +static void __exit omahauart_exit(void) +{ + uart_unregister_driver(&omaha_reg); +} + +module_init(omahauart_init); +module_exit(omahauart_exit); diff -urN orig/drivers/serial/sa1100.c linux/drivers/serial/sa1100.c --- orig/drivers/serial/sa1100.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/serial/sa1100.c Thu Oct 24 10:53:25 2002 @@ -0,0 +1,786 @@ +/* + * linux/drivers/char/serial_sa1100.c + * + * Driver for SA11x0 serial ports + * + * Based on drivers/char/serial.c, by Linus Torvalds, Theodore Ts'o. + * + * Copyright (C) 2000 Deep Blue Solutions Ltd. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + * + * $Id: sa1100.c,v 1.14.2.4 2002/10/24 09:53:25 rmk Exp $ + * + */ +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include + +#if defined(CONFIG_SERIAL_SA1100_CONSOLE) && defined(CONFIG_MAGIC_SYSRQ) +#define SUPPORT_SYSRQ +#endif + +#include + +/* We've been assigned a range on the "Low-density serial ports" major */ +#define SERIAL_SA1100_MAJOR 204 +#define CALLOUT_SA1100_MAJOR 205 +#define MINOR_START 5 + +#define NR_PORTS 3 + +#define SA1100_ISR_PASS_LIMIT 256 + +/* + * Convert from ignore_status_mask or read_status_mask to UTSR[01] + */ +#define SM_TO_UTSR0(x) ((x) & 0xff) +#define SM_TO_UTSR1(x) ((x) >> 8) +#define UTSR0_TO_SM(x) ((x)) +#define UTSR1_TO_SM(x) ((x) << 8) + +#define UART_GET_UTCR0(port) __raw_readl((port)->membase + UTCR0) +#define UART_GET_UTCR1(port) __raw_readl((port)->membase + UTCR1) +#define UART_GET_UTCR2(port) __raw_readl((port)->membase + UTCR2) +#define UART_GET_UTCR3(port) __raw_readl((port)->membase + UTCR3) +#define UART_GET_UTSR0(port) __raw_readl((port)->membase + UTSR0) +#define UART_GET_UTSR1(port) __raw_readl((port)->membase + UTSR1) +#define UART_GET_CHAR(port) __raw_readl((port)->membase + UTDR) + +#define UART_PUT_UTCR0(port,v) __raw_writel((v),(port)->membase + UTCR0) +#define UART_PUT_UTCR1(port,v) __raw_writel((v),(port)->membase + UTCR1) +#define UART_PUT_UTCR2(port,v) __raw_writel((v),(port)->membase + UTCR2) +#define UART_PUT_UTCR3(port,v) __raw_writel((v),(port)->membase + UTCR3) +#define UART_PUT_UTSR0(port,v) __raw_writel((v),(port)->membase + UTSR0) +#define UART_PUT_UTSR1(port,v) __raw_writel((v),(port)->membase + UTSR1) +#define UART_PUT_CHAR(port,v) __raw_writel((v),(port)->membase + UTDR) + +/* + * This is the size of our serial port register set. + */ +#define UART_PORT_SIZE 0x24 + +static struct tty_driver normal, callout; +static struct tty_struct *sa1100_table[NR_PORTS]; +static struct termios *sa1100_termios[NR_PORTS], *sa1100_termios_locked[NR_PORTS]; +static int (*sa1100_open)(struct uart_port *, struct uart_info *); +static void (*sa1100_close)(struct uart_port *, struct uart_info *); +#ifdef SUPPORT_SYSRQ +static struct console sa1100_console; +#endif + +/* + * interrupts disabled on entry + */ +static void sa1100_stop_tx(struct uart_port *port, u_int from_tty) +{ + u32 utcr3 = UART_GET_UTCR3(port); + UART_PUT_UTCR3(port, utcr3 & ~UTCR3_TIE); + port->read_status_mask &= ~UTSR0_TO_SM(UTSR0_TFS); +} + +/* + * interrupts may not be disabled on entry + */ +static void sa1100_start_tx(struct uart_port *port, u_int nonempty, u_int from_tty) +{ + if (nonempty) { + unsigned long flags; + u32 utcr3; + + local_irq_save(flags); + utcr3 = UART_GET_UTCR3(port); + port->read_status_mask |= UTSR0_TO_SM(UTSR0_TFS); + UART_PUT_UTCR3(port, utcr3 | UTCR3_TIE); + local_irq_restore(flags); + } +} + +/* + * Interrupts enabled + */ +static void sa1100_stop_rx(struct uart_port *port) +{ + u32 utcr3 = UART_GET_UTCR3(port); + UART_PUT_UTCR3(port, utcr3 & ~UTCR3_RIE); +} + +/* + * No modem control lines + */ +static void sa1100_enable_ms(struct uart_port *port) +{ +} + +static void +sa1100_rx_chars(struct uart_info *info, struct pt_regs *regs) +{ + struct tty_struct *tty = info->tty; + unsigned int status, ch, flg, ignored = 0; + struct uart_port *port = info->port; + + status = UTSR1_TO_SM(UART_GET_UTSR1(port)) | UTSR0_TO_SM(UART_GET_UTSR0(port)); + while (status & UTSR1_TO_SM(UTSR1_RNE)) { + ch = UART_GET_CHAR(port); + + if (tty->flip.count >= TTY_FLIPBUF_SIZE) + goto ignore_char; + port->icount.rx++; + + flg = TTY_NORMAL; + + /* + * note that the error handling code is + * out of the main execution path + */ + if (status & UTSR1_TO_SM(UTSR1_PRE | UTSR1_FRE | UTSR1_ROR)) + goto handle_error; + + if (uart_handle_sysrq_char(info, ch, regs)) + goto ignore_char; + + error_return: + *tty->flip.flag_buf_ptr++ = flg; + *tty->flip.char_buf_ptr++ = ch; + tty->flip.count++; + ignore_char: + status = UTSR1_TO_SM(UART_GET_UTSR1(port)) | UTSR0_TO_SM(UART_GET_UTSR0(port)); + } +out: + tty_flip_buffer_push(tty); + return; + +handle_error: + if (status & UTSR1_TO_SM(UTSR1_PRE)) + port->icount.parity++; + else if (status & UTSR1_TO_SM(UTSR1_FRE)) + port->icount.frame++; + if (status & UTSR1_TO_SM(UTSR1_ROR)) + port->icount.overrun++; + + if (status & port->ignore_status_mask) { + if (++ignored > 100) + goto out; + goto ignore_char; + } + + status &= port->read_status_mask; + + if (status & UTSR1_TO_SM(UTSR1_PRE)) + flg = TTY_PARITY; + else if (status & UTSR1_TO_SM(UTSR1_FRE)) + flg = TTY_FRAME; + + if (status & UTSR1_TO_SM(UTSR1_ROR)) { + /* + * overrun does *not* affect the character + * we read from the FIFO + */ + *tty->flip.flag_buf_ptr++ = flg; + *tty->flip.char_buf_ptr++ = ch; + tty->flip.count++; + if (tty->flip.count >= TTY_FLIPBUF_SIZE) + goto ignore_char; + ch = 0; + flg = TTY_OVERRUN; + } +#ifdef SUPPORT_SYSRQ + info->sysrq = 0; +#endif + goto error_return; +} + +static void sa1100_tx_chars(struct uart_info *info) +{ + struct uart_port *port = info->port; + + if (port->x_char) { + UART_PUT_CHAR(port, port->x_char); + port->icount.tx++; + port->x_char = 0; + return; + } + if (info->xmit.head == info->xmit.tail + || info->tty->stopped + || info->tty->hw_stopped) { + sa1100_stop_tx(info->port, 0); + return; + } + + /* + * Tried using FIFO (not checking TNF) for fifo fill: + * still had the '4 bytes repeated' problem. + */ + while (UART_GET_UTSR1(port) & UTSR1_TNF) { + UART_PUT_CHAR(port, info->xmit.buf[info->xmit.tail]); + info->xmit.tail = (info->xmit.tail + 1) & (UART_XMIT_SIZE - 1); + port->icount.tx++; + if (info->xmit.head == info->xmit.tail) + break; + } + + if (CIRC_CNT(info->xmit.head, info->xmit.tail, UART_XMIT_SIZE) < + WAKEUP_CHARS) + uart_event(info, EVT_WRITE_WAKEUP); + + if (info->xmit.head == info->xmit.tail) + sa1100_stop_tx(info->port, 0); +} + +static void sa1100_int(int irq, void *dev_id, struct pt_regs *regs) +{ + struct uart_info *info = dev_id; + struct uart_port *port = info->port; + unsigned int status, pass_counter = 0; + + status = UART_GET_UTSR0(port); + status &= (SM_TO_UTSR0(port->read_status_mask) | ~UTSR0_TFS); + do { + if (status & (UTSR0_RFS | UTSR0_RID)) { + /* Clear the receiver idle bit, if set */ + if (status & UTSR0_RID) + UART_PUT_UTSR0(port, UTSR0_RID); + sa1100_rx_chars(info, regs); + } + + /* Clear the relevent break bits */ + if (status & (UTSR0_RBB | UTSR0_REB)) + UART_PUT_UTSR0(port, status & (UTSR0_RBB | UTSR0_REB)); + + if (status & UTSR0_RBB) + port->icount.brk++; + + if (status & UTSR0_REB) { +#ifdef SUPPORT_SYSRQ + if (port->line == sa1100_console.index && + !info->sysrq) { + info->sysrq = jiffies + HZ*5; + } +#endif + } + if (status & UTSR0_TFS) + sa1100_tx_chars(info); + if (pass_counter++ > SA1100_ISR_PASS_LIMIT) + break; + status = UART_GET_UTSR0(port); + status &= (SM_TO_UTSR0(port->read_status_mask) | ~UTSR0_TFS); + } while (status & (UTSR0_TFS | UTSR0_RFS | UTSR0_RID)); +} + +/* + * Return TIOCSER_TEMT when transmitter is not busy. + */ +static u_int sa1100_tx_empty(struct uart_port *port) +{ + return UART_GET_UTSR1(port) & UTSR1_TBY ? 0 : TIOCSER_TEMT; +} + +static u_int sa1100_get_mctrl(struct uart_port *port) +{ + return TIOCM_CTS | TIOCM_DSR | TIOCM_CAR; +} + +static void sa1100_set_mctrl(struct uart_port *port, u_int mctrl) +{ +} + +/* + * Interrupts always disabled. + */ +static void sa1100_break_ctl(struct uart_port *port, int break_state) +{ + u_int utcr3; + + utcr3 = UART_GET_UTCR3(port); + if (break_state == -1) + utcr3 |= UTCR3_BRK; + else + utcr3 &= ~UTCR3_BRK; + UART_PUT_UTCR3(port, utcr3); +} + +static int sa1100_startup(struct uart_port *port, struct uart_info *info) +{ + int retval; + + /* + * Allocate the IRQ + */ + retval = request_irq(port->irq, sa1100_int, 0, "serial_sa1100", info); + if (retval) + return retval; + + /* + * If there is a specific "open" function (to register + * control line interrupts) + */ + if (sa1100_open) { + retval = sa1100_open(port, info); + if (retval) { + free_irq(port->irq, info); + return retval; + } + } + + /* + * Finally, clear and enable interrupts + */ + UART_PUT_UTSR0(port, -1); + UART_PUT_UTCR3(port, UTCR3_RXE | UTCR3_TXE | UTCR3_RIE); + + return 0; +} + +static void sa1100_shutdown(struct uart_port *port, struct uart_info *info) +{ + /* + * Free the interrupt + */ + free_irq(port->irq, info); + + /* + * If there is a specific "close" function (to unregister + * control line interrupts) + */ + if (sa1100_close) + sa1100_close(port, info); + + /* + * Disable all interrupts, port and break condition. + */ + UART_PUT_UTCR3(port, 0); +} + +static void sa1100_change_speed(struct uart_port *port, u_int cflag, u_int iflag, u_int quot) +{ + unsigned long flags; + u_int utcr0, old_utcr3; + + /* byte size and parity */ + switch (cflag & CSIZE) { + case CS7: utcr0 = 0; break; + default: utcr0 = UTCR0_DSS; break; + } + if (cflag & CSTOPB) + utcr0 |= UTCR0_SBS; + if (cflag & PARENB) { + utcr0 |= UTCR0_PE; + if (!(cflag & PARODD)) + utcr0 |= UTCR0_OES; + } + + port->read_status_mask &= UTSR0_TO_SM(UTSR0_TFS); + port->read_status_mask |= UTSR1_TO_SM(UTSR1_ROR); + if (iflag & INPCK) + port->read_status_mask |= UTSR1_TO_SM(UTSR1_FRE | UTSR1_PRE); + if (iflag & (BRKINT | PARMRK)) + port->read_status_mask |= UTSR0_TO_SM(UTSR0_RBB | UTSR0_REB); + + /* + * Characters to ignore + */ + port->ignore_status_mask = 0; + if (iflag & IGNPAR) + port->ignore_status_mask |= UTSR1_TO_SM(UTSR1_FRE | UTSR1_PRE); + if (iflag & IGNBRK) { + port->ignore_status_mask |= UTSR0_TO_SM(UTSR0_RBB | UTSR0_REB); + /* + * If we're ignoring parity and break indicators, + * ignore overruns too (for real raw support). + */ + if (iflag & IGNPAR) + port->ignore_status_mask |= UTSR1_TO_SM(UTSR1_ROR); + } + + /* first, disable interrupts and drain transmitter */ + local_irq_save(flags); + old_utcr3 = UART_GET_UTCR3(port); + UART_PUT_UTCR3(port, old_utcr3 & ~(UTCR3_RIE | UTCR3_TIE)); + local_irq_restore(flags); + while (UART_GET_UTSR1(port) & UTSR1_TBY); + + /* then, disable everything */ + UART_PUT_UTCR3(port, 0); + + /* set the parity, stop bits and data size */ + UART_PUT_UTCR0(port, utcr0); + + /* set the baud rate */ + quot -= 1; + UART_PUT_UTCR1(port, ((quot & 0xf00) >> 8)); + UART_PUT_UTCR2(port, (quot & 0xff)); + + UART_PUT_UTSR0(port, -1); + + UART_PUT_UTCR3(port, old_utcr3); +} + +static const char *sa1100_type(struct uart_port *port) +{ + return port->type == PORT_SA1100 ? "SA1100" : NULL; +} + +/* + * Release the memory region(s) being used by 'port'. + */ +static void sa1100_release_port(struct uart_port *port) +{ + release_mem_region(port->mapbase, UART_PORT_SIZE); +} + +/* + * Request the memory region(s) being used by 'port'. + */ +static int sa1100_request_port(struct uart_port *port) +{ + return request_mem_region(port->mapbase, UART_PORT_SIZE, + "serial_sa1100") != NULL ? 0 : -EBUSY; +} + +/* + * Configure/autoconfigure the port. + */ +static void sa1100_config_port(struct uart_port *port, int flags) +{ + if (flags & UART_CONFIG_TYPE && sa1100_request_port(port) == 0) + port->type = PORT_SA1100; +} + +/* + * Verify the new serial_struct (for TIOCSSERIAL). + * The only change we allow are to the flags and type, and + * even then only between PORT_SA1100 and PORT_UNKNOWN + */ +static int sa1100_verify_port(struct uart_port *port, struct serial_struct *ser) +{ + int ret = 0; + if (ser->type != PORT_UNKNOWN && ser->type != PORT_SA1100) + ret = -EINVAL; + if (port->irq != ser->irq) + ret = -EINVAL; + if (ser->io_type != SERIAL_IO_MEM) + ret = -EINVAL; + if (port->uartclk / 16 != ser->baud_base) + ret = -EINVAL; + if ((void *)port->mapbase != ser->iomem_base) + ret = -EINVAL; + if (port->iobase != ser->port) + ret = -EINVAL; + if (ser->hub6 != 0) + ret = -EINVAL; + return ret; +} + +static struct uart_ops sa1100_pops = { + tx_empty: sa1100_tx_empty, + set_mctrl: sa1100_set_mctrl, + get_mctrl: sa1100_get_mctrl, + stop_tx: sa1100_stop_tx, + start_tx: sa1100_start_tx, + stop_rx: sa1100_stop_rx, + enable_ms: sa1100_enable_ms, + break_ctl: sa1100_break_ctl, + startup: sa1100_startup, + shutdown: sa1100_shutdown, + change_speed: sa1100_change_speed, + type: sa1100_type, + release_port: sa1100_release_port, + request_port: sa1100_request_port, + config_port: sa1100_config_port, + verify_port: sa1100_verify_port, +}; + +static struct uart_port sa1100_ports[NR_PORTS]; + +/* + * Setup the SA1100 serial ports. Note that we don't include the IrDA + * port here since we have our own SIR/FIR driver (see drivers/net/irda) + * + * Note also that we support "console=ttySAx" where "x" is either 0 or 1. + * Which serial port this ends up being depends on the machine you're + * running this kernel on. I'm not convinced that this is a good idea, + * but that's the way it traditionally works. + * + * Note that NanoEngine UART3 becomes UART2, and UART2 is no longer + * used here. + */ +static void sa1100_init_ports(void) +{ + static int first = 1; + int i; + + if (!first) + return; + first = 0; + + for (i = 0; i < NR_PORTS; i++) { + sa1100_ports[i].uartclk = 3686400; + sa1100_ports[i].ops = &sa1100_pops; + sa1100_ports[i].fifosize = 8; + } + + /* + * make transmit lines outputs, so that when the port + * is closed, the output is in the MARK state. + */ + PPDR |= PPC_TXD1 | PPC_TXD3; + PPSR |= PPC_TXD1 | PPC_TXD3; +} + +void __init sa1100_register_uart_fns(struct sa1100_port_fns *fns) +{ + if (fns->enable_ms) + sa1100_pops.enable_ms = fns->enable_ms; + if (fns->get_mctrl) + sa1100_pops.get_mctrl = fns->get_mctrl; + if (fns->set_mctrl) + sa1100_pops.set_mctrl = fns->set_mctrl; + sa1100_open = fns->open; + sa1100_close = fns->close; + sa1100_pops.pm = fns->pm; + sa1100_pops.set_wake = fns->set_wake; +} + +void __init sa1100_register_uart(int idx, int port) +{ + if (idx >= NR_PORTS) { + printk(KERN_ERR __FUNCTION__ ": bad index number %d\n", idx); + return; + } + + switch (port) { + case 1: + sa1100_ports[idx].membase = (void *)&Ser1UTCR0; + sa1100_ports[idx].mapbase = _Ser1UTCR0; + sa1100_ports[idx].irq = IRQ_Ser1UART; + sa1100_ports[idx].iotype = SERIAL_IO_MEM; + sa1100_ports[idx].flags = ASYNC_BOOT_AUTOCONF; + break; + + case 2: + sa1100_ports[idx].membase = (void *)&Ser2UTCR0; + sa1100_ports[idx].mapbase = _Ser2UTCR0; + sa1100_ports[idx].irq = IRQ_Ser2ICP; + sa1100_ports[idx].iotype = SERIAL_IO_MEM; + sa1100_ports[idx].flags = ASYNC_BOOT_AUTOCONF; + break; + + case 3: + sa1100_ports[idx].membase = (void *)&Ser3UTCR0; + sa1100_ports[idx].mapbase = _Ser3UTCR0; + sa1100_ports[idx].irq = IRQ_Ser3UART; + sa1100_ports[idx].iotype = SERIAL_IO_MEM; + sa1100_ports[idx].flags = ASYNC_BOOT_AUTOCONF; + break; + + default: + printk(KERN_ERR __FUNCTION__ ": bad port number %d\n", port); + } +} + + +#ifdef CONFIG_SERIAL_SA1100_CONSOLE + +/* + * Interrupts are disabled on entering + */ +static void sa1100_console_write(struct console *co, const char *s, u_int count) +{ + struct uart_port *port = sa1100_ports + co->index; + u_int old_utcr3, status, i; + + /* + * First, save UTCR3 and then disable interrupts + */ + old_utcr3 = UART_GET_UTCR3(port); + UART_PUT_UTCR3(port, (old_utcr3 & ~(UTCR3_RIE | UTCR3_TIE)) | UTCR3_TXE); + + /* + * Now, do each character + */ + for (i = 0; i < count; i++) { + do { + status = UART_GET_UTSR1(port); + } while (!(status & UTSR1_TNF)); + UART_PUT_CHAR(port, s[i]); + if (s[i] == '\n') { + do { + status = UART_GET_UTSR1(port); + } while (!(status & UTSR1_TNF)); + UART_PUT_CHAR(port, '\r'); + } + } + + /* + * Finally, wait for transmitter to become empty + * and restore UTCR3 + */ + do { + status = UART_GET_UTSR1(port); + } while (status & UTSR1_TBY); + UART_PUT_UTCR3(port, old_utcr3); +} + +static kdev_t sa1100_console_device(struct console *co) +{ + return MKDEV(SERIAL_SA1100_MAJOR, MINOR_START + co->index); +} + +/* + * If the port was already initialised (eg, by a boot loader), try to determine + * the current setup. + */ +static void __init +sa1100_console_get_options(struct uart_port *port, int *baud, int *parity, int *bits) +{ + u_int utcr3; + + utcr3 = UART_GET_UTCR3(port) & (UTCR3_RXE | UTCR3_TXE); + if (utcr3 == (UTCR3_RXE | UTCR3_TXE)) { + /* ok, the port was enabled */ + u_int utcr0, quot; + + utcr0 = UART_GET_UTCR0(port); + + *parity = 'n'; + if (utcr0 & UTCR0_PE) { + if (utcr0 & UTCR0_OES) + *parity = 'e'; + else + *parity = 'o'; + } + + if (utcr0 & UTCR0_DSS) + *bits = 8; + else + *bits = 7; + + quot = UART_GET_UTCR2(port) | UART_GET_UTCR1(port) << 8; + quot &= 0xfff; + *baud = port->uartclk / (16 * (quot + 1)); + } +} + +static int __init +sa1100_console_setup(struct console *co, char *options) +{ + struct uart_port *port; + int baud = CONFIG_SA1100_DEFAULT_BAUDRATE; + int bits = 8; + int parity = 'n'; + int flow = 'n'; + + /* + * Check whether an invalid uart number has been specified, and + * if so, search for the first available port that does have + * console support. + */ + port = uart_get_console(sa1100_ports, NR_PORTS, co); + + if (options) + uart_parse_options(options, &baud, &parity, &bits, &flow); + else + sa1100_console_get_options(port, &baud, &parity, &bits); + + return uart_set_options(port, co, baud, parity, bits, flow); +} + +static struct console sa1100_console = { + name: "ttySA", + write: sa1100_console_write, + device: sa1100_console_device, + setup: sa1100_console_setup, + flags: CON_PRINTBUFFER, + index: -1, +}; + +void __init sa1100_rs_console_init(void) +{ + sa1100_init_ports(); + register_console(&sa1100_console); +} + +#define SA1100_CONSOLE &sa1100_console +#else +#define SA1100_CONSOLE NULL +#endif + +static struct uart_driver sa1100_reg = { + owner: THIS_MODULE, + normal_major: SERIAL_SA1100_MAJOR, +#ifdef CONFIG_DEVFS_FS + normal_name: "ttySA%d", + callout_name: "cusa%d", +#else + normal_name: "ttySA", + callout_name: "cusa", +#endif + normal_driver: &normal, + callout_major: CALLOUT_SA1100_MAJOR, + callout_driver: &callout, + table: sa1100_table, + termios: sa1100_termios, + termios_locked: sa1100_termios_locked, + minor: MINOR_START, + nr: NR_PORTS, + port: sa1100_ports, + cons: SA1100_CONSOLE, +}; + +static int __init sa1100_serial_init(void) +{ + sa1100_init_ports(); + return uart_register_driver(&sa1100_reg); +} + +static void __exit sa1100_serial_exit(void) +{ + uart_unregister_driver(&sa1100_reg); +} + +module_init(sa1100_serial_init); +module_exit(sa1100_serial_exit); + +EXPORT_NO_SYMBOLS; + +MODULE_AUTHOR("Deep Blue Solutions Ltd"); +MODULE_DESCRIPTION("SA1100 generic serial port driver"); +MODULE_LICENSE("GPL"); + diff -urN orig/drivers/serial/uart00.c linux/drivers/serial/uart00.c --- orig/drivers/serial/uart00.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/serial/uart00.c Fri Feb 21 15:17:33 2003 @@ -0,0 +1,912 @@ +/* + * linux/drivers/char/serial_uart00.c + * + * Driver for UART00 serial ports + * + * Based on drivers/char/serial_amba.c, by ARM Limited & + * Deep Blue Solutions Ltd. + * Copyright 2001 Altera Corporation + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + * + * $Id: uart00.c,v 1.3.2.5 2002/10/24 09:53:26 rmk Exp $ + * + */ +#include +#include + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include + +#if defined(CONFIG_SERIAL_UART00_CONSOLE) && defined(CONFIG_MAGIC_SYSRQ) +#define SUPPORT_SYSRQ +#endif + +#include +#include +#define UART00_TYPE (volatile unsigned int*) +#include +#include + +#undef DEBUG +#define UART_NR 2 + +#define SERIAL_UART00_NAME "ttyUA" +#define SERIAL_UART00_MAJOR 204 +#define SERIAL_UART00_MINOR 16 /* Temporary - will change in future */ +#define SERIAL_UART00_NR UART_NR +#define UART_PORT_SIZE 0x50 + +#define CALLOUT_UART00_NAME "cuaua" +#define CALLOUT_UART00_MAJOR 205 +#define CALLOUT_UART00_MINOR 16 /* Temporary - will change in future */ +#define CALLOUT_UART00_NR UART_NR + +static struct tty_driver normal, callout; +static struct tty_struct *uart00_table[UART_NR]; +static struct termios *uart00_termios[UART_NR], *uart00_termios_locked[UART_NR]; +static struct console uart00_console; +static struct uart_driver uart00_reg; + + +#define UART00_ISR_PASS_LIMIT 256 + +/* + * Access macros for the UART00 UARTs + */ +#define UART_GET_INT_STATUS(p) inl(UART_ISR((p)->membase)) +#define UART_PUT_IES(p, c) outl(c,UART_IES((p)->membase)) +#define UART_GET_IES(p) inl(UART_IES((p)->membase)) +#define UART_PUT_IEC(p, c) outl(c,UART_IEC((p)->membase)) +#define UART_GET_IEC(p) inl(UART_IEC((p)->membase)) +#define UART_PUT_CHAR(p, c) outl(c,UART_TD((p)->membase)) +#define UART_GET_CHAR(p) inl(UART_RD((p)->membase)) +#define UART_GET_RSR(p) inl(UART_RSR((p)->membase)) +#define UART_GET_RDS(p) inl(UART_RDS((p)->membase)) +#define UART_GET_MSR(p) inl(UART_MSR((p)->membase)) +#define UART_GET_MCR(p) inl(UART_MCR((p)->membase)) +#define UART_PUT_MCR(p, c) outl(c,UART_MCR((p)->membase)) +#define UART_GET_MC(p) inl(UART_MC((p)->membase)) +#define UART_PUT_MC(p, c) outl(c,UART_MC((p)->membase)) +#define UART_GET_TSR(p) inl(UART_TSR((p)->membase)) +#define UART_GET_DIV_HI(p) inl(UART_DIV_HI((p)->membase)) +#define UART_PUT_DIV_HI(p,c) outl(c,UART_DIV_HI((p)->membase)) +#define UART_GET_DIV_LO(p) inl(UART_DIV_LO((p)->membase)) +#define UART_PUT_DIV_LO(p,c) outl(c,UART_DIV_LO((p)->membase)) +#define UART_RX_DATA(s) ((s) & UART_RSR_RX_LEVEL_MSK) +#define UART_TX_READY(s) (((s) & UART_TSR_TX_LEVEL_MSK) < 15) +//#define UART_TX_EMPTY(p) ((UART_GET_FR(p) & UART00_UARTFR_TMSK) == 0) + +static void uart00_stop_tx(struct uart_port *port, u_int from_tty) +{ + + UART_PUT_IEC(port, UART_IEC_TIE_MSK); +} + +static void uart00_stop_rx(struct uart_port *port) +{ + + UART_PUT_IEC(port, UART_IEC_RE_MSK); +} + +static void uart00_enable_ms(struct uart_port *port) +{ + + UART_PUT_IES(port, UART_IES_ME_MSK); +} + +static void +uart00_rx_chars(struct uart_info *info, struct pt_regs *regs) +{ + struct tty_struct *tty = info->tty; + unsigned int status, ch, rds, flg, ignored = 0; + struct uart_port *port = info->port; + + + status = UART_GET_RSR(port); + while (UART_RX_DATA(status)) { + + /* + * We need to read rds before reading the + * character from the fifo + */ + rds = UART_GET_RDS(port); + ch = UART_GET_CHAR(port); + port->icount.rx++; + + if (tty->flip.count >= TTY_FLIPBUF_SIZE) + goto ignore_char; + + flg = TTY_NORMAL; + + /* + * Note that the error handling code is + * out of the main execution path + */ + if (rds & (UART_RDS_BI_MSK |UART_RDS_FE_MSK| + UART_RDS_PE_MSK |UART_RDS_PE_MSK)) + goto handle_error; + if (uart_handle_sysrq_char(info, ch, regs)) + goto ignore_char; + + error_return: + *tty->flip.flag_buf_ptr++ = flg; + *tty->flip.char_buf_ptr++ = ch; + tty->flip.count++; + ignore_char: + status = UART_GET_RSR(port); + } +out: + tty_flip_buffer_push(tty); + return; + +handle_error: + if (rds & UART_RDS_BI_MSK) { + status &= ~(UART_RDS_FE_MSK | UART_RDS_PE_MSK); + port->icount.brk++; + +#ifdef SUPPORT_SYSRQ + if (uart_handle_break(info, &uart00_console)) + goto ignore_char; +#endif + } else if (rds & UART_RDS_PE_MSK) + port->icount.parity++; + else if (rds & UART_RDS_PE_MSK) + port->icount.frame++; + if (rds & UART_RDS_OE_MSK) + port->icount.overrun++; + + if (rds & port->ignore_status_mask) { + if (++ignored > 100) + goto out; + goto ignore_char; + } + rds &= port->read_status_mask; + + if (rds & UART_RDS_BI_MSK) + flg = TTY_BREAK; + else if (rds & UART_RDS_PE_MSK) + flg = TTY_PARITY; + else if (rds & UART_RDS_FE_MSK) + flg = TTY_FRAME; + + if (status & UART_RDS_OE_MSK) { + /* + * CHECK: does overrun affect the current character? + * ASSUMPTION: it does not. + */ + *tty->flip.flag_buf_ptr++ = flg; + *tty->flip.char_buf_ptr++ = ch; + tty->flip.count++; + if (tty->flip.count >= TTY_FLIPBUF_SIZE) + goto ignore_char; + ch = 0; + flg = TTY_OVERRUN; + } +#ifdef SUPPORT_SYSRQ + info->sysrq = 0; +#endif + goto error_return; +} + +static void uart00_tx_chars(struct uart_info *info) +{ + int count; + struct uart_port *port=info->port; + + if (port->x_char) { + while((UART_GET_TSR(port)& UART_TSR_TX_LEVEL_MSK)==15); + UART_PUT_CHAR(port, port->x_char); + port->icount.tx++; + port->x_char = 0; + + return; + } + if (info->xmit.head == info->xmit.tail + || info->tty->stopped + || info->tty->hw_stopped) { + uart00_stop_tx(info->port, 0); + return; + } + + count = port->fifosize >> 1; + do { + while((UART_GET_TSR(port)& UART_TSR_TX_LEVEL_MSK)==15); + UART_PUT_CHAR(port, info->xmit.buf[info->xmit.tail]); + info->xmit.tail = (info->xmit.tail + 1) & (UART_XMIT_SIZE - 1); + port->icount.tx++; + if (info->xmit.head == info->xmit.tail) + break; + } while (--count > 0); + + if (CIRC_CNT(info->xmit.head, + info->xmit.tail, + UART_XMIT_SIZE) < WAKEUP_CHARS) + uart_event(info, EVT_WRITE_WAKEUP); + + if (info->xmit.head == info->xmit.tail) + uart00_stop_tx(info->port, 0); +} + +static void uart00_start_tx(struct uart_port *port, u_int nonempty, u_int from_tty) +{ + struct uart_info *info=(struct uart_info*)(port->iobase); + + UART_PUT_IES(port,UART_IES_TIE_MSK ); + uart00_tx_chars(info); +} + +static void uart00_modem_status(struct uart_info *info) +{ + unsigned int status; + struct uart_icount *icount = &info->port->icount; + + + status = UART_GET_MSR(info->port); + + if (!status & (UART_MSR_DCTS_MSK | UART_MSR_DDSR_MSK | + UART_MSR_TERI_MSK | UART_MSR_DDCD_MSK)) + return; + + if (status & UART_MSR_DDCD_MSK) { + icount->dcd++; +#ifdef CONFIG_HARD_PPS + if ((info->flags & ASYNC_HARDPPS_CD) && + (status & UART_MSR_DCD_MSK)) + hardpps(); +#endif + if (info->flags & ASYNC_CHECK_CD) { + if (status & UART_MSR_DCD_MSK) + wake_up_interruptible(&info->open_wait); + else if (!((info->flags & ASYNC_CALLOUT_ACTIVE) && + (info->flags & ASYNC_CALLOUT_NOHUP))) { + if (info->tty) + tty_hangup(info->tty); + } + } + } + + if (status & UART_MSR_DDSR_MSK) + icount->dsr++; + + if (status & UART_MSR_DCTS_MSK) { + icount->cts++; + + if (info->flags & ASYNC_CTS_FLOW) { + status &= UART_MSR_CTS_MSK; + + if (info->tty->hw_stopped) { + if (status) { + info->tty->hw_stopped = 0; + info->ops->start_tx(info->port, 1, 0); + uart_event(info, EVT_WRITE_WAKEUP); + } + } else { + if (!status) { + info->tty->hw_stopped = 1; + info->ops->stop_tx(info->port, 0); + } + } + } + } + wake_up_interruptible(&info->delta_msr_wait); + +} + +static void uart00_int(int irq, void *dev_id, struct pt_regs *regs) +{ + struct uart_info *info = dev_id; + unsigned int status, pass_counter = 0; + + status = UART_GET_INT_STATUS(info->port); + do { + + if (status & UART_ISR_RI_MSK) + uart00_rx_chars(info, regs); + if (status & (UART_ISR_TI_MSK | UART_ISR_TII_MSK)) + uart00_tx_chars(info); + if (status & UART_ISR_MI_MSK) + uart00_modem_status(info); + if (pass_counter++ > UART00_ISR_PASS_LIMIT) + break; + + status = UART_GET_INT_STATUS(info->port); + } while (status); +} + +static u_int uart00_tx_empty(struct uart_port *port) +{ + return UART_GET_TSR(port) & UART_TSR_TX_LEVEL_MSK? 0 : TIOCSER_TEMT; +} + +static u_int uart00_get_mctrl(struct uart_port *port) +{ + unsigned int result = 0; + unsigned int status; + + status = UART_GET_MSR(port); + if (status & UART_MSR_DCD_MSK) + result |= TIOCM_CAR; + if (status & UART_MSR_DSR_MSK) + result |= TIOCM_DSR; + if (status & UART_MSR_CTS_MSK) + result |= TIOCM_CTS; + if (status & UART_MCR_RI_MSK) + result |= TIOCM_RNG; + + return result; +} + +static void uart00_set_mctrl(struct uart_port *port, u_int mctrl) +{ + unsigned char mcr = 0; + + if (mctrl & TIOCM_RTS) + mcr |= UART_MCR_RTS_MSK; + if (mctrl & TIOCM_DTR) + mcr |= UART_MCR_DTR_MSK; + if (mctrl & TIOCM_LOOP) + mcr |= UART_MCR_LB_MSK; + + UART_PUT_MCR(port, mcr); + } + +static void uart00_break_ctl(struct uart_port *port, int break_state) +{ + unsigned int mcr; + + mcr = UART_GET_MCR(port); + if (break_state == -1) + mcr |= UART_MCR_BR_MSK; + else + mcr &= ~UART_MCR_BR_MSK; + UART_PUT_MCR(port, mcr); +} + +static inline u_int uart_calculate_quot(struct uart_info *info, u_int baud) +{ + u_int quot; + + /* Special case: B0 rate */ + if (!baud) + baud = 9600; + + quot = (info->port->uartclk / (16 * baud)-1) ; + + return quot; +} +static void uart00_change_speed(struct uart_port *port, u_int cflag, u_int iflag, u_int quot) +{ + u_int uart_mc=0, old_ies; + unsigned long flags; + +#ifdef DEBUG + printk("uart00_set_cflag(0x%x) called\n", cflag); +#endif + /* byte size and parity */ + switch (cflag & CSIZE) { + case CS5: uart_mc = UART_MC_CLS_CHARLEN_5; break; + case CS6: uart_mc = UART_MC_CLS_CHARLEN_6; break; + case CS7: uart_mc = UART_MC_CLS_CHARLEN_7; break; + default: uart_mc = UART_MC_CLS_CHARLEN_8; break; // CS8 + } + if (cflag & CSTOPB) + uart_mc|= UART_MC_ST_TWO; + if (cflag & PARENB) { + uart_mc |= UART_MC_PE_MSK; + if (!(cflag & PARODD)) + uart_mc |= UART_MC_EP_MSK; + } + + port->read_status_mask = UART_RDS_OE_MSK; + if (iflag & INPCK) + port->read_status_mask |= UART_RDS_FE_MSK | UART_RDS_PE_MSK; + if (iflag & (BRKINT | PARMRK)) + port->read_status_mask |= UART_RDS_BI_MSK; + + /* + * Characters to ignore + */ + port->ignore_status_mask = 0; + if (iflag & IGNPAR) + port->ignore_status_mask |= UART_RDS_FE_MSK | UART_RDS_PE_MSK; + if (iflag & IGNBRK) { + port->ignore_status_mask |= UART_RDS_BI_MSK; + /* + * If we're ignoring parity and break indicators, + * ignore overruns to (for real raw support). + */ + if (iflag & IGNPAR) + port->ignore_status_mask |= UART_RDS_OE_MSK; + } + + /* first, disable everything */ + save_flags(flags); cli(); + old_ies = UART_GET_IES(port); + + if ((port->flags & ASYNC_HARDPPS_CD) || + (cflag & CRTSCTS) || !(cflag & CLOCAL)) + old_ies |= UART_IES_ME_MSK; + + + /* Set baud rate */ + UART_PUT_DIV_LO(port, (quot & 0xff)); + UART_PUT_DIV_HI(port, ((quot & 0xf00) >> 8)); + + + UART_PUT_MC(port, uart_mc); + UART_PUT_IES(port, old_ies); + + restore_flags(flags); +} + +static int uart00_startup(struct uart_port *port, struct uart_info *info) +{ + int retval; + + /* + * Use iobase to store a pointer to info. We need this to start a + * transmission as the tranmittr interrupt is only generated on + * the transition to the idle state + */ + + port->iobase=(u_int)info; + + /* + * Allocate the IRQ + */ + retval = request_irq(port->irq, uart00_int, 0, "uart00", info); + if (retval) + return retval; + + /* + * Finally, enable interrupts. Use the TII interrupt to minimise + * the number of interrupts generated. If higher performance is + * needed, consider using the TI interrupt with a suitable FIFO + * threshold + */ + UART_PUT_IES(port, UART_IES_RE_MSK | UART_IES_TIE_MSK); + + return 0; +} + +static void uart00_shutdown(struct uart_port *port, struct uart_info *info) +{ + /* + * disable all interrupts, disable the port + */ + UART_PUT_IEC(port, 0xff); + + /* disable break condition and fifos */ + UART_PUT_MCR(port, UART_GET_MCR(port) &~UART_MCR_BR_MSK); + + /* + * Free the interrupt + */ + free_irq(port->irq, info); +} + +static const char *uart00_type(struct uart_port *port) +{ + return port->type == PORT_UART00 ? "Altera UART00" : NULL; +} + +/* + * Release the memory region(s) being used by 'port' + */ +static void uart00_release_port(struct uart_port *port) +{ + release_mem_region(port->mapbase, UART_PORT_SIZE); + +#ifdef CONFIG_ARCH_CAMELOT + if(port->membase!=(void*)IO_ADDRESS(EXC_UART00_BASE)){ + iounmap(port->membase); + } +#endif +} + +/* + * Request the memory region(s) being used by 'port' + */ +static int uart00_request_port(struct uart_port *port) +{ + int result; + + result = request_mem_region(port->mapbase, UART_PORT_SIZE, + "serial_uart00") != NULL ? 0 : -EBUSY; + if (result) + return result; + + port->membase = ioremap(port->mapbase, SZ_4K); + if (!port->membase) { + printk(KERN_ERR "serial00: cannot map io memory\n"); + release_mem_region(port->mapbase, UART_PORT_SIZE); + } + + return port->membase ? 0 : -ENOMEM; +} + +/* + * Configure/autoconfigure the port. + */ +static void uart00_config_port(struct uart_port *port, int flags) +{ + if (flags & UART_CONFIG_TYPE) { + if (uart00_request_port(port) == 0) + port->type = PORT_UART00; + } +} + +/* + * verify the new serial_struct (for TIOCSSERIAL). + */ +static int uart00_verify_port(struct uart_port *port, struct serial_struct *ser) +{ + int ret = 0; + if (ser->type != PORT_UNKNOWN && ser->type != PORT_UART00) + ret = -EINVAL; + if (ser->irq < 0 || ser->irq >= NR_IRQS) + ret = -EINVAL; + if (ser->baud_base < 9600) + ret = -EINVAL; + return ret; +} + +static struct uart_ops uart00_pops = { + tx_empty: uart00_tx_empty, + set_mctrl: uart00_set_mctrl, + get_mctrl: uart00_get_mctrl, + stop_tx: uart00_stop_tx, + start_tx: uart00_start_tx, + stop_rx: uart00_stop_rx, + enable_ms: uart00_enable_ms, + break_ctl: uart00_break_ctl, + startup: uart00_startup, + shutdown: uart00_shutdown, + change_speed: uart00_change_speed, + type: uart00_type, + release_port: uart00_release_port, + request_port: uart00_request_port, + config_port: uart00_config_port, + verify_port: uart00_verify_port, +}; + +static struct uart_port uart00_ports[UART_NR] = { + +#ifdef CONFIG_ARCH_CAMELOT +{ + membase: (void*)IO_ADDRESS(EXC_UART00_BASE), + mapbase: EXC_UART00_BASE, + iotype: SERIAL_IO_MEM, + irq: IRQ_UART, + uartclk: EXC_AHB2_CLK_FREQUENCY, + fifosize: 16, + ops: &uart00_pops, + flags: ASYNC_BOOT_AUTOCONF, +} +#endif +}; + +#ifdef CONFIG_SERIAL_UART00_CONSOLE +static void uart00_console_write(struct console *co, const char *s, unsigned count) +{ +#ifdef CONFIG_ARCH_CAMELOT + struct uart_port *port = &uart00_ports[0]; + unsigned int status, old_ies; + int i; + + /* + * First save the CR then disable the interrupts + */ + old_ies = UART_GET_IES(port); + UART_PUT_IEC(port,0xff); + + /* + * Now, do each character + */ + for (i = 0; i < count; i++) { + do { + status = UART_GET_TSR(port); + } while (!UART_TX_READY(status)); + UART_PUT_CHAR(port, s[i]); + if (s[i] == '\n') { + do { + status = UART_GET_TSR(port); + } while (!UART_TX_READY(status)); + UART_PUT_CHAR(port, '\r'); + } + } + + /* + * Finally, wait for transmitter to become empty + * and restore the IES + */ + do { + status = UART_GET_TSR(port); + } while (status & UART_TSR_TX_LEVEL_MSK); + UART_PUT_IES(port, old_ies); +#endif +} + +static kdev_t uart00_console_device(struct console *co) +{ + return MKDEV(SERIAL_UART00_MAJOR, SERIAL_UART00_MINOR + co->index); +} + +static void /*__init*/ uart00_console_get_options(struct uart_port *port, int *baud, int *parity, int *bits) +{ + u_int uart_mc, quot; + uart_mc= UART_GET_MC(port); + + *parity = 'n'; + if (uart_mc & UART_MC_PE_MSK) { + if (uart_mc & UART_MC_EP_MSK) + *parity = 'e'; + else + *parity = 'o'; + } + + switch (uart_mc & UART_MC_CLS_MSK){ + + case UART_MC_CLS_CHARLEN_5: + *bits = 5; + break; + case UART_MC_CLS_CHARLEN_6: + *bits = 6; + break; + case UART_MC_CLS_CHARLEN_7: + *bits = 7; + break; + case UART_MC_CLS_CHARLEN_8: + *bits = 8; + break; + } + quot = UART_GET_DIV_LO(port) | (UART_GET_DIV_HI(port) << 8); + *baud = port->uartclk / (16 *quot ); +} + +static int __init uart00_console_setup(struct console *co, char *options) +{ + struct uart_port *port; + int baud = 38400; + int bits = 8; + int parity = 'n'; + int flow= 'n'; + +#ifdef CONFIG_ARCH_CAMELOT + /* + * Check whether an invalid uart number has been specified, and + * if so, search for the first available port that does have + * console support. + */ + port = &uart00_ports[0]; + co->index = 0; +#else + return -ENODEV; +#endif + + if (options) + uart_parse_options(options, &baud, &parity, &bits, &flow); + else + uart00_console_get_options(port, &baud, &parity, &bits); + + return uart_set_options(port, co, baud, parity, bits, flow); +} + +static struct console uart00_console = { + name: SERIAL_UART00_NAME, + write: uart00_console_write, + device: uart00_console_device, + setup: uart00_console_setup, + flags: CON_PRINTBUFFER, + index: 0, +}; + +void __init uart00_console_init(void) +{ + register_console(&uart00_console); +} + +#define UART00_CONSOLE &uart00_console +#else +#define UART00_CONSOLE NULL +#endif + +static struct uart_driver uart00_reg = { + owner: NULL, + normal_major: SERIAL_UART00_MAJOR, + normal_name: SERIAL_UART00_NAME, + normal_driver: &normal, + callout_major: CALLOUT_UART00_MAJOR, + callout_name: CALLOUT_UART00_NAME, + callout_driver: &callout, + table: uart00_table, + termios: uart00_termios, + termios_locked: uart00_termios_locked, + minor: SERIAL_UART00_MINOR, + nr: UART_NR, + port: uart00_ports, + state: NULL, + cons: UART00_CONSOLE, +}; + +struct dev_port_entry{ + struct uart_port *port; +}; + +static struct dev_port_entry dev_port_map[UART_NR]; + +#ifdef CONFIG_PLD_HOTSWAP +/* + * Keep a mapping of dev_info addresses -> port lines to use when + * removing ports dev==NULL indicates unused entry + */ + +struct uart00_ps_data{ + unsigned int clk; + unsigned int fifosize; +}; + +int uart00_add_device(struct pldhs_dev_info* dev_info, void* dev_ps_data) +{ + struct uart00_ps_data* dev_ps=dev_ps_data; + struct uart_port * port; + int i,result; + + i=0; + while(dev_port_map[i].port) + i++; + + if(i==UART_NR){ + printk(KERN_WARNING "uart00: Maximum number of ports reached\n"); + return 0; + } + + port=kmalloc(sizeof(struct uart_port),GFP_KERNEL); + if(!port) + return -ENOMEM; + + printk("clk=%d fifo=%d\n",dev_ps->clk,dev_ps->fifosize); + port->membase=0; + port->mapbase=dev_info->base_addr; + port->iotype=SERIAL_IO_MEM; + port->irq=dev_info->irq; + port->uartclk=dev_ps->clk; + port->fifosize=dev_ps->fifosize; + port->ops=&uart00_pops; + port->line=i; + port->flags=ASYNC_BOOT_AUTOCONF; + + result=uart_register_port(&uart00_reg, port); + if(result<0){ + printk("uart_register_port returned %d\n",result); + return result; + } + dev_port_map[i].port=port; + printk("uart00: added device at %lx as ttyUA%d\n",dev_port_map[i].port->mapbase,i); + return 0; + +} + +int uart00_remove_devices(void) +{ + int i,result; + + + result=0; + for(i=1;iuartclk,port->fifosize); + len+=PLDHS_READ_PROC_DATA(buf+len,"uart00",i, + port->mapbase,port->irq,ps_data); + + } + } + *eof=1; + return len; +} + + + +#endif +struct pld_hotswap_ops uart00_pldhs_ops={ + name: "uart00", + add_device: uart00_add_device, + remove_devices:uart00_remove_devices, + proc_read:uart00_proc_read + +}; + +#endif + +static int __init uart00_init(void) +{ + int ret; + int i; + + for(i=0;i + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License. + * + * This is the machine specific part of the Assabet/UDA1341 support. + * This driver makes use of the UDA1341 and the sa1100-audio modules. + * + * History: + * + * 2000-05-21 Nicolas Pitre Initial release. + * + * 2001-06-03 Nicolas Pitre Made this file a separate module, based on + * the former sa1100-uda1341.c driver. + * + * 2001-07-17 Nicolas Pitre Supports 44100Hz and 22050Hz samplerate now. + * + * 2001-08-03 Russell King Fix left/right channel swap. + * Attempt to reduce power consumption when idle. + * + * 2001-09-23 Russell King Remove old L3 bus driver. + * + * Please note that fiddling too much with MDREFR results in oopses, so we don't + * touch MDREFR unnecessarily, which means we don't touch it on close. + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include + +#include "sa1100-audio.h" + +/* + * Define this to fix the power drain on early Assabets + */ +#define FIX_POWER_DRAIN + +/* + * Debugging? + */ +#undef DEBUG + + +#ifdef DEBUG +#define DPRINTK( x... ) printk( ##x ) +#else +#define DPRINTK( x... ) +#endif + + +#define AUDIO_RATE_DEFAULT 44100 + +/* + * Mixer (UDA1341) interface + */ + +static struct l3_client uda1341; + +static int +mixer_ioctl(struct inode *inode, struct file *file, uint cmd, ulong arg) +{ + /* + * We only accept mixer (type 'M') ioctls. + */ + if (_IOC_TYPE(cmd) != 'M') + return -EINVAL; + + return l3_command(&uda1341, cmd, (void *)arg); +} + +static struct file_operations assabet_mixer_fops = { + ioctl: mixer_ioctl, + owner: THIS_MODULE +}; + + +/* + * Audio interface + */ +static long audio_samplerate = AUDIO_RATE_DEFAULT; + +/* + * FIXME: what about SFRM going high when SSP is disabled? + */ +static void assabet_set_samplerate(long val) +{ + struct uda1341_cfg cfg; + u_int clk_ref, clk_div; + + /* We don't want to mess with clocks when frames are in flight */ + Ser4SSCR0 &= ~SSCR0_SSE; + /* wait for any frame to complete */ + udelay(125); + + /* + * Our clock source is derived from the CPLD on which we don't have + * much control unfortunately. This was intended for a fixed 48000 Hz + * samplerate assuming a core clock of 221.2 MHz. The CPLD appears + * to divide the memory clock so there is a ratio of 4608 between + * the core clock and the resulting samplerate (obtained by + * measurements, the CPLD equations should confirm that). + * + * Still we can play with the SA1110's clock divisor for the SSP port + * to get half the samplerate as well. + * + * Apparently the clock sent to the SA1110 for the SSP port is further + * more divided from the clock sent to the UDA1341 (some people tried + * to be too clever in their design, or simply failed to read the + * SA1110 manual). If it was the same clock we would have been able + * to support a third samplerate with the UDA1341's 384FS mode. + * + * At least it would have been a minimum acceptable solution to be + * able to set the CPLD divisor by software. The iPAQ design is + * certainly a better example to follow for a new design. + */ + clk_ref = cpufreq_get(0) * 1000 / 4608; + if (val > clk_ref*4/7) { + audio_samplerate = clk_ref; + cfg.fs = 256; + clk_div = SSCR0_SerClkDiv(2); + } else { + audio_samplerate = clk_ref/2; + cfg.fs = 512; + clk_div = SSCR0_SerClkDiv(4); + } + + cfg.format = FMT_LSB16; + l3_command(&uda1341, L3_UDA1341_CONFIGURE, &cfg); + + Ser4SSCR0 = (Ser4SSCR0 & ~0xff00) + clk_div + SSCR0_SSE; +} + +/* + * Initialise the Assabet audio driver. + * + * Note that we have to be careful with the order that we do things here; + * there is a D-type flip flop which is clocked from the SFRM line which + * indicates whether the same is for the left or right channel to the + * UDA1341. + * + * When you disable the SSP (by clearing SSCR0_SSE) it appears that the + * SFRM signal can float high. When you re-enable the SSP, you clock the + * flip flop once, and end up swapping the left and right channels. + * + * The ASSABET_BCR_CODEC_RST line will force this flip flop into a known + * state, but this line resets other devices as well! + * + * In addition to the above, it appears that powering down the UDA1341 on + * early Assabets leaves the UDA_WS actively driving a logic '1' into the + * chip, wasting power! (you can tell this by D11 being half-on). We + * attempt to correct this by kicking the flip flop on init/open/close. + * We should probably do this on PM resume as well. + * + * (Note the ordering of ASSABET_BCR_AUDIO_ON, SFRM and ASSABET_BCR_CODEC_RST + * is important). + */ +static void assabet_audio_init(void *dummy) +{ + unsigned long flags; + unsigned int mdrefr; + + local_irq_save(flags); + + /* + * Enable the power for the UDA1341 before driving any signals. + * We leave the audio amp (LM4880) disabled for now. + */ + ASSABET_BCR_set(ASSABET_BCR_AUDIO_ON); + +#ifdef FIX_POWER_DRAIN + GPSR = GPIO_SSP_SFRM; + GPCR = GPIO_SSP_SFRM; +#endif + + ASSABET_BCR_set(ASSABET_BCR_CODEC_RST); + ASSABET_BCR_clear(ASSABET_BCR_STEREO_LB); + + /* + * Setup the SSP uart. + */ + PPAR |= PPAR_SPR; + Ser4SSCR0 = SSCR0_DataSize(16) + SSCR0_TI + SSCR0_SerClkDiv(2); + Ser4SSCR1 = SSCR1_SClkIactL + SSCR1_SClk1P + SSCR1_ExtClk; + GAFR |= GPIO_SSP_TXD | GPIO_SSP_RXD | GPIO_SSP_SCLK | GPIO_SSP_CLK; + GPDR |= GPIO_SSP_TXD | GPIO_SSP_SCLK | GPIO_SSP_SFRM; + GPDR &= ~(GPIO_SSP_RXD | GPIO_SSP_CLK); + Ser4SSCR0 |= SSCR0_SSE; + + /* + * Only give SFRM to the SSP after it has been enabled. + */ + GAFR |= GPIO_SSP_SFRM; + + /* + * The assabet board uses the SDRAM clock as the source clock for + * audio. This is supplied to the SA11x0 from the CPLD on pin 19. + * At 206MHz we need to run the audio clock (SDRAM bank 2) + * at half speed. This clock will scale with core frequency so + * the audio sample rate will also scale. The CPLD on Assabet + * will need to be programmed to match the core frequency. + */ + mdrefr = MDREFR; + if ((mdrefr & (MDREFR_K2DB2 | MDREFR_K2RUN | MDREFR_EAPD | + MDREFR_KAPD)) != (MDREFR_K2DB2 | MDREFR_K2RUN)) { + mdrefr |= MDREFR_K2DB2 | MDREFR_K2RUN; + mdrefr &= ~(MDREFR_EAPD | MDREFR_KAPD); + MDREFR = mdrefr; + (void) MDREFR; + } + local_irq_restore(flags); + + /* Wait for the UDA1341 to wake up */ + mdelay(1); + + l3_open(&uda1341); + + assabet_set_samplerate(audio_samplerate); + + /* Enable the audio power */ + ASSABET_BCR_clear(ASSABET_BCR_QMUTE | ASSABET_BCR_SPK_OFF); +} + +/* + * Shutdown the Assabet audio driver. + * + * We have to be careful about the SFRM line here for the same reasons + * described in the initialisation comments above. This basically means + * that we must hand the SSP pins back to the GPIO module before disabling + * the SSP. + * + * In addition, to reduce power drain, we toggle the SFRM line once so + * that the UDA_WS line is at logic 0. + * + * We can't clear ASSABET_BCR_CODEC_RST without knowing if the UCB1300 or + * ADV7171 driver is still active. If it is, then we still need to play + * games, so we might as well leave ASSABET_BCR_CODEC_RST set. + */ +static void assabet_audio_shutdown(void *dummy) +{ + ASSABET_BCR_set(ASSABET_BCR_STEREO_LB | ASSABET_BCR_QMUTE | + ASSABET_BCR_SPK_OFF); + + l3_close(&uda1341); + + GAFR &= ~(GPIO_SSP_TXD | GPIO_SSP_RXD | GPIO_SSP_SCLK | GPIO_SSP_SFRM); + Ser4SSCR0 = 0; + +#ifdef FIX_POWER_DRAIN + GPSR = GPIO_SSP_SFRM; + GPCR = GPIO_SSP_SFRM; +#endif + + /* disable the audio power */ + ASSABET_BCR_clear(ASSABET_BCR_AUDIO_ON); +} + +static int assabet_audio_ioctl( struct inode *inode, struct file *file, + uint cmd, ulong arg) +{ + long val; + int ret = 0; + + /* + * These are platform dependent ioctls which are not handled by the + * generic sa1100-audio module. + */ + switch (cmd) { + case SNDCTL_DSP_STEREO: + ret = get_user(val, (int *) arg); + if (ret) + return ret; + /* the UDA1341 is stereo only */ + ret = (val == 0) ? -EINVAL : 1; + return put_user(ret, (int *) arg); + + case SNDCTL_DSP_CHANNELS: + case SOUND_PCM_READ_CHANNELS: + /* the UDA1341 is stereo only */ + return put_user(2, (long *) arg); + + case SNDCTL_DSP_SPEED: + ret = get_user(val, (long *) arg); + if (ret) break; + assabet_set_samplerate(val); + /* fall through */ + + case SOUND_PCM_READ_RATE: + return put_user(audio_samplerate, (long *) arg); + + case SNDCTL_DSP_SETFMT: + case SNDCTL_DSP_GETFMTS: + /* we can do signed 16-bit only */ + return put_user(AFMT_S16_LE, (long *) arg); + + default: + /* Maybe this is meant for the mixer (As per OSS Docs) */ + return mixer_ioctl(inode, file, cmd, arg); + } + + return ret; +} + +static audio_stream_t output_stream, input_stream; + +static audio_state_t audio_state = { + output_stream: &output_stream, + output_dma: DMA_Ser4SSPWr, + output_id: "Assabet UDA1341 out", + input_stream: &input_stream, + input_dma: DMA_Ser4SSPRd, + input_id: "Assabet UDA1341 in", + need_tx_for_rx: 1, + hw_init: assabet_audio_init, + hw_shutdown: assabet_audio_shutdown, + client_ioctl: assabet_audio_ioctl, + sem: __MUTEX_INITIALIZER(audio_state.sem), +}; + +static int assabet_audio_open(struct inode *inode, struct file *file) +{ + return sa1100_audio_attach(inode, file, &audio_state); +} + +/* + * Missing fields of this structure will be patched with the call + * to sa1100_audio_attach(). + */ +static struct file_operations assabet_audio_fops = { + open: assabet_audio_open, + owner: THIS_MODULE +}; + + +static int audio_dev_id, mixer_dev_id; + +static int __init assabet_uda1341_init(void) +{ + int ret; + + if (!machine_is_assabet()) + return -ENODEV; + + ret = l3_attach_client(&uda1341, "l3-bit-sa1100-gpio", "uda1341"); + if (ret) + goto out; + + /* register devices */ + audio_dev_id = register_sound_dsp(&assabet_audio_fops, -1); + mixer_dev_id = register_sound_mixer(&assabet_mixer_fops, -1); + +#ifdef FIX_POWER_DRAIN + { + unsigned long flags; + local_irq_save(flags); + ASSABET_BCR_set(ASSABET_BCR_CODEC_RST); + GPSR = GPIO_SSP_SFRM; + GPDR |= GPIO_SSP_SFRM; + GPCR = GPIO_SSP_SFRM; + local_irq_restore(flags); + } +#endif + + printk(KERN_INFO "Sound: Assabet UDA1341: dsp id %d mixer id %d\n", + audio_dev_id, mixer_dev_id); + return 0; + +release_l3: + l3_detach_client(&uda1341); +out: + return ret; +} + +static void __exit assabet_uda1341_exit(void) +{ + unregister_sound_dsp(audio_dev_id); + unregister_sound_mixer(mixer_dev_id); + l3_detach_client(&uda1341); +} + +module_init(assabet_uda1341_init); +module_exit(assabet_uda1341_exit); + +MODULE_AUTHOR("Nicolas Pitre"); +MODULE_DESCRIPTION("Glue audio driver for the SA1110 Assabet board & Philips UDA1341 codec."); + +EXPORT_NO_SYMBOLS; diff -urN orig/drivers/sound/h3600-uda1341.c linux/drivers/sound/h3600-uda1341.c --- orig/drivers/sound/h3600-uda1341.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/sound/h3600-uda1341.c Thu Nov 29 23:07:46 2001 @@ -0,0 +1,352 @@ +/* + * Glue audio driver for the Compaq iPAQ H3600 & Philips UDA1341 codec. + * + * Copyright (c) 2000 Nicolas Pitre + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License. + * + * This is the machine specific part of the Compaq iPAQ (aka Bitsy) support. + * This driver makes use of the UDA1341 and the sa1100-audio modules. + * + * History: + * + * 2000-05-21 Nicolas Pitre Initial UDA1341 driver release. + * + * 2000-07-?? George France Bitsy support. + * + * 2000-12-13 Deborah Wallach Fixed power handling for iPAQ/h3600 + * + * 2001-06-03 Nicolas Pitre Made this file a separate module, based on + * the former sa1100-uda1341.c driver. + * + * 2001-07-13 Nicolas Pitre Fixes for all supported samplerates. + * + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +//#include + +#include "sa1100-audio.h" + + +#undef DEBUG +#ifdef DEBUG +#define DPRINTK( x... ) printk( ##x ) +#else +#define DPRINTK( x... ) +#endif + + +#define AUDIO_NAME "Bitsy_UDA1341" + +#define AUDIO_RATE_DEFAULT 44100 + + +static struct l3_client uda1341; + +static int +mixer_ioctl(struct inode *inode, struct file *file, uint cmd, ulong arg) +{ + /* + * We only accept mixer (type 'M') ioctls. + */ + if (_IOC_TYPE(cmd) != 'M') + return -EINVAL; + + return l3_command(&uda1341, cmd, (void *)arg); +} + +static struct file_operations h3600_mixer_fops = { + ioctl: mixer_ioctl, + owner: THIS_MODULE +}; + + +/* + * Audio interface + */ + +static long audio_samplerate = AUDIO_RATE_DEFAULT; + +/* + * Stop-gap solution until rest of hh.org HAL stuff is merged. + */ +#define GPIO_H3600_CLK_SET0 GPIO_GPIO (12) +#define GPIO_H3600_CLK_SET1 GPIO_GPIO (13) +static void h3600_set_audio_clock(long val) +{ + switch (val) { + case 24000: case 32000: case 48000: /* 00: 12.288 MHz */ + GPCR = GPIO_H3600_CLK_SET0 | GPIO_H3600_CLK_SET1; + break; + + case 22050: case 29400: case 44100: /* 01: 11.2896 MHz */ + GPSR = GPIO_H3600_CLK_SET0; + GPCR = GPIO_H3600_CLK_SET1; + break; + + case 8000: case 10666: case 16000: /* 10: 4.096 MHz */ + GPCR = GPIO_H3600_CLK_SET0; + GPSR = GPIO_H3600_CLK_SET1; + break; + + case 10985: case 14647: case 21970: /* 11: 5.6245 MHz */ + GPSR = GPIO_H3600_CLK_SET0 | GPIO_H3600_CLK_SET1; + break; + } +} + +static void h3600_set_samplerate(long val) +{ + struct uda1341_cfg cfg; + int clk_div = 0; + + /* We don't want to mess with clocks when frames are in flight */ + Ser4SSCR0 &= ~SSCR0_SSE; + /* wait for any frame to complete */ + udelay(125); + + /* + * We have the following clock sources: + * 4.096 MHz, 5.6245 MHz, 11.2896 MHz, 12.288 MHz + * Those can be divided either by 256, 384 or 512. + * This makes up 12 combinations for the following samplerates... + */ + if (val >= 48000) + val = 48000; + else if (val >= 44100) + val = 44100; + else if (val >= 32000) + val = 32000; + else if (val >= 29400) + val = 29400; + else if (val >= 24000) + val = 24000; + else if (val >= 22050) + val = 22050; + else if (val >= 21970) + val = 21970; + else if (val >= 16000) + val = 16000; + else if (val >= 14647) + val = 14647; + else if (val >= 10985) + val = 10985; + else if (val >= 10666) + val = 10666; + else + val = 8000; + + /* Set the external clock generator */ + h3600_set_audio_clock(val); + + /* Select the clock divisor */ + switch (val) { + case 8000: + case 10985: + case 22050: + case 24000: + cfg.fs = 512; + clk_div = SSCR0_SerClkDiv(16); + break; + case 16000: + case 21970: + case 44100: + case 48000: + cfg.fs = 256; + clk_div = SSCR0_SerClkDiv(8); + break; + case 10666: + case 14647: + case 29400: + case 32000: + cfg.fs = 384; + clk_div = SSCR0_SerClkDiv(12); + break; + } + + cfg.format = FMT_LSB16; + l3_command(&uda1341, L3_UDA1341_CONFIGURE, &cfg); + Ser4SSCR0 = (Ser4SSCR0 & ~0xff00) + clk_div + SSCR0_SSE; + audio_samplerate = val; +} + +static void h3600_audio_init(void *dummy) +{ + unsigned long flags; + + /* Setup the uarts */ + local_irq_save(flags); + GAFR |= (GPIO_SSP_CLK); + GPDR &= ~(GPIO_SSP_CLK); + Ser4SSCR0 = 0; + Ser4SSCR0 = SSCR0_DataSize(16) + SSCR0_TI + SSCR0_SerClkDiv(8); + Ser4SSCR1 = SSCR1_SClkIactL + SSCR1_SClk1P + SSCR1_ExtClk; + Ser4SSCR0 |= SSCR0_SSE; + + /* Enable the audio power */ + + clr_h3600_egpio(IPAQ_EGPIO_CODEC_NRESET); + set_h3600_egpio(IPAQ_EGPIO_AUDIO_ON); + set_h3600_egpio(IPAQ_EGPIO_QMUTE); + local_irq_restore(flags); + + /* external clock configuration */ + h3600_set_samplerate(audio_samplerate); + + /* Wait for the UDA1341 to wake up */ + set_h3600_egpio(IPAQ_EGPIO_CODEC_NRESET); + mdelay(1); + + /* make the left and right channels unswapped (flip the WS latch ) */ + Ser4SSDR = 0; + + /* Initialize the UDA1341 internal state */ + l3_open(&uda1341); + + clr_h3600_egpio(IPAQ_EGPIO_QMUTE); +} + +static void h3600_audio_shutdown(void *dummy) +{ + /* disable the audio power and all signals leading to the audio chip */ + l3_close(&uda1341); + Ser4SSCR0 = 0; + clr_h3600_egpio(IPAQ_EGPIO_CODEC_NRESET); + clr_h3600_egpio(IPAQ_EGPIO_AUDIO_ON); + clr_h3600_egpio(IPAQ_EGPIO_QMUTE); +} + +static int h3600_audio_ioctl(struct inode *inode, struct file *file, + uint cmd, ulong arg) +{ + long val; + int ret = 0; + + /* + * These are platform dependent ioctls which are not handled by the + * generic sa1100-audio module. + */ + switch (cmd) { + case SNDCTL_DSP_STEREO: + ret = get_user(val, (int *) arg); + if (ret) + return ret; + /* the UDA1341 is stereo only */ + ret = (val == 0) ? -EINVAL : 1; + return put_user(ret, (int *) arg); + + case SNDCTL_DSP_CHANNELS: + case SOUND_PCM_READ_CHANNELS: + /* the UDA1341 is stereo only */ + return put_user(2, (long *) arg); + + case SNDCTL_DSP_SPEED: + ret = get_user(val, (long *) arg); + if (ret) break; + h3600_set_samplerate(val); + /* fall through */ + + case SOUND_PCM_READ_RATE: + return put_user(audio_samplerate, (long *) arg); + + case SNDCTL_DSP_SETFMT: + case SNDCTL_DSP_GETFMTS: + /* we can do 16-bit only */ + return put_user(AFMT_S16_LE, (long *) arg); + + default: + /* Maybe this is meant for the mixer (As per OSS Docs) */ + return mixer_ioctl(inode, file, cmd, arg); + } + + return ret; +} + +static audio_stream_t output_stream, input_stream; + +static audio_state_t audio_state = { + output_stream: &output_stream, + output_dma: DMA_Ser4SSPWr, + output_id: "UDA1341 out", + input_stream: &input_stream, + input_dma: DMA_Ser4SSPRd, + input_id: "UDA1341 in", + need_tx_for_rx: 1, + hw_init: h3600_audio_init, + hw_shutdown: h3600_audio_shutdown, + client_ioctl: h3600_audio_ioctl, + sem: __MUTEX_INITIALIZER(audio_state.sem), +}; + +static int h3600_audio_open(struct inode *inode, struct file *file) +{ + return sa1100_audio_attach(inode, file, &audio_state); +} + +/* + * Missing fields of this structure will be patched with the call + * to sa1100_audio_attach(). + */ +static struct file_operations h3600_audio_fops = { + open: h3600_audio_open, + owner: THIS_MODULE +}; + + +static int audio_dev_id, mixer_dev_id; + +static int __init h3600_uda1341_init(void) +{ + int ret; + + if (!machine_is_h3xxx()) + return -ENODEV; + + ret = l3_attach_client(&uda1341, "l3-bit-sa1100-gpio", "uda1341"); + if (ret) + goto out; + + /* register devices */ + audio_dev_id = register_sound_dsp(&h3600_audio_fops, -1); + mixer_dev_id = register_sound_mixer(&h3600_mixer_fops, -1); + + printk( KERN_INFO "iPAQ audio support initialized\n" ); + return 0; + +release_l3: + l3_detach_client(&uda1341); +out: + return ret; +} + +static void __exit h3600_uda1341_exit(void) +{ + unregister_sound_dsp(audio_dev_id); + unregister_sound_mixer(mixer_dev_id); + l3_detach_client(&uda1341); +} + +module_init(h3600_uda1341_init); +module_exit(h3600_uda1341_exit); + +MODULE_AUTHOR("Nicolas Pitre, George France"); +MODULE_DESCRIPTION("Glue audio driver for the Compaq iPAQ H3600 & Philips UDA1341 codec."); + +EXPORT_NO_SYMBOLS; diff -urN orig/drivers/sound/pangolin-uda1341.c linux/drivers/sound/pangolin-uda1341.c --- orig/drivers/sound/pangolin-uda1341.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/sound/pangolin-uda1341.c Wed Nov 14 18:08:22 2001 @@ -0,0 +1,322 @@ +/* + * Glue audio driver for the SA1110 Pangolin board & Philips UDA1341 codec. + * + * Copyright (c) 2000 Nicolas Pitre + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License. + * + * This is the machine specific part of the Pangolin/UDA1341 support. + * This driver makes use of the UDA1341 and the sa1100-audio modules. + * + * History: + * + * 2000-05-21 Nicolas Pitre Initial release. + * + * 2001-06-03 Nicolas Pitre Made this file a separate module, based on + * the former sa1100-uda1341.c driver. + * + * 2001-07-17 Nicolas Pitre Supports 44100Hz and 22050Hz samplerate now. + * + * 2001-08-06 Richard Fan Pangolin Support + * + * 2001-09-23 Russell King Update inline with Assabet driver + * Remove old L3 bus driver + * + * Note: this should probably be merged with the Assabet audio driver, + * and become the "SDRAM-clock driven" SA1100 audio driver. + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include + +#include "sa1100-audio.h" + +/* + * Debugging? + */ +#undef DEBUG + + +#ifdef DEBUG +#define DPRINTK( x... ) printk( ##x ) +#else +#define DPRINTK( x... ) +#endif + + +#define AUDIO_RATE_DEFAULT 44100 + +#define QmutePin GPIO_GPIO(4) +#define SpeakerOffPin GPIO_GPIO(5) + +/* + * Mixer (UDA1341) interface + */ + +static struct l3_client uda1341; + +static int +mixer_ioctl(struct inode *inode, struct file *file, uint cmd, ulong arg) +{ + /* + * We only accept mixer (type 'M') ioctls. + */ + if (_IOC_TYPE(cmd) != 'M') + return -EINVAL; + + return l3_command(&uda1341, cmd, (void *)arg); +} + +static struct file_operations pangolin_mixer_fops = { + ioctl: mixer_ioctl, + owner: THIS_MODULE +}; + + +/* + * Audio interface + */ +static long audio_samplerate = AUDIO_RATE_DEFAULT; + +static void pangolin_set_samplerate(long val) +{ + struct uda1341_cfg cfg; + int clk_div; + + /* We don't want to mess with clocks when frames are in flight */ + Ser4SSCR0 &= ~SSCR0_SSE; + /* wait for any frame to complete */ + udelay(125); + + /* + * Our clock source is derived from the CPLD on which we don't have + * much control unfortunately. This was intended for a fixed 44100Hz + * samplerate assuming a core clock of 206 MHz. Still we can play + * with the SA1110's clock divisor for the SSP port to get a 22050Hz + * samplerate. + * + * Apparently the clock sent to the SA1110 for the SSP port is + * divided from the clock sent to the UDA1341 (some people tried to + * be too clever in their design, or simply failed to read the SA1110 + * manual). If it was the same source we would have been able to + * support a third samplerate. + * + * At least it would have been a minimum acceptable solution to be + * able to set the CPLD divisor by software. The iPAQ design is + * certainly a better example to follow for a new design. + */ + if (val >= 44100) { + audio_samplerate = 44100; + cfg.fs = 256; + clk_div = SSCR0_SerClkDiv(2); + } else { + audio_samplerate = 22050; + cfg.fs = 512; + clk_div = SSCR0_SerClkDiv(4); + } + + cfg.format = FMT_LSB16; + l3_command(&uda1341, L3_UDA1341_CONFIGURE, &cfg); + + Ser4SSCR0 = (Ser4SSCR0 & ~0xff00) + clk_div + SSCR0_SSE; +} + +static void pangolin_audio_init(void *dummy) +{ + unsigned long flags; + unsigned int mdrefr; + + local_irq_save(flags); + + /* + * Setup the SSP uart. + */ + PPAR |= PPAR_SPR; + Ser4SSCR0 = SSCR0_DataSize(16) + SSCR0_TI + SSCR0_SerClkDiv(2); + Ser4SSCR1 = SSCR1_SClkIactL + SSCR1_SClk1P + SSCR1_ExtClk; + GAFR |= GPIO_SSP_TXD | GPIO_SSP_RXD | GPIO_SSP_SCLK | GPIO_SSP_CLK | + GPIO_SSP_SFRM; + GPDR |= GPIO_SSP_TXD | GPIO_SSP_SCLK | GPIO_SSP_SFRM; + GPDR &= ~(GPIO_SSP_RXD | GPIO_SSP_CLK); + Ser4SSCR0 |= SSCR0_SSE; + + GAFR &= ~(SpeakerOffPin | QmutePin); + GPDR |= (SpeakerOffPin | QmutePin); + GPCR = SpeakerOffPin; + + /* + * The assabet board uses the SDRAM clock as the source clock for + * audio. This is supplied to the SA11x0 from the CPLD on pin 19. + * At 206MHz we need to run the audio clock (SDRAM bank 2) + * at half speed. This clock will scale with core frequency so + * the audio sample rate will also scale. The CPLD on Assabet + * will need to be programmed to match the core frequency. + */ + mdrefr = MDREFR; + if ((mdrefr & (MDREFR_K2DB2 | MDREFR_K2RUN | MDREFR_EAPD | + MDREFR_KAPD)) != (MDREFR_K2DB2 | MDREFR_K2RUN)) { + mdrefr |= MDREFR_K2DB2 | MDREFR_K2RUN; + mdrefr &= ~(MDREFR_EAPD | MDREFR_KAPD); + MDREFR = mdrefr; + (void) MDREFR; + } + local_irq_restore(flags); + + /* Wait for the UDA1341 to wake up */ + mdelay(100); + + l3_open(&uda1341); + + pangolin_set_samplerate(audio_samplerate); + + GPCR = QmutePin; +} + +static void pangolin_audio_shutdown(void *dummy) +{ + GPSR = QmutePin; + + l3_close(&uda1341); + + Ser4SSCR0 = 0; + MDREFR &= ~(MDREFR_K2DB2 | MDREFR_K2RUN); +} + +static int pangolin_audio_ioctl( struct inode *inode, struct file *file, + uint cmd, ulong arg) +{ + long val; + int ret = 0; + + /* + * These are platform dependent ioctls which are not handled by the + * generic sa1100-audio module. + */ + switch (cmd) { + case SNDCTL_DSP_STEREO: + ret = get_user(val, (int *) arg); + if (ret) + return ret; + /* the UDA1341 is stereo only */ + ret = (val == 0) ? -EINVAL : 1; + return put_user(ret, (int *) arg); + + case SNDCTL_DSP_CHANNELS: + case SOUND_PCM_READ_CHANNELS: + /* the UDA1341 is stereo only */ + return put_user(2, (long *) arg); + + case SNDCTL_DSP_SPEED: + ret = get_user(val, (long *) arg); + if (ret) break; + pangolin_set_samplerate(val); + /* fall through */ + + case SOUND_PCM_READ_RATE: + return put_user(audio_samplerate, (long *) arg); + + case SNDCTL_DSP_SETFMT: + case SNDCTL_DSP_GETFMTS: + /* we can do signed 16-bit only */ + return put_user(AFMT_S16_LE, (long *) arg); + + default: + /* Maybe this is meant for the mixer (As per OSS Docs) */ + return mixer_ioctl(inode, file, cmd, arg); + } + + return ret; +} + +static audio_stream_t output_stream, input_stream; + +static audio_state_t audio_state = { + output_stream: &output_stream, + output_dma: DMA_Ser4SSPWr, + output_id: "Pangolin UDA1341 out", + input_stream: &input_stream, + input_dma: DMA_Ser4SSPRd, + input_id: "Pangolin UDA1341 in", + need_tx_for_rx: 1, + hw_init: pangolin_audio_init, + hw_shutdown: pangolin_audio_shutdown, + client_ioctl: pangolin_audio_ioctl, + sem: __MUTEX_INITIALIZER(audio_state.sem), +}; + +static int pangolin_audio_open(struct inode *inode, struct file *file) +{ + return sa1100_audio_attach(inode, file, &audio_state); +} + +/* + * Missing fields of this structure will be patched with the call + * to sa1100_audio_attach(). + */ +static struct file_operations pangolin_audio_fops = { + open: pangolin_audio_open, + owner: THIS_MODULE +}; + + +static int audio_dev_id, mixer_dev_id; + +static int __init pangolin_uda1341_init(void) +{ + unsigned long flags; + int ret; + + if (!machine_is_pangolin()) + return -ENODEV; + + ret = l3_attach_client(&uda1341, "l3-bit-sa1100-gpio", "uda1341"); + if (ret) + goto out; + + /* register devices */ + audio_dev_id = register_sound_dsp(&pangolin_audio_fops, -1); + mixer_dev_id = register_sound_mixer(&pangolin_mixer_fops, -1); + + local_irq_save(flags); + GAFR &= ~(SpeakerOffPin | QmutePin); + GPDR |= (SpeakerOffPin | QmutePin); + local_irq_restore(flags); + + printk(KERN_INFO "Pangolin UDA1341 audio driver initialized\n"); + return 0; + +release_l3: + l3_detach_client(&uda1341); +out: + return ret; +} + +static void __exit pangolin_uda1341_exit(void) +{ + unregister_sound_dsp(audio_dev_id); + unregister_sound_mixer(mixer_dev_id); + l3_detach_client(&uda1341); +} + +module_init(pangolin_uda1341_init); +module_exit(pangolin_uda1341_exit); + +MODULE_AUTHOR("Nicolas Pitre"); +MODULE_DESCRIPTION("Glue audio driver for the SA1110 Pangolin board & Philips UDA1341 codec."); + +EXPORT_NO_SYMBOLS; diff -urN orig/drivers/sound/sa1100-audio.c linux/drivers/sound/sa1100-audio.c --- orig/drivers/sound/sa1100-audio.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/sound/sa1100-audio.c Fri Dec 13 14:09:01 2002 @@ -0,0 +1,976 @@ +/* + * Common audio handling for the SA11x0 processor + * + * Copyright (C) 2000, 2001 Nicolas Pitre + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License. + * + * + * This module handles the generic buffering/DMA/mmap audio interface for + * codecs connected to the SA1100 chip. All features depending on specific + * hardware implementations like supported audio formats or samplerates are + * relegated to separate specific modules. + * + * + * History: + * + * 2000-05-21 Nicolas Pitre Initial release. + * + * 2000-06-10 Erik Bunce Add initial poll support. + * + * 2000-08-22 Nicolas Pitre Removed all DMA stuff. Now using the + * generic SA1100 DMA interface. + * + * 2000-11-30 Nicolas Pitre - Validation of opened instances; + * - Power handling at open/release time instead + * of driver load/unload; + * + * 2001-06-03 Nicolas Pitre Made this file a separate module, based on + * the former sa1100-uda1341.c driver. + * + * 2001-07-22 Nicolas Pitre - added mmap() and realtime support + * - corrected many details to better comply + * with the OSS API + * + * 2001-10-19 Nicolas Pitre - brought DMA registration processing + * into this module for better ressource + * management. This also fixes a bug + * with the suspend/resume logic. + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include + +#include "sa1100-audio.h" + + +#undef DEBUG +/* #define DEBUG 1 */ +#ifdef DEBUG +#define DPRINTK( x... ) printk( ##x ) +#else +#define DPRINTK( x... ) +#endif + + +#define AUDIO_NAME "sa1100-audio" +#define AUDIO_NBFRAGS_DEFAULT 8 +#define AUDIO_FRAGSIZE_DEFAULT 8192 + +#define NEXT_BUF(_s_,_b_) { \ + (_s_)->_b_##_idx++; \ + (_s_)->_b_##_idx %= (_s_)->nbfrags; \ + (_s_)->_b_ = (_s_)->buffers + (_s_)->_b_##_idx; } + +#define AUDIO_ACTIVE(state) ((state)->rd_ref || (state)->wr_ref) + +/* + * This function frees all buffers + */ + +static void audio_clear_buf(audio_stream_t * s) +{ + DPRINTK("audio_clear_buf\n"); + + /* ensure DMA won't run anymore */ + s->active = 0; + s->stopped = 0; + sa1100_dma_flush_all(s->dma_ch); + + if (s->buffers) { + int frag; + for (frag = 0; frag < s->nbfrags; frag++) { + if (!s->buffers[frag].master) + continue; + consistent_free(s->buffers[frag].start, + s->buffers[frag].master, + s->buffers[frag].dma_addr); + } + kfree(s->buffers); + s->buffers = NULL; + } + + s->buf_idx = 0; + s->buf = NULL; +} + + +/* + * This function allocates the buffer structure array and buffer data space + * according to the current number of fragments and fragment size. + */ + +static int audio_setup_buf(audio_stream_t * s) +{ + int frag; + int dmasize = 0; + char *dmabuf = NULL; + dma_addr_t dmaphys = 0; + + if (s->buffers) + return -EBUSY; + + s->buffers = (audio_buf_t *) + kmalloc(sizeof(audio_buf_t) * s->nbfrags, GFP_KERNEL); + if (!s->buffers) + goto err; + memset(s->buffers, 0, sizeof(audio_buf_t) * s->nbfrags); + + for (frag = 0; frag < s->nbfrags; frag++) { + audio_buf_t *b = &s->buffers[frag]; + + /* + * Let's allocate non-cached memory for DMA buffers. + * We try to allocate all memory at once. + * If this fails (a common reason is memory fragmentation), + * then we allocate more smaller buffers. + */ + if (!dmasize) { + dmasize = (s->nbfrags - frag) * s->fragsize; + do { + dmabuf = consistent_alloc(GFP_KERNEL|GFP_DMA, + dmasize, + &dmaphys); + if (!dmabuf) + dmasize -= s->fragsize; + } while (!dmabuf && dmasize); + if (!dmabuf) + goto err; + b->master = dmasize; + memzero(dmabuf, dmasize); + } + + b->start = dmabuf; + b->dma_addr = dmaphys; + b->stream = s; + sema_init(&b->sem, 1); + DPRINTK("buf %d: start %p dma %p\n", frag, b->start, + b->dma_addr); + + dmabuf += s->fragsize; + dmaphys += s->fragsize; + dmasize -= s->fragsize; + } + + s->buf_idx = 0; + s->buf = &s->buffers[0]; + s->bytecount = 0; + s->getptrCount = 0; + s->fragcount = 0; + + return 0; + +err: + printk(AUDIO_NAME ": unable to allocate audio memory\n "); + audio_clear_buf(s); + return -ENOMEM; +} + + +/* + * This function yanks all buffers from the DMA code's control and + * resets them ready to be used again. + */ + +static void audio_reset_buf(audio_stream_t * s) +{ + int frag; + + s->active = 0; + s->stopped = 0; + sa1100_dma_flush_all(s->dma_ch); + if (s->buffers) { + for (frag = 0; frag < s->nbfrags; frag++) { + audio_buf_t *b = &s->buffers[frag]; + b->size = 0; + sema_init(&b->sem, 1); + } + } + s->bytecount = 0; + s->getptrCount = 0; + s->fragcount = 0; +} + + +/* + * DMA callback functions + */ + +static void audio_dmaout_done_callback(void *buf_id, int size) +{ + audio_buf_t *b = (audio_buf_t *) buf_id; + audio_stream_t *s = b->stream; + + /* Accounting */ + s->bytecount += size; + s->fragcount++; + + /* Recycle buffer */ + if (s->mapped) + sa1100_dma_queue_buffer(s->dma_ch, buf_id, + b->dma_addr, s->fragsize); + else + up(&b->sem); + + /* And any process polling on write. */ + wake_up(&s->wq); +} + +static void audio_dmain_done_callback(void *buf_id, int size) +{ + audio_buf_t *b = (audio_buf_t *) buf_id; + audio_stream_t *s = b->stream; + + /* Accounting */ + s->bytecount += size; + s->fragcount++; + + /* Recycle buffer */ + if (s->mapped) { + sa1100_dma_queue_buffer(s->dma_ch, buf_id, + b->dma_addr, s->fragsize); + } else { + b->size = size; + up(&b->sem); + } + + /* And any process polling on write. */ + wake_up(&s->wq); +} + +static int audio_sync(struct file *file) +{ + audio_state_t *state = (audio_state_t *)file->private_data; + audio_stream_t *s = state->output_stream; + audio_buf_t *b; + + DPRINTK("audio_sync\n"); + + if (!(file->f_mode & FMODE_WRITE) || !s->buffers || s->mapped) + return 0; + + /* + * Send current buffer if it contains data. Be sure to send + * a full sample count. + */ + b = s->buf; + if (b->size &= ~3) { + down(&b->sem); + sa1100_dma_queue_buffer(s->dma_ch, (void *) b, + b->dma_addr, b->size); + b->size = 0; + NEXT_BUF(s, buf); + } + + /* + * Let's wait for the last buffer we sent i.e. the one before the + * current buf_idx. When we acquire the semaphore, this means either: + * - DMA on the buffer completed or + * - the buffer was already free thus nothing else to sync. + */ + b = s->buffers + ((s->nbfrags + s->buf_idx - 1) % s->nbfrags); + if (down_interruptible(&b->sem)) + return -EINTR; + up(&b->sem); + return 0; +} + + +static int audio_write(struct file *file, const char *buffer, + size_t count, loff_t * ppos) +{ + const char *buffer0 = buffer; + audio_state_t *state = (audio_state_t *)file->private_data; + audio_stream_t *s = state->output_stream; + int chunksize, ret = 0; + + DPRINTK("audio_write: count=%d\n", count); + + if (ppos != &file->f_pos) + return -ESPIPE; + if (s->mapped) + return -ENXIO; + if (!s->buffers && audio_setup_buf(s)) + return -ENOMEM; + + while (count > 0) { + audio_buf_t *b = s->buf; + + /* Wait for a buffer to become free */ + if (file->f_flags & O_NONBLOCK) { + ret = -EAGAIN; + if (down_trylock(&b->sem)) + break; + } else { + ret = -ERESTARTSYS; + if (down_interruptible(&b->sem)) + break; + } + + /* Feed the current buffer */ + chunksize = s->fragsize - b->size; + if (chunksize > count) + chunksize = count; + DPRINTK("write %d to %d\n", chunksize, s->buf_idx); + if (copy_from_user(b->start + b->size, buffer, chunksize)) { + up(&b->sem); + return -EFAULT; + } + b->size += chunksize; + buffer += chunksize; + count -= chunksize; + if (b->size < s->fragsize) { + up(&b->sem); + break; + } + + /* Send current buffer to dma */ + s->active = 1; + sa1100_dma_queue_buffer(s->dma_ch, (void *) b, + b->dma_addr, b->size); + b->size = 0; /* indicate that the buffer has been sent */ + NEXT_BUF(s, buf); + } + + if ((buffer - buffer0)) + ret = buffer - buffer0; + DPRINTK("audio_write: return=%d\n", ret); + return ret; +} + + +static inline void audio_check_tx_spin(audio_state_t *state) +{ + /* + * With some codecs like the Philips UDA1341 we must ensure + * there is an output stream at any time while recording since + * this is how the UDA1341 gets its clock from the SA1100. + * So while there is no playback data to send, the output DMA + * will spin with all zeroes. We use the cache flush special + * area for that. + */ + if (state->need_tx_for_rx && !state->tx_spinning) { + sa1100_dma_set_spin(state->output_stream->dma_ch, + (dma_addr_t)FLUSH_BASE_PHYS, 2048); + state->tx_spinning = 1; + } +} + + +static void audio_prime_dma(audio_stream_t *s) +{ + int i; + + s->active = 1; + for (i = 0; i < s->nbfrags; i++) { + audio_buf_t *b = s->buf; + down(&b->sem); + sa1100_dma_queue_buffer(s->dma_ch, (void *) b, + b->dma_addr, s->fragsize); + NEXT_BUF(s, buf); + } +} + + +static int audio_read(struct file *file, char *buffer, + size_t count, loff_t * ppos) +{ + char *buffer0 = buffer; + audio_state_t *state = (audio_state_t *)file->private_data; + audio_stream_t *s = state->input_stream; + int chunksize, ret = 0; + + DPRINTK("audio_read: count=%d\n", count); + + if (ppos != &file->f_pos) + return -ESPIPE; + if (s->mapped) + return -ENXIO; + + if (!s->active) { + if (!s->buffers && audio_setup_buf(s)) + return -ENOMEM; + audio_check_tx_spin(state); + audio_prime_dma(s); + } + + while (count > 0) { + audio_buf_t *b = s->buf; + + /* Wait for a buffer to become full */ + if (file->f_flags & O_NONBLOCK) { + ret = -EAGAIN; + if (down_trylock(&b->sem)) + break; + } else { + ret = -ERESTARTSYS; + if (down_interruptible(&b->sem)) + break; + } + + /* Grab data from the current buffer */ + chunksize = b->size; + if (chunksize > count) + chunksize = count; + DPRINTK("read %d from %d\n", chunksize, s->buf_idx); + if (copy_to_user(buffer, + b->start + s->fragsize - b->size, + chunksize)) { + up(&b->sem); + return -EFAULT; + } + b->size -= chunksize; + buffer += chunksize; + count -= chunksize; + if (b->size > 0) { + up(&b->sem); + break; + } + + /* Make current buffer available for DMA again */ + sa1100_dma_queue_buffer(s->dma_ch, (void *) b, + b->dma_addr, s->fragsize); + NEXT_BUF(s, buf); + } + + if ((buffer - buffer0)) + ret = buffer - buffer0; + DPRINTK("audio_read: return=%d\n", ret); + return ret; +} + + +static int audio_mmap(struct file *file, struct vm_area_struct *vma) +{ + audio_state_t *state = (audio_state_t *)file->private_data; + audio_stream_t *s; + unsigned long size, vma_addr; + int i, ret; + + if (vma->vm_pgoff != 0) + return -EINVAL; + + if (vma->vm_flags & VM_WRITE) { + if (!state->wr_ref) + return -EINVAL;; + s = state->output_stream; + } else if (vma->vm_flags & VM_READ) { + if (!state->rd_ref) + return -EINVAL; + s = state->input_stream; + } else return -EINVAL; + + if (s->mapped) + return -EINVAL; + size = vma->vm_end - vma->vm_start; + if (size != s->fragsize * s->nbfrags) + return -EINVAL; + if (!s->buffers && audio_setup_buf(s)) + return -ENOMEM; + vma_addr = vma->vm_start; + for (i = 0; i < s->nbfrags; i++) { + audio_buf_t *buf = &s->buffers[i]; + if (!buf->master) + continue; + ret = remap_page_range(vma_addr, buf->dma_addr, buf->master, vma->vm_page_prot); + if (ret) + return ret; + vma_addr += buf->master; + } + s->mapped = 1; + + return 0; +} + + +static unsigned int audio_poll(struct file *file, + struct poll_table_struct *wait) +{ + audio_state_t *state = (audio_state_t *)file->private_data; + audio_stream_t *is = state->input_stream; + audio_stream_t *os = state->output_stream; + unsigned int mask = 0; + int i; + + DPRINTK("audio_poll(): mode=%s%s\n", + (file->f_mode & FMODE_READ) ? "r" : "", + (file->f_mode & FMODE_WRITE) ? "w" : ""); + + if (file->f_mode & FMODE_READ) { + /* Start audio input if not already active */ + if (!is->active) { + if (!is->buffers && audio_setup_buf(is)) + return -ENOMEM; + audio_check_tx_spin(state); + audio_prime_dma(is); + } + poll_wait(file, &is->wq, wait); + } + + if (file->f_mode & FMODE_WRITE) { + if (!os->buffers && audio_setup_buf(os)) + return -ENOMEM; + poll_wait(file, &os->wq, wait); + } + + if (file->f_mode & FMODE_READ) { + if (is->mapped) { +/* if the buffer is mapped assume we care that there are more bytes available than + when we last asked using SNDCTL_DSP_GETxPTR */ + if (is->bytecount != is->getptrCount) + mask |= POLLIN | POLLRDNORM; + } else { + for (i = 0; i < is->nbfrags; i++) { + if (atomic_read(&is->buffers[i].sem.count) > 0) { + mask |= POLLIN | POLLRDNORM; + break; + } + } + } + } + if (file->f_mode & FMODE_WRITE) { + if (os->mapped) { + if (os->bytecount != os->getptrCount) + mask |= POLLOUT | POLLWRNORM; + } else { + for (i = 0; i < os->nbfrags; i++) { + if (atomic_read(&os->buffers[i].sem.count) > 0) { + mask |= POLLOUT | POLLWRNORM; + break; + } + } + } + } + + DPRINTK("audio_poll() returned mask of %s%s\n", + (mask & POLLIN) ? "r" : "", + (mask & POLLOUT) ? "w" : ""); + + return mask; +} + + +static loff_t audio_llseek(struct file *file, loff_t offset, int origin) +{ + return -ESPIPE; +} + + +static int audio_set_fragments(audio_stream_t *s, int val) +{ + if (s->active) + return -EBUSY; + if (s->buffers) + audio_clear_buf(s); + s->nbfrags = (val >> 16) & 0x7FFF; + val &= 0xffff; + if (val < 4) + val = 4; + if (val > 15) + val = 15; + s->fragsize = 1 << val; + if (s->nbfrags < 2) + s->nbfrags = 2; + if (s->nbfrags * s->fragsize > 128 * 1024) + s->nbfrags = 128 * 1024 / s->fragsize; + if (audio_setup_buf(s)) + return -ENOMEM; + return val|(s->nbfrags << 16); +} + +static int audio_ioctl(struct inode *inode, struct file *file, + uint cmd, ulong arg) +{ + audio_state_t *state = (audio_state_t *)file->private_data; + audio_stream_t *os = state->output_stream; + audio_stream_t *is = state->input_stream; + long val; + + /* dispatch based on command */ + switch (cmd) { + case OSS_GETVERSION: + return put_user(SOUND_VERSION, (int *)arg); + + case SNDCTL_DSP_GETBLKSIZE: + if (file->f_mode & FMODE_WRITE) + return put_user(os->fragsize, (int *)arg); + else + return put_user(is->fragsize, (int *)arg); + + case SNDCTL_DSP_GETCAPS: + val = DSP_CAP_REALTIME|DSP_CAP_TRIGGER|DSP_CAP_MMAP; + if (is && os) + val |= DSP_CAP_DUPLEX; + return put_user(val, (int *)arg); + + case SNDCTL_DSP_SETFRAGMENT: + if (get_user(val, (long *) arg)) + return -EFAULT; + if (file->f_mode & FMODE_READ) { + int ret = audio_set_fragments(is, val); + if (ret < 0) + return ret; + ret = put_user(ret, (int *)arg); + if (ret) + return ret; + } + if (file->f_mode & FMODE_WRITE) { + int ret = audio_set_fragments(os, val); + if (ret < 0) + return ret; + ret = put_user(ret, (int *)arg); + if (ret) + return ret; + } + return 0; + + case SNDCTL_DSP_SYNC: + return audio_sync(file); + + case SNDCTL_DSP_SETDUPLEX: + return 0; + + case SNDCTL_DSP_POST: + return 0; + + case SNDCTL_DSP_GETTRIGGER: + val = 0; + if (file->f_mode & FMODE_READ && is->active && !is->stopped) + val |= PCM_ENABLE_INPUT; + if (file->f_mode & FMODE_WRITE && os->active && !os->stopped) + val |= PCM_ENABLE_OUTPUT; + return put_user(val, (int *)arg); + + case SNDCTL_DSP_SETTRIGGER: + if (get_user(val, (int *)arg)) + return -EFAULT; + if (file->f_mode & FMODE_READ) { + if (val & PCM_ENABLE_INPUT) { + if (!is->active) { + if (!is->buffers && audio_setup_buf(is)) + return -ENOMEM; + audio_prime_dma(is); + } + audio_check_tx_spin(state); + if (is->stopped) { + is->stopped = 0; + sa1100_dma_resume(is->dma_ch); + } + } else { + sa1100_dma_stop(is->dma_ch); + is->stopped = 1; + } + } + if (file->f_mode & FMODE_WRITE) { + if (val & PCM_ENABLE_OUTPUT) { + if (!os->active) { + if (!os->buffers && audio_setup_buf(os)) + return -ENOMEM; + if (os->mapped) + audio_prime_dma(os); + } + if (os->stopped) { + os->stopped = 0; + sa1100_dma_resume(os->dma_ch); + } + } else { + sa1100_dma_stop(os->dma_ch); + os->stopped = 1; + } + } + return 0; + + case SNDCTL_DSP_GETOPTR: + case SNDCTL_DSP_GETIPTR: + { + count_info inf = { 0, }; + audio_stream_t *s = (cmd == SNDCTL_DSP_GETOPTR) ? os : is; + audio_buf_t *b; + dma_addr_t ptr; + int bytecount, offset, flags; + + if ((s == is && !(file->f_mode & FMODE_READ)) || + (s == os && !(file->f_mode & FMODE_WRITE))) + return -EINVAL; + if (s->active) { + save_flags_cli(flags); + if (sa1100_dma_get_current(s->dma_ch, (void *)&b, &ptr) == 0) { + offset = ptr - b->dma_addr; + inf.ptr = (b - s->buffers) * s->fragsize + offset; + } else offset = 0; + bytecount = s->bytecount + offset; + s->getptrCount = s->bytecount; /* so poll can tell if it changes */ + inf.blocks = s->fragcount; + s->fragcount = 0; + restore_flags(flags); + if (bytecount < 0) + bytecount = 0; + inf.bytes = bytecount; + } + return copy_to_user((void *)arg, &inf, sizeof(inf)); + } + + case SNDCTL_DSP_GETOSPACE: + { + audio_buf_info inf = { 0, }; + int i; + + if (!(file->f_mode & FMODE_WRITE)) + return -EINVAL; + if (!os->buffers && audio_setup_buf(os)) + return -ENOMEM; + for (i = 0; i < os->nbfrags; i++) { + if (atomic_read(&os->buffers[i].sem.count) > 0) { + if (os->buffers[i].size == 0) + inf.fragments++; + inf.bytes += os->fragsize - os->buffers[i].size; + } + } + inf.fragstotal = os->nbfrags; + inf.fragsize = os->fragsize; + return copy_to_user((void *)arg, &inf, sizeof(inf)); + } + + case SNDCTL_DSP_GETISPACE: + { + audio_buf_info inf = { 0, }; + int i; + + if (!(file->f_mode & FMODE_READ)) + return -EINVAL; + if (!is->buffers && audio_setup_buf(is)) + return -ENOMEM; + for (i = 0; i < is->nbfrags; i++) { + if (atomic_read(&is->buffers[i].sem.count) > 0) { + if (is->buffers[i].size == is->fragsize) + inf.fragments++; + inf.bytes += is->buffers[i].size; + } + } + inf.fragstotal = is->nbfrags; + inf.fragsize = is->fragsize; + return copy_to_user((void *)arg, &inf, sizeof(inf)); + } + + case SNDCTL_DSP_NONBLOCK: + file->f_flags |= O_NONBLOCK; + return 0; + + case SNDCTL_DSP_RESET: + if (file->f_mode & FMODE_READ) { + if (state->tx_spinning) { + sa1100_dma_set_spin(os->dma_ch, 0, 0); + state->tx_spinning = 0; + } + audio_reset_buf(is); + } + if (file->f_mode & FMODE_WRITE) { + audio_reset_buf(os); + } + return 0; + + default: + /* + * Let the client of this module handle the + * non generic ioctls + */ + return state->client_ioctl(inode, file, cmd, arg); + } + + return 0; +} + + +static int audio_release(struct inode *inode, struct file *file) +{ + audio_state_t *state = (audio_state_t *)file->private_data; + DPRINTK("audio_release\n"); + + down(&state->sem); + + if (file->f_mode & FMODE_READ) { + if (state->tx_spinning) { + sa1100_dma_set_spin(state->output_stream->dma_ch, 0, 0); + state->tx_spinning = 0; + } + audio_clear_buf(state->input_stream); + if (!state->skip_dma_init) { + sa1100_free_dma(state->input_stream->dma_ch); + if (state->need_tx_for_rx && !state->wr_ref) + sa1100_free_dma(state->output_stream->dma_ch); + } + state->rd_ref = 0; + } + + if (file->f_mode & FMODE_WRITE) { + audio_sync(file); + audio_clear_buf(state->output_stream); + if (!state->skip_dma_init) + if (!state->need_tx_for_rx || !state->rd_ref) + sa1100_free_dma(state->output_stream->dma_ch); + state->wr_ref = 0; + } + + if (!AUDIO_ACTIVE(state)) { + if (state->hw_shutdown) + state->hw_shutdown(state->data); +#ifdef CONFIG_PM + pm_unregister(state->pm_dev); +#endif + } + + up(&state->sem); + return 0; +} + + +#ifdef CONFIG_PM + +static int audio_pm_callback(struct pm_dev *pm_dev, pm_request_t req, void *data) +{ + audio_state_t *state = (audio_state_t *)pm_dev->data; + + switch (req) { + case PM_SUSPEND: /* enter D1-D3 */ + if (state->output_stream) + sa1100_dma_sleep(state->output_stream->dma_ch); + if (state->input_stream) + sa1100_dma_sleep(state->input_stream->dma_ch); + if (AUDIO_ACTIVE(state) && state->hw_shutdown) + state->hw_shutdown(state->data); + break; + case PM_RESUME: /* enter D0 */ + if (AUDIO_ACTIVE(state) && state->hw_init) + state->hw_init(state->data); + if (state->input_stream) + sa1100_dma_wakeup(state->input_stream->dma_ch); + if (state->output_stream) + sa1100_dma_wakeup(state->output_stream->dma_ch); + break; + } + return 0; +} + +#endif + + +int sa1100_audio_attach(struct inode *inode, struct file *file, + audio_state_t *state) +{ + int err, need_tx_dma; + + DPRINTK("audio_open\n"); + + down(&state->sem); + + /* access control */ + err = -ENODEV; + if ((file->f_mode & FMODE_WRITE) && !state->output_stream) + goto out; + if ((file->f_mode & FMODE_READ) && !state->input_stream) + goto out; + err = -EBUSY; + if ((file->f_mode & FMODE_WRITE) && state->wr_ref) + goto out; + if ((file->f_mode & FMODE_READ) && state->rd_ref) + goto out; + err = -EINVAL; + if ((file->f_mode & FMODE_READ) && state->need_tx_for_rx && !state->output_stream) + goto out; + + /* request DMA channels */ + if (state->skip_dma_init) + goto skip_dma; + need_tx_dma = ((file->f_mode & FMODE_WRITE) || + ((file->f_mode & FMODE_READ) && state->need_tx_for_rx)); + if (state->wr_ref || (state->rd_ref && state->need_tx_for_rx)) + need_tx_dma = 0; + if (need_tx_dma) { + err = sa1100_request_dma(&state->output_stream->dma_ch, + state->output_id, + state->output_dma); + if (err) + goto out; + } + if (file->f_mode & FMODE_READ) { + err = sa1100_request_dma(&state->input_stream->dma_ch, + state->input_id, + state->input_dma); + if (err) { + if (need_tx_dma) + sa1100_free_dma(state->output_stream->dma_ch); + goto out; + } + } +skip_dma: + + /* now complete initialisation */ + if (!AUDIO_ACTIVE(state)) { + if (state->hw_init) + state->hw_init(state->data); +#ifdef CONFIG_PM + state->pm_dev = pm_register(PM_SYS_DEV, 0, audio_pm_callback); + if (state->pm_dev) + state->pm_dev->data = state; +#endif + } + + if ((file->f_mode & FMODE_WRITE)) { + state->wr_ref = 1; + audio_clear_buf(state->output_stream); + state->output_stream->fragsize = AUDIO_FRAGSIZE_DEFAULT; + state->output_stream->nbfrags = AUDIO_NBFRAGS_DEFAULT; + state->output_stream->mapped = 0; + sa1100_dma_set_callback(state->output_stream->dma_ch, + audio_dmaout_done_callback); + init_waitqueue_head(&state->output_stream->wq); + } + if (file->f_mode & FMODE_READ) { + state->rd_ref = 1; + audio_clear_buf(state->input_stream); + state->input_stream->fragsize = AUDIO_FRAGSIZE_DEFAULT; + state->input_stream->nbfrags = AUDIO_NBFRAGS_DEFAULT; + state->input_stream->mapped = 0; + sa1100_dma_set_callback(state->input_stream->dma_ch, + audio_dmain_done_callback); + init_waitqueue_head(&state->input_stream->wq); + } + + file->private_data = state; + file->f_op->release = audio_release; + file->f_op->write = audio_write; + file->f_op->read = audio_read; + file->f_op->mmap = audio_mmap; + file->f_op->poll = audio_poll; + file->f_op->ioctl = audio_ioctl; + file->f_op->llseek = audio_llseek; + err = 0; + +out: + up(&state->sem); + return err; +} + +EXPORT_SYMBOL(sa1100_audio_attach); + +MODULE_AUTHOR("Nicolas Pitre"); +MODULE_DESCRIPTION("Common audio handling for the SA11x0 processor"); +MODULE_LICENSE("GPL"); diff -urN orig/drivers/sound/sa1100-audio.h linux/drivers/sound/sa1100-audio.h --- orig/drivers/sound/sa1100-audio.h Thu Jan 1 01:00:00 1970 +++ linux/drivers/sound/sa1100-audio.h Fri Dec 13 14:09:01 2002 @@ -0,0 +1,68 @@ +/* + * Common audio handling for the SA11x0 + * + * Copyright (c) 2000 Nicolas Pitre + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License. + */ + + +/* + * Buffer Management + */ + +typedef struct { + int size; /* buffer size */ + char *start; /* points to actual buffer */ + dma_addr_t dma_addr; /* physical buffer address */ + struct semaphore sem; /* down before touching the buffer */ + int master; /* owner for buffer allocation, contain size when true */ + struct audio_stream_s *stream; /* owning stream */ +} audio_buf_t; + +typedef struct audio_stream_s { + audio_buf_t *buffers; /* pointer to audio buffer structures */ + audio_buf_t *buf; /* current buffer used by read/write */ + u_int buf_idx; /* index for the pointer above... */ + u_int fragsize; /* fragment i.e. buffer size */ + u_int nbfrags; /* nbr of fragments i.e. buffers */ + int bytecount; /* nbr of processed bytes */ + int getptrCount; /* value of bytecount last time anyone asked via GETxPTR */ + int fragcount; /* nbr of fragment transitions */ + dmach_t dma_ch; /* DMA channel ID */ + wait_queue_head_t wq; /* for poll */ + int mapped:1; /* mmap()'ed buffers */ + int active:1; /* actually in progress */ + int stopped:1; /* might be active but stopped */ +} audio_stream_t; + +/* + * State structure for one instance + */ + +typedef struct { + audio_stream_t *output_stream; + audio_stream_t *input_stream; + dma_device_t output_dma; + dma_device_t input_dma; + char *output_id; + char *input_id; + int rd_ref:1; /* open reference for recording */ + int wr_ref:1; /* open reference for playback */ + int need_tx_for_rx:1; /* if data must be sent while receiving */ + int tx_spinning:1; /* tx spinning active */ + int skip_dma_init:1; /* hack for the SA1111 */ + void *data; + void (*hw_init)(void *); + void (*hw_shutdown)(void *); + int (*client_ioctl)(struct inode *, struct file *, uint, ulong); + struct pm_dev *pm_dev; + struct semaphore sem; /* to protect against races in attach() */ +} audio_state_t; + +/* + * Functions exported by this module + */ +extern int sa1100_audio_attach( struct inode *inode, struct file *file, + audio_state_t *state); diff -urN orig/drivers/sound/sa1100ssp.c linux/drivers/sound/sa1100ssp.c --- orig/drivers/sound/sa1100ssp.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/sound/sa1100ssp.c Wed Nov 14 18:08:47 2001 @@ -0,0 +1,182 @@ +/* + * Glue audio driver for a simple DAC on the SA1100's SSP port + * + * Copyright (c) 2001 Nicolas Pitre + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License. + * + * History: + * + * 2001-06-04 Nicolas Pitre Initial release. + * + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include + +#include "sa1100-audio.h" + + +#undef DEBUG +#ifdef DEBUG +#define DPRINTK( x... ) printk( ##x ) +#else +#define DPRINTK( x... ) +#endif + + +#define AUDIO_NAME "SA1100 SSP audio" + +#define AUDIO_FMT AFMT_S16_LE +#define AUDIO_CHANNELS 2 + +static int sample_rate = 44100; + + +static void ssp_audio_init(void) +{ + if (machine_is_lart()) { + unsigned long flags; + local_irq_save(flags); + + /* LART has the SSP port rewired to GPIO 10-13, 19 */ + /* alternate functions for the GPIOs */ + GAFR |= ( GPIO_SSP_TXD | GPIO_SSP_RXD | GPIO_SSP_SCLK | + GPIO_SSP_SFRM | GPIO_SSP_CLK ); + + /* Set the direction: 10, 12, 13 output; 11, 19 input */ + GPDR |= ( GPIO_SSP_TXD | GPIO_SSP_SCLK | GPIO_SSP_SFRM ); + GPDR &= ~( GPIO_SSP_RXD | GPIO_SSP_CLK ); + + /* enable SSP pin swap */ + PPAR |= PPAR_SPR; + + local_irq_restore(flags); + } + + /* turn on the SSP */ + Ser4SSCR0 = 0; + Ser4SSCR0 = (SSCR0_DataSize(16) | SSCR0_TI | SSCR0_SerClkDiv(2) | + SSCR0_SSE); + Ser4SSCR1 = (SSCR1_SClkIactL | SSCR1_SClk1P | SSCR1_ExtClk); +} + +static void ssp_audio_shutdown(void) +{ + Ser4SSCR0 = 0; +} + +static int ssp_audio_ioctl( struct inode *inode, struct file *file, + uint cmd, ulong arg) +{ + long val; + int ret = 0; + + /* + * These are platform dependent ioctls which are not handled by the + * generic sa1100-audio module. + */ + switch (cmd) { + case SNDCTL_DSP_STEREO: + ret = get_user(val, (int *) arg); + if (ret) + return ret; + /* Simple standard DACs are stereo only */ + ret = (val == 0) ? -EINVAL : 1; + return put_user(ret, (int *) arg); + + case SNDCTL_DSP_CHANNELS: + case SOUND_PCM_READ_CHANNELS: + /* Simple standard DACs are stereo only */ + return put_user(AUDIO_CHANNELS, (long *) arg); + + case SNDCTL_DSP_SPEED: + case SOUND_PCM_READ_RATE: + /* We assume the clock doesn't change */ + return put_user(sample_rate, (long *) arg); + + case SNDCTL_DSP_SETFMT: + case SNDCTL_DSP_GETFMTS: + /* Simple standard DACs are 16-bit only */ + return put_user(AUDIO_FMT, (long *) arg); + + default: + return -EINVAL; + } + + return ret; +} + +static audio_stream_t output_stream; + +static audio_state_t audio_state = { + output_stream: &output_stream, + output_dma: DMA_Ser4SSPWr, + output_id: "Generic SSP sound", + hw_init: ssp_audio_init, + hw_shutdown: ssp_audio_shutdown, + client_ioctl: ssp_audio_ioctl, + sem: __MUTEX_INITIALIZER(audio_state.sem), +}; + +static int ssp_audio_open(struct inode *inode, struct file *file) +{ + return sa1100_audio_attach(inode, file, &audio_state); +} + +/* + * Missing fields of this structure will be patched with the call + * to sa1100_audio_attach(). + */ +static struct file_operations ssp_audio_fops = { + open: ssp_audio_open, + owner: THIS_MODULE +}; + +static int audio_dev_id; + +static int __init sa1100ssp_audio_init(void) +{ + int ret; + + if (!machine_is_lart()) { + printk(KERN_ERR AUDIO_NAME ": no support for this SA-1100 design!\n"); + /* look at ssp_audio_init() for specific initialisations */ + return -ENODEV; + } + + /* register devices */ + audio_dev_id = register_sound_dsp(&ssp_audio_fops, -1); + + printk( KERN_INFO AUDIO_NAME " initialized\n" ); + return 0; +} + +static void __exit sa1100ssp_audio_exit(void) +{ + unregister_sound_dsp(audio_dev_id); +} + +module_init(sa1100ssp_audio_init); +module_exit(sa1100ssp_audio_exit); + +MODULE_AUTHOR("Nicolas Pitre"); +MODULE_DESCRIPTION("Glue audio driver for a simple DAC on the SA1100's SSP port"); + +MODULE_PARM(sample_rate, "i"); +MODULE_PARM_DESC(sample_rate, "Sample rate of the audio DAC, default is 44100"); + +EXPORT_NO_SYMBOLS; diff -urN orig/drivers/sound/sa1111-ac97.c linux/drivers/sound/sa1111-ac97.c --- orig/drivers/sound/sa1111-ac97.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/sound/sa1111-ac97.c Thu Feb 27 23:20:11 2003 @@ -0,0 +1,518 @@ +/* + * Glue audio driver for the CS4205 and CS4201 AC'97 codecs. + * largely based on the framework provided by sa1111-uda1341.c. + * + * Copyright (c) 2002 Bertrik Sikken (bertrik.sikken@technolution.nl) + * Copyright (c) 2002 Robert Whaley (rwhaley@applieddata.net) + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License. + * + * This driver makes use of the ac97_codec module (for mixer registers) + * and the sa1100-audio module (for DMA). + * + * History: + * + * 2002-04-04 Initial version. + * 2002-04-10 Updated mtd_audio_init to improve choppy sound + * and hanging sound issue. + * 2002-05-16 Updated for ADS Bitsy+ Robert Whaley + * 2002-06-28 Cleanup and added retry for read register timeouts + * 2002-08-14 Updated for ADS AGC Robert Whaley + * 2002-12-26 Cleanup, remove CONFIG_PM (it's handled by sa1100-audio.c) + * + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include + +#include "sa1100-audio.h" + +/* SAC FIFO depth, low nibble is transmit fifo, high nibble is receive FIFO */ +#define SAC_FIFO_DEPTH 0x77 + +// #define DEBUG + +#ifdef DEBUG +#define DPRINTK( x... ) printk( ##x ) +#else +#define DPRINTK( x... ) +#endif + +/* + Our codec data +*/ +static struct ac97_codec ac97codec; +static int audio_dev_id, mixer_dev_id; +static audio_stream_t output_stream, input_stream; + +/* proc info */ + +struct proc_dir_entry *ac97_ps; + +static int sa1111_ac97_set_adc_rate(long rate); +static void sa1111_ac97_write_reg(struct ac97_codec *dev, u8 reg, u16 val); +static u16 sa1111_ac97_read_reg(struct ac97_codec *dev, u8 reg); + +static int +mixer_ioctl(struct inode *inode, struct file *file, uint cmd, ulong arg) +{ + /* + * We only accept mixer (type 'M') ioctls. + */ + if (_IOC_TYPE(cmd) != 'M') { + return -EINVAL; + } + + /* pass the ioctl to the ac97 mixer */ + return ac97codec.mixer_ioctl(&ac97codec, cmd, arg); +} + + +static struct file_operations sa1111_ac97_mixer_fops = { + ioctl: mixer_ioctl, + owner: THIS_MODULE +}; + +static void sa1111_ac97_power_off(void *dummy) +{ +#ifdef CONFIG_SA1100_ADSBITSYPLUS + /* turn off audio and audio amp */ + ADS_CPLD_PCON |= (ADS_PCON_AUDIO_ON | ADS_PCON_AUDIOPA_ON); + + /* make GPIO11 high impeadence */ + GPDR &= ~GPIO_GPIO11; + + /* disable SACR0 so we can make these pins high impeadence */ + SACR0 &= ~SACR0_ENB; + + /* make BIT_CLK, SDATA_OUT, and SYNC high impeadence */ + PC_DDR |= (GPIO_GPIO1 | GPIO_GPIO2 | GPIO_GPIO3); + +#endif + +#ifdef CONFIG_SA1100_ADSAGC + /* turn off audio and audio amp */ + ADS_CR1 &= ~(ADS_CR1_CODEC | ADS_CR1_AMP); + + /* disable SACR0 so we can make these pins high impeadence */ + SACR0 &= ~SACR0_ENB; + + /* make BIT_CLK, SDATA_OUT, and SYNC high impeadence */ + PC_DDR |= (GPIO_GPIO1 | GPIO_GPIO2 | GPIO_GPIO3); + +#endif +} + + +static void sa1111_ac97_power_on(void *dummy) +{ + int ret, i; + + /* disable L3 */ + SACR1 = 0; + + SKPCR |= (SKPCR_ACCLKEN); /* enable ac97 clock */ + udelay(50); + + /* BIT_CLK is input to SA1111, DMA thresholds 9 (both dirs) */ + SACR0 |= SACR0_BCKD | (SAC_FIFO_DEPTH << 8); + + /* reset SAC registers */ + SACR0 &= ~SACR0_RST; + udelay(50); + SACR0 |= SACR0_RST; + udelay(50); + SACR0 &= ~SACR0_RST; + + /* setup SA1111 to use AC'97 */ + SBI_SKCR |= SKCR_SELAC; /* select ac97 */ + udelay(50); + + /* issue a cold AC97 reset */ +#ifdef CONFIG_SA1100_ADSBITSYPLUS + + /* initialize reset line */ + GAFR &= ~GPIO_GPIO11; + GPDR |= GPIO_GPIO11; + GPSR = GPIO_GPIO11; + + /* turn on audio and audio amp */ + ADS_CPLD_PCON &= ~(ADS_PCON_AUDIO_ON | ADS_PCON_AUDIOPA_ON); + mdelay(5); + + /* reset by lowering the reset pin momentarily */ + DPRINTK("reseting codec via GPIO11\n"); + GPCR = GPIO_GPIO11; + udelay(5); + GPSR = GPIO_GPIO11; + udelay(10); + +#endif +#ifdef CONFIG_SA1100_ADSAGC + + /* turn on audio and audio amp */ + DPRINTK("before turning on power. ADS_CR1: %x\n", ADS_CR1); + ADS_CR1 |= (ADS_CR1_AMP | ADS_CR1_CODEC); + DPRINTK("after turnning on power. ADS_CR1: %x\n", ADS_CR1); + mdelay(5); + + /* reset by lowering the reset pin momentarily */ + DPRINTK("reseting codec via CPLD\n"); + ADS_CR1 |= ADS_CR1_AUDIO_RST; + DPRINTK("after reset1. ADS_CR1: %x\n", ADS_CR1); + udelay(5); + ADS_CR1 &= ~ADS_CR1_AUDIO_RST; + DPRINTK("after reset2. ADS_CR1: %x\n", ADS_CR1); + udelay(10); + +#endif + SACR2 = 0; + udelay(50); + + DPRINTK("before SW reset: SACR2: %x\n", SACR2); + SACR2 = SACR2_RESET; + DPRINTK("after SW reset: SACR2: %x\n", SACR2); + udelay(50); + + /* set AC97 slot 3 and 4 (PCM out) to valid */ + SACR2 = (SACR2_RESET | SACR2_TS3V | SACR2_TS4V); + + /* enable SAC */ + SACR0 |= SACR0_ENB; + + i = 100; + while (!(SASR1 & SASR1_CRDY)) { + if (!i--) { + printk("Didn't get CRDY. SASR1=%x SKID=%x\n", SASR1, SBI_SKID); + break; + } + udelay(50); + } + + if (!(ret = ac97_probe_codec(&ac97codec))) { + printk("ac97_probe_codec failed (%d)\n", ret); + return; + } + + /* mic ADC on, disable VRA, disable VRM */ + sa1111_ac97_write_reg(&ac97codec, AC97_EXTENDED_STATUS, 0x0200); +} + + +/* + * Audio interface + */ + + +static int sa1111_ac97_audio_ioctl(struct inode *inode, struct file *file, + uint cmd, ulong arg) +{ + long val; + int ret = 0; + + DPRINTK("sa1111_ac97_audio_ioctl\n"); + + /* + * These are platform dependent ioctls which are not handled by the + * generic sa1100-audio module. + */ + switch (cmd) { + case SNDCTL_DSP_STEREO: + ret = get_user(val, (int *) arg); + if (ret) { + return ret; + } + /* the cs42xx is stereo only */ + ret = (val == 0) ? -EINVAL : 1; + return put_user(ret, (int *) arg); + + case SNDCTL_DSP_CHANNELS: + case SOUND_PCM_READ_CHANNELS: + /* the cs42xx is stereo only */ + return put_user(2, (long *) arg); + +#define SA1100_AC97_IOCTL_EXTRAS + +#ifdef SA1100_AC97_IOCTL_EXTRAS + +#define SNDCTL_DSP_AC97_CMD _SIOWR('P', 99, int) +#define SNDCTL_DSP_INPUT_SPEED _SIOWR('P', 98, int) +#define SOUND_PCM_READ_INPUT_RATE _SIOWR('P', 97, int) + + case SNDCTL_DSP_AC97_CMD: + + ret = get_user(val, (long *) arg); + if (ret) { + break; + } + sa1111_ac97_write_reg(&ac97codec, (u8) ((val & 0xff000000) >> 24), (u16) (val & 0xffff)); + return 0; + + + case SNDCTL_DSP_INPUT_SPEED: + ret = get_user(val, (long *) arg); + // acc code here to set the speed + if (ret) { + break; + } + // note that this only changes the ADC rate, not the + // rate of the DAC. + ret = sa1111_ac97_set_adc_rate(val); + if (ret) + break; + return put_user(val, (long *) arg); + + case SOUND_PCM_READ_INPUT_RATE: + + return put_user((long) sa1111_ac97_read_reg(&ac97codec, 0x32), (long *) arg); + + +#endif + + case SNDCTL_DSP_SPEED: + ret = get_user(val, (long *) arg); + if (ret) { + break; + } + + case SOUND_PCM_READ_RATE: + /* only 48 kHz playback is supported by the SA1111 */ + return put_user(48000L, (long *) arg); + + case SNDCTL_DSP_SETFMT: + case SNDCTL_DSP_GETFMTS: + /* we can do 16-bit only */ + return put_user(AFMT_S16_LE, (long *) arg); + + default: + /* Maybe this is meant for the mixer (As per OSS Docs) */ + return mixer_ioctl(inode, file, cmd, arg); + } + + return ret; +} + + +static audio_state_t audio_state = { + output_stream: &output_stream, + input_stream: &input_stream, + skip_dma_init: 1, /* done locally */ + hw_init: sa1111_ac97_power_on, + hw_shutdown: sa1111_ac97_power_off, + client_ioctl: sa1111_ac97_audio_ioctl, + sem: __MUTEX_INITIALIZER(audio_state.sem), +}; + + +static int sa1111_ac97_audio_open(struct inode *inode, struct file *file) +{ + return sa1100_audio_attach(inode, file, &audio_state); +} + + +/* + * Missing fields of this structure will be patched with the call + * to sa1100_audio_attach(). + */ +static struct file_operations sa1111_ac97_audio_fops = { + open: sa1111_ac97_audio_open, + owner: THIS_MODULE +}; + + +static void sa1111_ac97_write_reg(struct ac97_codec *dev, u8 reg, u16 val) +{ + int i; + + /* reset status bits */ + SASCR = SASCR_DTS; + + /* write command and data registers */ + ACCAR = reg << 12; + ACCDR = val << 4; + + /* wait for data to be transmitted */ + i = 0; + while ((SASR1 & SASR1_CADT) == 0) { + udelay(50); + if (++i > 10) { + DPRINTK("sa1111_ac97_write_reg failed (data not transmitted. SASR1: %x)\n", SASR1); + break; + } + } + + DPRINTK("<%03d> sa1111_ac97_write_reg, [%02X]=%04X\n", i, reg, val); +} + + +static u16 sa1111_ac97_read_reg(struct ac97_codec *dev, u8 reg) +{ + u16 val; + int i; + int retry = 10; + + do { + /* reset status bits */ + SASCR = SASCR_RDD | SASCR_STO; + + /* write command register */ + ACCAR = (reg | 0x80) << 12; + ACCDR = 0; + + /* wait for SADR bit in SASR1 */ + i = 0; + while ((SASR1 & SASR1_SADR) == 0) { + udelay(50); + if (++i > 10) { + DPRINTK("<---> sa1111_ac97_read_reg failed\n"); + retry--; + break; + } + if ((SASR1 & SASR1_RSTO) != 0) { + DPRINTK("sa1111_ac97_read_reg *timeout*\n"); + retry--; + break; + } + } + + } while ((SASR1 & SASR1_SADR) == 0 && retry > 0); + + val = ACSDR >> 4; + + DPRINTK("<%03d> sa1111_ac97_read_reg, [%02X]=%04X\n", i, reg, val); + return val; +} + + +/* wait for codec ready */ +static void sa1111_ac97_ready(struct ac97_codec *dev) +{ + int i; + u16 val; + + i = 0; + while ((SASR1 & SASR1_CRDY) == 0) { + udelay(50); + if (++i > 10) { + DPRINTK("sa1111_ac97_ready failed\n"); + return; + } + } + DPRINTK("codec_ready bit took %d cycles\n", i); + + /* Wait for analog parts of codec to initialise */ + i = 0; + do { + val = sa1111_ac97_read_reg(&ac97codec, AC97_POWER_CONTROL); + if (++i > 100) { + break; + } + mdelay(10); + } while ((val & 0xF) != 0xF || val == 0xFFFF); + + /* the cs42xx typically takes 150 ms to initialise */ + + DPRINTK("analog init took %d cycles\n", i); +} + + +static int __init sa1111_ac97_init(void) +{ + int ret; + + // SBI_SKCR |= SKCR_RCLKEN; + + DPRINTK("sa1111_ac97_init\n"); + + /* install the ac97 mixer module */ + ac97codec.codec_read = sa1111_ac97_read_reg; + ac97codec.codec_write = sa1111_ac97_write_reg; + ac97codec.codec_wait = sa1111_ac97_ready; + + /* Acquire and initialize DMA */ + ret = sa1111_sac_request_dma(&output_stream.dma_ch, "SA1111 audio out", + SA1111_SAC_XMT_CHANNEL); + if (ret < 0) { + printk("DMA request for SAC output failed\n"); + return ret; + } + + ret = sa1111_sac_request_dma(&input_stream.dma_ch, "SA1111 audio in", + SA1111_SAC_RCV_CHANNEL); + if (ret < 0) { + printk("DMA request for SAC input failed\n"); + sa1100_free_dma(output_stream.dma_ch); + return ret; + } + /* register devices */ + audio_dev_id = register_sound_dsp(&sa1111_ac97_audio_fops, -1); + mixer_dev_id = register_sound_mixer(&sa1111_ac97_mixer_fops, -1); + + + /* setup proc entry */ + ac97_ps = create_proc_read_entry ("driver/sa1111-ac97", 0, NULL, + ac97_read_proc, &ac97codec); + + return 0; +} + + +static void __exit sa1111_ac97_exit(void) +{ + SKPCR &= ~SKPCR_ACCLKEN; /* disable ac97 clock */ + SBI_SKCR &= ~SKCR_SELAC; /* deselect ac97 */ + + unregister_sound_dsp(audio_dev_id); + unregister_sound_mixer(mixer_dev_id); + sa1100_free_dma(output_stream.dma_ch); + sa1100_free_dma(input_stream.dma_ch); +} + +static int sa1111_ac97_set_adc_rate(long rate) +{ + + // note this only changes the rate of the ADC, the DAC is fixed at 48K. + // this is due to limitations of the SA1111 chip + + u16 code = rate; + + switch (rate) { + case 8000: + case 11025: + case 16000: + case 22050: + case 32000: + case 44100: + case 48000: + break; + default: + return -1; + } + sa1111_ac97_write_reg(&ac97codec, 0x2A, 0x0001); + sa1111_ac97_write_reg(&ac97codec, 0x32, code); + return 0; +} + +module_init(sa1111_ac97_init); +module_exit(sa1111_ac97_exit); + +MODULE_AUTHOR("Bertrik Sikken, Technolution B.V., Netherlands"); +MODULE_DESCRIPTION("Glue audio driver for AC'97 codec"); +MODULE_LICENSE("GPL"); + +EXPORT_NO_SYMBOLS; diff -urN orig/drivers/sound/sa1111-uda1341.c linux/drivers/sound/sa1111-uda1341.c --- orig/drivers/sound/sa1111-uda1341.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/sound/sa1111-uda1341.c Sat Feb 2 23:47:57 2002 @@ -0,0 +1,282 @@ +/* + * Glue audio driver for the SA1111 compagnon chip & Philips UDA1341 codec. + * + * Copyright (c) 2000 John Dorsey + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License. + * + * History: + * + * 2000-09-04 John Dorsey SA-1111 Serial Audio Controller support + * was initially added to the sa1100-uda1341.c + * driver. + * + * 2001-06-03 Nicolas Pitre Made this file a separate module, based on + * the former sa1100-uda1341.c driver. + * + * 2001-09-23 Russell King Remove old L3 bus driver. + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include + +#include "sa1100-audio.h" + +#undef DEBUG +#ifdef DEBUG +#define DPRINTK( x... ) printk( ##x ) +#else +#define DPRINTK( x... ) +#endif + + +/* + * Definitions + */ + +#define AUDIO_RATE_DEFAULT 22050 + +#define AUDIO_CLK_BASE 561600 + + + +/* + * Mixer interface + */ + +static struct l3_client uda1341; + +static int +mixer_ioctl(struct inode *inode, struct file *file, uint cmd, ulong arg) +{ + /* + * We only accept mixer (type 'M') ioctls. + */ + if (_IOC_TYPE(cmd) != 'M') + return -EINVAL; + + return l3_command(&uda1341, cmd, (void *)arg); +} + +static struct file_operations uda1341_mixer_fops = { + ioctl: mixer_ioctl, + owner: THIS_MODULE +}; + + +/* + * Audio interface + */ + +static int audio_clk_div = (AUDIO_CLK_BASE + AUDIO_RATE_DEFAULT/2)/AUDIO_RATE_DEFAULT; + +static void sa1111_audio_init(void *dummy) +{ +#ifdef CONFIG_ASSABET_NEPONSET + if (machine_is_assabet()) { + /* Select I2S audio (instead of AC-Link) */ + AUD_CTL = AUD_SEL_1341; + } +#endif +#ifdef CONFIG_SA1100_JORNADA720 + if (machine_is_jornada720()) { + /* LDD4 is speaker, LDD3 is microphone */ + PPSR &= ~(PPC_LDD3 | PPC_LDD4); + PPDR |= PPC_LDD3 | PPC_LDD4; + PPSR |= PPC_LDD4; /* enable speaker */ + PPSR |= PPC_LDD3; /* enable microphone */ + } +#endif + + SBI_SKCR &= ~SKCR_SELAC; + + /* Enable the I2S clock and L3 bus clock: */ + SKPCR |= (SKPCR_I2SCLKEN | SKPCR_L3CLKEN); + + /* Activate and reset the Serial Audio Controller */ + SACR0 |= (SACR0_ENB | SACR0_RST); + mdelay(5); + SACR0 &= ~SACR0_RST; + + /* For I2S, BIT_CLK is supplied internally. The "SA-1111 + * Specification Update" mentions that the BCKD bit should + * be interpreted as "0 = output". Default clock divider + * is 22.05kHz. + * + * Select I2S, L3 bus. "Recording" and "Replaying" + * (receive and transmit) are enabled. + */ + SACR1 = SACR1_L3EN; + SKAUD = audio_clk_div - 1; + + /* Initialize the UDA1341 internal state */ + l3_open(&uda1341); +} + +static void sa1111_audio_shutdown(void *dummy) +{ + l3_close(&uda1341); + SACR0 &= ~SACR0_ENB; +} + +static int sa1111_audio_ioctl( struct inode *inode, struct file *file, + uint cmd, ulong arg) +{ + long val; + int ret = 0; + + switch (cmd) { + case SNDCTL_DSP_STEREO: + ret = get_user(val, (int *) arg); + if (ret) + return ret; + /* the UDA1341 is stereo only */ + ret = (val == 0) ? -EINVAL : 1; + return put_user(ret, (int *) arg); + + case SNDCTL_DSP_CHANNELS: + case SOUND_PCM_READ_CHANNELS: + /* the UDA1341 is stereo only */ + return put_user(2, (long *) arg); + + case SNDCTL_DSP_SPEED: + ret = get_user(val, (long *) arg); + if (ret) break; + if (val < 8000) val = 8000; + if (val > 48000) val = 48000; + audio_clk_div = (AUDIO_CLK_BASE + val/2)/val; + SKAUD = audio_clk_div - 1; + /* fall through */ + + case SOUND_PCM_READ_RATE: + return put_user(AUDIO_CLK_BASE/audio_clk_div, (long *) arg); + + case SNDCTL_DSP_SETFMT: + case SNDCTL_DSP_GETFMTS: + /* we can do 16-bit only */ + return put_user(AFMT_S16_LE, (long *) arg); + + default: + /* Maybe this is meant for the mixer (as per OSS Docs) */ + return mixer_ioctl(inode, file, cmd, arg); + } + + return ret; +} + +static audio_stream_t output_stream, input_stream; + +static audio_state_t audio_state = { + output_stream: &output_stream, + input_stream: &input_stream, + skip_dma_init: 1, /* done locally */ + hw_init: sa1111_audio_init, + hw_shutdown: sa1111_audio_shutdown, + client_ioctl: sa1111_audio_ioctl, + sem: __MUTEX_INITIALIZER(audio_state.sem), +}; + +static int sa1111_audio_open(struct inode *inode, struct file *file) +{ + return sa1100_audio_attach(inode, file, &audio_state); +} + +/* + * Missing fields of this structure will be patched with the call + * to sa1100_audio_attach(). + */ +static struct file_operations sa1111_audio_fops = { + open: sa1111_audio_open, + owner: THIS_MODULE +}; + +static int audio_dev_id, mixer_dev_id; + +static int __init sa1111_uda1341_init(void) +{ + struct uda1341_cfg cfg; + int ret; + + if ( !( (machine_is_assabet() && machine_has_neponset()) || + machine_is_jornada720() || + machine_is_badge4() )) + return -ENODEV; + + if (!request_mem_region(_SACR0, 512, "sound")) + return -EBUSY; + + ret = l3_attach_client(&uda1341, "l3-sa1111", "uda1341"); + if (ret) + goto out; + + /* Acquire and initialize DMA */ + ret = sa1111_sac_request_dma(&output_stream.dma_ch, "SA1111 audio out", + SA1111_SAC_XMT_CHANNEL); + if (ret < 0) + goto release_l3; + + ret = sa1111_sac_request_dma(&input_stream.dma_ch, "SA1111 audio in", + SA1111_SAC_RCV_CHANNEL); + if (ret < 0) + goto release_dma; + + cfg.fs = 256; + cfg.format = FMT_I2S; + l3_command(&uda1341, L3_UDA1341_CONFIGURE, &cfg); + + /* register devices */ + audio_dev_id = register_sound_dsp(&sa1111_audio_fops, -1); + mixer_dev_id = register_sound_mixer(&uda1341_mixer_fops, -1); + + printk(KERN_INFO "Sound: SA1111 UDA1341: dsp id %d mixer id %d\n", + audio_dev_id, mixer_dev_id); + return 0; + +release_dma: + sa1100_free_dma(output_stream.dma_ch); +release_l3: + l3_detach_client(&uda1341); +out: + release_mem_region(_SACR0, 512); + return ret; +} + +static void __exit sa1111_uda1341_exit(void) +{ + unregister_sound_dsp(audio_dev_id); + unregister_sound_mixer(mixer_dev_id); + sa1100_free_dma(output_stream.dma_ch); + sa1100_free_dma(input_stream.dma_ch); + l3_detach_client(&uda1341); + + release_mem_region(_SACR0, 512); +} + +module_init(sa1111_uda1341_init); +module_exit(sa1111_uda1341_exit); + +MODULE_AUTHOR("John Dorsey, Nicolas Pitre"); +MODULE_DESCRIPTION("Glue audio driver for the SA1111 compagnon chip & Philips UDA1341 codec."); + +EXPORT_NO_SYMBOLS; diff -urN orig/drivers/sound/uda1341.c linux/drivers/sound/uda1341.c --- orig/drivers/sound/uda1341.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/sound/uda1341.c Wed Oct 24 23:05:21 2001 @@ -0,0 +1,511 @@ +/* + * Philips UDA1341 mixer device driver + * + * Copyright (c) 2000 Nicolas Pitre + * + * Portions are Copyright (C) 2000 Lernout & Hauspie Speech Products, N.V. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License. + * + * History: + * + * 2000-05-21 Nicolas Pitre Initial release. + * + * 2000-08-19 Erik Bunce More inline w/ OSS API and UDA1341 docs + * including fixed AGC and audio source handling + * + * 2000-11-30 Nicolas Pitre - More mixer functionalities. + * + * 2001-06-03 Nicolas Pitre Made this file a separate module, based on + * the former sa1100-uda1341.c driver. + * + * 2001-08-13 Russell King Re-written as part of the L3 interface + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include + +#define DEF_VOLUME 65 + +/* + * UDA1341 L3 address and command types + */ +#define UDA1341_L3ADDR 5 +#define UDA1341_DATA0 (UDA1341_L3ADDR << 2 | 0) +#define UDA1341_DATA1 (UDA1341_L3ADDR << 2 | 1) +#define UDA1341_STATUS (UDA1341_L3ADDR << 2 | 2) + +struct uda1341_regs { + unsigned char stat0; +#define STAT0 0x00 +#define STAT0_RST (1 << 6) +#define STAT0_SC_MASK (3 << 4) +#define STAT0_SC_512FS (0 << 4) +#define STAT0_SC_384FS (1 << 4) +#define STAT0_SC_256FS (2 << 4) +#define STAT0_IF_MASK (7 << 1) +#define STAT0_IF_I2S (0 << 1) +#define STAT0_IF_LSB16 (1 << 1) +#define STAT0_IF_LSB18 (2 << 1) +#define STAT0_IF_LSB20 (3 << 1) +#define STAT0_IF_MSB (4 << 1) +#define STAT0_IF_LSB16MSB (5 << 1) +#define STAT0_IF_LSB18MSB (6 << 1) +#define STAT0_IF_LSB20MSB (7 << 1) +#define STAT0_DC_FILTER (1 << 0) + + unsigned char stat1; +#define STAT1 0x80 +#define STAT1_DAC_GAIN (1 << 6) /* gain of DAC */ +#define STAT1_ADC_GAIN (1 << 5) /* gain of ADC */ +#define STAT1_ADC_POL (1 << 4) /* polarity of ADC */ +#define STAT1_DAC_POL (1 << 3) /* polarity of DAC */ +#define STAT1_DBL_SPD (1 << 2) /* double speed playback */ +#define STAT1_ADC_ON (1 << 1) /* ADC powered */ +#define STAT1_DAC_ON (1 << 0) /* DAC powered */ + + unsigned char data0_0; +#define DATA0 0x00 +#define DATA0_VOLUME_MASK 0x3f +#define DATA0_VOLUME(x) (x) + + unsigned char data0_1; +#define DATA1 0x40 +#define DATA1_BASS(x) ((x) << 2) +#define DATA1_BASS_MASK (15 << 2) +#define DATA1_TREBLE(x) ((x)) +#define DATA1_TREBLE_MASK (3) + + unsigned char data0_2; +#define DATA2 0x80 +#define DATA2_PEAKAFTER (1 << 5) +#define DATA2_DEEMP_NONE (0 << 3) +#define DATA2_DEEMP_32KHz (1 << 3) +#define DATA2_DEEMP_44KHz (2 << 3) +#define DATA2_DEEMP_48KHz (3 << 3) +#define DATA2_MUTE (1 << 2) +#define DATA2_FILTER_FLAT (0 << 0) +#define DATA2_FILTER_MIN (1 << 0) +#define DATA2_FILTER_MAX (3 << 0) + +#define EXTADDR(n) (0xc0 | (n)) +#define EXTDATA(d) (0xe0 | (d)) + + unsigned char ext0; +#define EXT0 0 +#define EXT0_CH1_GAIN(x) (x) + + unsigned char ext1; +#define EXT1 1 +#define EXT1_CH2_GAIN(x) (x) + + unsigned char ext2; +#define EXT2 2 +#define EXT2_MIC_GAIN_MASK (7 << 2) +#define EXT2_MIC_GAIN(x) ((x) << 2) +#define EXT2_MIXMODE_DOUBLEDIFF (0) +#define EXT2_MIXMODE_CH1 (1) +#define EXT2_MIXMODE_CH2 (2) +#define EXT2_MIXMODE_MIX (3) + + unsigned char ext4; +#define EXT4 4 +#define EXT4_AGC_ENABLE (1 << 4) +#define EXT4_INPUT_GAIN_MASK (3) +#define EXT4_INPUT_GAIN(x) ((x) & 3) + + unsigned char ext5; +#define EXT5 5 +#define EXT5_INPUT_GAIN(x) ((x) >> 2) + + unsigned char ext6; +#define EXT6 6 +#define EXT6_AGC_CONSTANT_MASK (7 << 2) +#define EXT6_AGC_CONSTANT(x) ((x) << 2) +#define EXT6_AGC_LEVEL_MASK (3) +#define EXT6_AGC_LEVEL(x) (x) +}; + +#define REC_MASK (SOUND_MASK_LINE | SOUND_MASK_MIC) +#define DEV_MASK (REC_MASK | SOUND_MASK_VOLUME | SOUND_MASK_BASS | SOUND_MASK_TREBLE) + +struct uda1341 { + struct uda1341_regs regs; + int active; + unsigned short volume; + unsigned short bass; + unsigned short treble; + unsigned short line; + unsigned short mic; + int mod_cnt; +}; + +#define ADD_FIELD(reg,field) \ + *p++ = reg | uda->regs.field + +#define ADD_EXTFIELD(reg,field) \ + *p++ = EXTADDR(reg); \ + *p++ = EXTDATA(uda->regs.field); + +static void uda1341_sync(struct l3_client *clnt) +{ + struct uda1341 *uda = clnt->driver_data; + char buf[24], *p = buf; + + ADD_FIELD(STAT0, stat0); + ADD_FIELD(STAT1, stat1); + + if (p != buf) + l3_write(clnt, UDA1341_STATUS, buf, p - buf); + + p = buf; + ADD_FIELD(DATA0, data0_0); + ADD_FIELD(DATA1, data0_1); + ADD_FIELD(DATA2, data0_2); + ADD_EXTFIELD(EXT0, ext0); + ADD_EXTFIELD(EXT1, ext1); + ADD_EXTFIELD(EXT2, ext2); + ADD_EXTFIELD(EXT4, ext4); + ADD_EXTFIELD(EXT5, ext5); + ADD_EXTFIELD(EXT6, ext6); + + if (p != buf) + l3_write(clnt, UDA1341_DATA0, buf, p - buf); +} + +static void uda1341_cmd_init(struct l3_client *clnt) +{ + struct uda1341 *uda = clnt->driver_data; + char buf[2]; + + uda->active = 1; + + buf[0] = uda->regs.stat0 | STAT0_RST; + buf[1] = uda->regs.stat0; + + l3_write(clnt, UDA1341_STATUS, buf, 2); + + /* resend all parameters */ + uda1341_sync(clnt); +} + +static int uda1341_configure(struct l3_client *clnt, struct uda1341_cfg *conf) +{ + struct uda1341 *uda = clnt->driver_data; + int ret = 0; + + uda->regs.stat0 &= ~(STAT0_SC_MASK | STAT0_IF_MASK); + + switch (conf->fs) { + case 512: uda->regs.stat0 |= STAT0_SC_512FS; break; + case 384: uda->regs.stat0 |= STAT0_SC_384FS; break; + case 256: uda->regs.stat0 |= STAT0_SC_256FS; break; + default: ret = -EINVAL; break; + } + + switch (conf->format) { + case FMT_I2S: uda->regs.stat0 |= STAT0_IF_I2S; break; + case FMT_LSB16: uda->regs.stat0 |= STAT0_IF_LSB16; break; + case FMT_LSB18: uda->regs.stat0 |= STAT0_IF_LSB18; break; + case FMT_LSB20: uda->regs.stat0 |= STAT0_IF_LSB20; break; + case FMT_MSB: uda->regs.stat0 |= STAT0_IF_MSB; break; + case FMT_LSB16MSB: uda->regs.stat0 |= STAT0_IF_LSB16MSB; break; + case FMT_LSB18MSB: uda->regs.stat0 |= STAT0_IF_LSB18MSB; break; + case FMT_LSB20MSB: uda->regs.stat0 |= STAT0_IF_LSB20MSB; break; + } + + if (ret == 0 && uda->active) { + char buf = uda->regs.stat0 | STAT0; + l3_write(clnt, UDA1341_STATUS, &buf, 1); + } + return ret; +} + +static int uda1341_update_direct(struct l3_client *clnt, int cmd, void *arg) +{ + struct uda1341 *uda = clnt->driver_data; + struct l3_gain *v = arg; + char newreg; + int val; + + switch (cmd) { + case L3_SET_VOLUME: /* set volume. val = 0 to 100 => 62 to 1 */ + uda->regs.data0_0 = DATA0_VOLUME(62 - ((v->left * 61) / 100)); + newreg = uda->regs.data0_0 | DATA0; + break; + + case L3_SET_BASS: /* set bass. val = 50 to 100 => 0 to 12 */ + val = v->left - 50; + if (val < 0) + val = 0; + uda->regs.data0_1 &= ~DATA1_BASS_MASK; + uda->regs.data0_1 |= DATA1_BASS((val * 12) / 50); + newreg = uda->regs.data0_1 | DATA1; + break; + + case L3_SET_TREBLE: /* set treble. val = 50 to 100 => 0 to 3 */ + val = v->left - 50; + if (val < 0) + val = 0; + uda->regs.data0_1 &= ~DATA1_TREBLE_MASK; + uda->regs.data0_1 |= DATA1_TREBLE((val * 3) / 50); + newreg = uda->regs.data0_1 | DATA1; + break; + + default: + return -EINVAL; + } + + if (uda->active) + l3_write(clnt, UDA1341_DATA0, &newreg, 1); + return 0; +} + +static int uda1341_update_indirect(struct l3_client *clnt, int cmd, void *arg) +{ + struct uda1341 *uda = clnt->driver_data; + struct l3_gain *gain = arg; + struct l3_agc *agc = arg; + char buf[8], *p = buf; + int val, ret = 0; + + switch (cmd) { + case L3_SET_GAIN: + val = 31 - (gain->left * 31 / 100); + switch (gain->channel) { + case 1: + uda->regs.ext0 = EXT0_CH1_GAIN(val); + ADD_EXTFIELD(EXT0, ext0); + break; + + case 2: + uda->regs.ext1 = EXT1_CH2_GAIN(val); + ADD_EXTFIELD(EXT1, ext1); + break; + + default: + ret = -EINVAL; + } + break; + + case L3_INPUT_AGC: + if (agc->channel == 2) { + if (agc->enable) + uda->regs.ext4 |= EXT4_AGC_ENABLE; + else + uda->regs.ext4 &= ~EXT4_AGC_ENABLE; +#if 0 + agc->level + agc->attack + agc->decay +#endif + ADD_EXTFIELD(EXT4, ext4); + } else + ret = -EINVAL; + break; + + default: + ret = -EINVAL; + break; + } + + if (ret == 0 && uda->active) + l3_write(clnt, UDA1341_DATA0, buf, p - buf); + + return ret; +} + +static int uda1341_mixer_ioctl(struct l3_client *clnt, int cmd, void *arg) +{ + struct uda1341 *uda = clnt->driver_data; + struct l3_gain gain; + int val, nr = _IOC_NR(cmd), ret = 0; + + if (cmd == SOUND_MIXER_INFO) { + struct mixer_info mi; + + strncpy(mi.id, "UDA1341", sizeof(mi.id)); + strncpy(mi.name, "Philips UDA1341", sizeof(mi.name)); + mi.modify_counter = uda->mod_cnt; + return copy_to_user(arg, &mi, sizeof(mi)); + } + + if (_IOC_DIR(cmd) & _IOC_WRITE) { + ret = get_user(val, (int *)arg); + if (ret) + goto out; + + gain.left = val & 255; + gain.right = val >> 8; + gain.channel = 0; + + switch (nr) { + case SOUND_MIXER_VOLUME: + uda->volume = val; + uda->mod_cnt++; + uda1341_update_direct(clnt, L3_SET_VOLUME, &gain); + break; + + case SOUND_MIXER_BASS: + uda->bass = val; + uda->mod_cnt++; + uda1341_update_direct(clnt, L3_SET_BASS, &gain); + break; + + case SOUND_MIXER_TREBLE: + uda->treble = val; + uda->mod_cnt++; + uda1341_update_direct(clnt, L3_SET_TREBLE, &gain); + break; + + case SOUND_MIXER_LINE: + uda->line = val; + gain.channel = 1; + uda->mod_cnt++; + uda1341_update_indirect(clnt, L3_SET_GAIN, &gain); + break; + + case SOUND_MIXER_MIC: + uda->mic = val; + gain.channel = 2; + uda->mod_cnt++; + uda1341_update_indirect(clnt, L3_SET_GAIN, &gain); + break; + + case SOUND_MIXER_RECSRC: + break; + + default: + ret = -EINVAL; + } + } + + if (ret == 0 && _IOC_DIR(cmd) & _IOC_READ) { + int nr = _IOC_NR(cmd); + ret = 0; + + switch (nr) { + case SOUND_MIXER_VOLUME: val = uda->volume; break; + case SOUND_MIXER_BASS: val = uda->bass; break; + case SOUND_MIXER_TREBLE: val = uda->treble; break; + case SOUND_MIXER_LINE: val = uda->line; break; + case SOUND_MIXER_MIC: val = uda->mic; break; + case SOUND_MIXER_RECSRC: val = REC_MASK; break; + case SOUND_MIXER_RECMASK: val = REC_MASK; break; + case SOUND_MIXER_DEVMASK: val = DEV_MASK; break; + case SOUND_MIXER_CAPS: val = 0; break; + case SOUND_MIXER_STEREODEVS: val = 0; break; + default: val = 0; ret = -EINVAL; break; + } + + if (ret == 0) + ret = put_user(val, (int *)arg); + } +out: + return ret; +} + +static int uda1341_attach(struct l3_client *clnt) +{ + struct uda1341 *uda; + + uda = kmalloc(sizeof(*uda), GFP_KERNEL); + if (!uda) + return -ENOMEM; + + memset(uda, 0, sizeof(*uda)); + + uda->volume = DEF_VOLUME | DEF_VOLUME << 8; + uda->bass = 50 | 50 << 8; + uda->treble = 50 | 50 << 8; + uda->line = 88 | 88 << 8; + uda->mic = 88 | 88 << 8; + + uda->regs.stat0 = STAT0_SC_256FS | STAT0_IF_LSB16; + uda->regs.stat1 = STAT1_DAC_GAIN | STAT1_ADC_GAIN | + STAT1_ADC_ON | STAT1_DAC_ON; + uda->regs.data0_0 = DATA0_VOLUME(62 - ((DEF_VOLUME * 61) / 100)); + uda->regs.data0_1 = DATA1_BASS(0) | DATA1_TREBLE(0); + uda->regs.data0_2 = DATA2_PEAKAFTER | DATA2_DEEMP_NONE | + DATA2_FILTER_MAX; + uda->regs.ext0 = EXT0_CH1_GAIN(4); + uda->regs.ext1 = EXT1_CH2_GAIN(4); + uda->regs.ext2 = EXT2_MIXMODE_MIX | EXT2_MIC_GAIN(4); + uda->regs.ext4 = EXT4_AGC_ENABLE | EXT4_INPUT_GAIN(0); + uda->regs.ext5 = EXT5_INPUT_GAIN(0); + uda->regs.ext6 = EXT6_AGC_CONSTANT(3) | EXT6_AGC_LEVEL(0); + + clnt->driver_data = uda; + + return 0; +} + +static void uda1341_detach(struct l3_client *clnt) +{ + kfree(clnt->driver_data); +} + +static int +uda1341_command(struct l3_client *clnt, int cmd, void *arg) +{ + int ret = -EINVAL; + + if (_IOC_TYPE(cmd) == 'M') + ret = uda1341_mixer_ioctl(clnt, cmd, arg); + else if (cmd == L3_UDA1341_CONFIGURE) + ret = uda1341_configure(clnt, arg); + + return ret; +} + +static int uda1341_open(struct l3_client *clnt) +{ + uda1341_cmd_init(clnt); + return 0; +} + +static void uda1341_close(struct l3_client *clnt) +{ + struct uda1341 *uda = clnt->driver_data; + uda->active = 0; +} + +static struct l3_ops uda1341_ops = { + open: uda1341_open, + command: uda1341_command, + close: uda1341_close, +}; + +static struct l3_driver uda1341 = { + name: UDA1341_NAME, + attach_client: uda1341_attach, + detach_client: uda1341_detach, + ops: &uda1341_ops, + owner: THIS_MODULE, +}; + +static int __init uda1341_init(void) +{ + return l3_add_driver(&uda1341); +} + +static void __exit uda1341_exit(void) +{ + l3_del_driver(&uda1341); +} + +module_init(uda1341_init); +module_exit(uda1341_exit); + +MODULE_AUTHOR("Nicolas Pitre"); +MODULE_DESCRIPTION("Philips UDA1341 CODEC driver"); diff -urN orig/drivers/sound/waveartist.c linux/drivers/sound/waveartist.c --- orig/drivers/sound/waveartist.c Fri Nov 16 10:10:11 2001 +++ linux/drivers/sound/waveartist.c Thu May 2 17:38:28 2002 @@ -247,17 +247,15 @@ printk("\n"); } - if (inb(io_base + STATR) & CMD_RF) { - int old_data; - - /* flush the port - */ + /* + * flush any stale command data from the port. + */ + while (inb(io_base + STATR) & CMD_RF) { + unsigned int old_data; old_data = inw(io_base + CMDR); - - if (debug_flg & DEBUG_CMD) - printk("flushed %04X...", old_data); - + printk("waveartist: flushing stale command data: 0x%04x pc=%p\n", + old_data, __builtin_return_address(0)); udelay(10); } @@ -287,16 +285,19 @@ resp[i] = inw(io_base + CMDR); } - if (debug_flg & DEBUG_CMD) { - if (!timed_out) { - printk("waveartist_cmd: resp="); - - for (i = 0; i < nr_resp; i++) - printk("%04X ", resp[i]); - - printk("\n"); - } else - printk("waveartist_cmd: timed out\n"); + if (debug_flg & DEBUG_CMD && !timed_out) { + printk("waveartist_cmd: resp="); + + for (i = 0; i < nr_resp; i++) + printk("%04X ", resp[i]); + printk("\n"); + } + + if (timed_out) { + printk(KERN_ERR "waveartist_cmd: command timed out:"); + for (i = 0; i < nr_cmd; i++) + printk(" %04x", cmd[i]); + printk("\n"); } return timed_out ? 1 : 0; @@ -495,7 +496,6 @@ audio_devs[dev]->flags & DMA_AUTOMODE && intrflag && count == devc->xfer_count) { - devc->audio_mode |= PCM_ENABLE_INPUT; return; /* * Auto DMA mode on. No need to react */ @@ -570,9 +570,6 @@ wavnc_info *devc = (wavnc_info *) audio_devs[dev]->devc; wavnc_port_info *portc = (wavnc_port_info *) audio_devs[dev]->portc; unsigned int speed, bits; - - if (devc->audio_mode) - return 0; speed = waveartist_get_speed(portc); bits = waveartist_get_bits(portc); diff -urN orig/drivers/ssi/Config.in linux/drivers/ssi/Config.in --- orig/drivers/ssi/Config.in Thu Jan 1 01:00:00 1970 +++ linux/drivers/ssi/Config.in Thu Feb 22 19:03:41 2001 @@ -0,0 +1,11 @@ + +mainmenu_option next_comment +comment 'Synchronous Serial Interface' +tristate 'Synchronous Serial Interface Support' CONFIG_SSI + +comment 'SSI Bus Drivers' +dep_tristate ' CLPS711X SSI support' CONFIG_SSI_CLPS711X $CONFIG_SSI $CONFIG_ARCH_CLPS711X + +comment 'SSI Device Drivers' +dep_tristate ' JUNO keyboard support' CONFIG_SSI_JUNO $CONFIG_SSI +endmenu diff -urN orig/drivers/ssi/Makefile linux/drivers/ssi/Makefile --- orig/drivers/ssi/Makefile Thu Jan 1 01:00:00 1970 +++ linux/drivers/ssi/Makefile Wed Dec 20 17:11:33 2000 @@ -0,0 +1,50 @@ +# +# Makefile for the SSI drivers +# +# Note! Dependencies are done automagically by 'make dep', which also +# removes any old dependencies. DON'T put your own dependencies here +# unless it's something special (ie not a .c file). +# +# Note 2! The CFLAGS definition is now inherited from the +# parent makefile. +# + +O_TARGET := ssi.o + +obj-y := +obj-m := +obj-n := +obj- := + +export-objs := +list-multi := + +obj-$(CONFIG_SSI) += ssi_core.o +obj-$(CONFIG_SSI_CLPS711X) += clps711x_ssi1.o +obj-y += juno.o + +# Extract lists of the multi-part drivers. +# The 'int-*' lists are intermediate files used to build the multi's. + +multi-y := $(filter $(list-multi), $(obj-y)) +multi-m := $(filter $(list-multi), $(obj-m)) +int-y := $(sort $(foreach m, $(multi-y), $($(basename $(m))-objs))) +int-m := $(sort $(foreach m, $(multi-m), $($(basename $(m))-objs))) + +# Files that are both resident and modular; remove from modular. + +obj-m := $(filter-out $(obj-y), $(obj-m)) +int-m := $(filter-out $(int-y), $(int-m)) + +# Take multi-part drivers out of obj-y and put components in. + +obj-y := $(filter-out $(list-multi), $(obj-y)) $(int-y) + +# Translate to Rules.make lists. + +O_OBJS := $(filter-out $(export-objs), $(obj-y)) +OX_OBJS := $(filter $(export-objs), $(obj-y)) +M_OBJS := $(sort $(filter-out $(export-objs), $(obj-m))) +MX_OBJS := $(sort $(filter $(export-objs), $(obj-m))) + +include $(TOPDIR)/Rules.make diff -urN orig/drivers/ssi/README linux/drivers/ssi/README --- orig/drivers/ssi/README Thu Jan 1 01:00:00 1970 +++ linux/drivers/ssi/README Wed Dec 20 17:16:57 2000 @@ -0,0 +1,86 @@ + Synchronous Serial Interface bus driver + --------------------------------------- + + EEEEE X X PPPP EEEEE RRRR IIIII M M EEEEE N N TTTTT AAA L + E X X P P E R R I MM MM E NN N T A A L + EEEE X PPPP EEEE RRRR I M M M EEEE N N N T AAAAA L + E X X P E R R I M M E N NN T A A L + EEEEE X X P EEEEE R R IIIII M M EEEEE N N T A A LLLLL + +This directory holds the SSI bus drivers. Basically, a SSI bus consists +of the following signals: + + stxd Transmit data + srxd Receive data + sclk Clock + sfrm Frame + Chip selects (1 - n) + +There may be more than one device on a SSI bus, and each device can +have different timing requirements. There are several frame formats: + +1. Texas Instruments Synchronous Serial Frame format + + sclk ____~_~_~_~_~_~_~_~____ + sfrm ____~~_________________ + stxd ------bn..........b0--- + srxd ------bn..........b0--- + + - data latched in on the falling edge of the clock + - data shifted out on the rising edge of the clock + +2. Motorola SPI frame format + + sclk ______~_~_~_~_~_~_~____ + sfrm ~~~~________________~~~ + stxd -----bn..........b0---- + srxd ----.bn..........b0---- + + - data latched in on the rising edge of the clock + - data shifted out on the falling edge of the clock, or falling edge + of sfrm + +3. National Microwire format + + sclk ______~_~_~_~_~_~_~_~_~_~_~_~_~_____ + sfrm ~~~~_____________________________~~~ + stxd -----bn......b0--------------------- + srxd -----------------bn..........b0.---- + + - data latched in on the rising edge of the clock + - data shifted out on the falling edge of the clock + - half duplex, one clock between transmission and reception + +Types of devices +---------------- + +The following types of devices can be found on a SSP bus: + + Sound chips + Keyboard devices + Touch screen devices + +Keyboard +-------- + +TX: +0. Format: cfglen = 8, framelen = 8, clkpol = 1, clk < 250kHz +1. select device +2. keyboard responds asserting key_atn +3. wait 0.1ms to 5ms +4. transmit data byte +5. wait >= 150us +6. repeat step 4 until all data sent +7. deselect device +8. keyboard responds de-asserting key_atn +9. wait >= 120us + +RX: +0. Format: cfglen = 8, framelen = 8, clkpol = 1, clk < 250kHz +1. keyboard asserts key_atn +2. select device after 0.1ms < 5ms +3. read data byte +4. wait 150us +5. if key_atn still asserted, goto 3 +6. deselect device +7. wait >= 120us diff -urN orig/drivers/ssi/clps711x_ssi1.c linux/drivers/ssi/clps711x_ssi1.c --- orig/drivers/ssi/clps711x_ssi1.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/ssi/clps711x_ssi1.c Wed Dec 20 17:11:33 2000 @@ -0,0 +1,237 @@ +/* + * linux/drivers/ssi/clps711x_ssi1.c + * + * SSI bus driver for the CLPS711x SSI1 bus. We support EP7212 + * extensions as well. + * + * Frame sizes can be between 4 and 24 bits. + * Config sizes can be between 4 and 16 bits. + */ +#include +#include +#include + +#include +#include +#include + +#include + +#include "ssi_bus.h" +#include "ssi_dev.h" + +#define EP7212 + +/* + * Port E on the P720T is used for the SPI bus chip selects + * 0 - Keyboard + * 1 - Touch screen + * 2 - CS4224 ADC/DAC + * 7 - idle + */ + +#if 0 +/* + * we place in the transmit buffer: + * + * received data (binary): + * 0xxxxxxxxxxxx000 + * where 'x' is the value + */ +struct ssi_dev ads7846_dev = { + name: "ADS7846", + id: 1, + proto: SSI_SPI, + cfglen: 8, + framelen: 24, + clkpol: 0, + clkfreq: 2500000, +}; + +/* + * we place in the transmit buffer: + * write: <20> ... + * received data discarded + */ +struct ssi_dev cs4224_dev = { + name: "CS4224", + id: 2, + proto: SSI_SPI, + cfglen: 8, + framelen: 8, + clkpol: 0, + clkfreq: 6000000, +}; +#endif + +/* + * Supplement with whatever method your board requires + */ +static void ssi1_select_id(int id) +{ + if (machine_is_p720t()) { + clps_writel(7, PEDDR); + clps_writel(id, PEDR); + } +} + +/* + * Select the specified device. The upper layers will have already + * checked that the bus transmit queue is idle. We need to make sure + * that the interface itself is idle. + */ +static int ssi1_select(struct ssi_bus *bus, struct ssi_dev *dev) +{ + u_int id = dev ? dev->id : 7; + u_int val; + + /* + * Make sure that the interface is idle + */ + do { + val = clps_readl(SYSFLG1); + } while (val & SYSFLG1_SSIBUSY); + + ssi1_select_id(7); + + if (dev) { + /* + * Select clock frequency. This is very rough, + * and assumes that we're operating in PLL mode. + */ + val = clps_readl(SYSCON1) & ~SYSCON1_ADCKSEL_MASK; +// if (dev->clkfreq <= 16000) /* <16kHz */ +// val |= SYSCON1_ADCKSEL(0); +// else if (dev->clkfreq < 64000) /* <64kHz */ +// val |= SYSCON1_ADCKSEL(1); +// else if (dev->clkfreq < 128000) /* <128kHz */ + val |= SYSCON1_ADCKSEL(2); +// else /* >= 128kHz */ +// val |= SYSCON1_ADCKSEL(3); + clps_writel(val, SYSCON1); + + bus->cfglen = dev->cfglen; + bus->framelen = dev->framelen; + bus->clkpol = dev->clkpol; + bus->proto = dev->proto; + +#ifdef EP7212 + /* + * Set the clock edge according to the device. + * (set clkpol if the device reads data on the + * falling edge of the clock signal). + */ + val = ep_readl(SYSCON3) & ~SYSCON3_ADCCKNSEN; + if (bus->clkpol && dev->proto != SSI_USAR) + val |= SYSCON3_ADCCKNSEN; + ep_writel(val, SYSCON3); +#endif + + /* + * Select the device + */ + ssi1_select_id(id); + +#ifdef EP7212 + /* + * If we are doing USAR, wait 30us, then set + * the clock line low. + */ + if (dev->proto == SSI_USAR) { + udelay(150); + + val |= SYSCON3_ADCCKNSEN; + ep_writel(val, SYSCON3); + } +#endif + } + + return 0; +} + +static void ssi1_int(int irq, void *dev_id, struct pt_regs *regs) +{ + struct ssi_bus *bus = (struct ssi_bus *)dev_id; + + /* + * Read the data word and queue it. + */ + ssi_core_rcv(bus, clps_readl(SYNCIO)); +} + +/* + * Enable transmission and/or of some bytes + */ +static int ssi1_trans(struct ssi_bus *bus, u_int data) +{ + u_int syncio; + +#ifdef EP7212 + data <<= 32 - bus->cfglen; + syncio = bus->cfglen | bus->framelen << 7 | data; +#else + syncio = data | bus->framelen << 8; +#endif + + clps_writel(syncio, SYNCIO); + clps_writel(syncio | SYNCIO_TXFRMEN, SYNCIO); + return 0; +} + +/* + * Initialise the SSI bus. + */ +static int ssi1_bus_init(struct ssi_bus *bus) +{ + int retval, val; + + retval = request_irq(IRQ_SSEOTI, ssi1_int, 0, "ssi1", bus); + if (retval) + return retval; + +#ifdef EP7212 + /* + * EP7212 only! Set the configuration command length. + * On the CLPS711x chips, it is fixed at 8 bits. + */ + val = ep_readl(SYSCON3); + val |= SYSCON3_ADCCON; + ep_writel(val, SYSCON3); +#endif + + ssi1_select(bus, NULL); + + PLD_SPI |= PLD_SPI_EN; + + return 0; +} + +static void ssi1_bus_exit(struct ssi_bus *bus) +{ + ssi1_select(bus, NULL); + + PLD_SPI &= ~PLD_SPI_EN; + + free_irq(IRQ_SSEOTI, bus); +} + +struct ssi_bus clps711x_ssi1_bus = { + name: "clps711x_ssi1", + init: ssi1_bus_init, + exit: ssi1_bus_exit, + select: ssi1_select, + trans: ssi1_trans, +}; + +static int __init clps711x_ssi1_init(void) +{ + return ssi_register_bus(&clps711x_ssi1_bus); +} + +static void __exit clps711x_ssi1_exit(void) +{ + ssi_unregister_bus(&clps711x_ssi1_bus); +} + +module_init(clps711x_ssi1_init); +module_exit(clps711x_ssi1_exit); diff -urN orig/drivers/ssi/juno.c linux/drivers/ssi/juno.c --- orig/drivers/ssi/juno.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/ssi/juno.c Wed Dec 20 17:11:33 2000 @@ -0,0 +1,131 @@ +#include +#include +#include +#include +#include + +#include +#include + +#include + +#include "ssi_bus.h" +#include "ssi_dev.h" + +extern struct ssi_bus clps711x_ssi1_bus; + +static u_int recvbuf[16]; +static volatile u_int ptr, rxed; + +static inline void juno_enable_irq(void) +{ + enable_irq(IRQ_EINT1); +} + +static inline void juno_disable_irq(void) +{ + disable_irq(IRQ_EINT1); +} + +static void juno_rcv(struct ssi_dev *dev, u_int data) +{ + if (ptr < 16) { + recvbuf[ptr] = data; + ptr++; + } else + printk("juno_rcv: %04x\n", data); + rxed = 1; +} + +static void juno_irq(int irq, void *dev_id, struct pt_regs *regs) +{ + struct ssi_dev *dev = dev_id; + + printk("juno_irq\n"); + + ssi_select_device(dev->bus, dev); + + ptr = 0; + do { + rxed = 0; + ssi_transmit_data(dev, 0xff); + while (rxed == 0); + udelay(150); + } while (PLD_INT & PLD_INT_KBD_ATN); + + ssi_select_device(dev->bus, NULL); + + { int i; + printk("juno_rcv: "); + for (i = 0; i < ptr; i++) + printk("%04x ", recvbuf[i]); + printk("\n"); + } +} + +static void juno_command(struct ssi_dev *dev, int cmd, int data) +{ + ssi_transmit_data(dev, cmd); + mdelay(1); + ssi_transmit_data(dev, data); + mdelay(1); + ssi_transmit_data(dev, 0xa0 ^ 0xc0); + mdelay(1); +} + +static int juno_dev_init(struct ssi_dev *dev) +{ + int retval; + + PLD_KBD |= PLD_KBD_EN; + ptr = 16; + + mdelay(20); + + retval = request_irq(IRQ_EINT1, juno_irq, 0, dev->name, dev); + if (retval) + return retval; + + juno_disable_irq(); + + if (ssi_select_device(dev->bus, dev) != 0) { + printk("juno: ssi_select_dev failed\n"); + return -EBUSY; + } + + mdelay(1); + + juno_command(dev, 0x80, 0x20); + + ssi_select_device(dev->bus, NULL); + + juno_enable_irq(); + + return 0; +} + +static struct ssi_dev juno_dev = { + name: "Juno", + id: 0, + proto: SSI_USAR, + cfglen: 8, + framelen: 8, + clkpol: 1, + clkfreq: 250000, + rcv: juno_rcv, + init: juno_dev_init, +}; + +static int __init juno_init(void) +{ + return ssi_register_device(&clps711x_ssi1_bus, &juno_dev); +} + +static void __exit juno_exit(void) +{ + ssi_unregister_device(&juno_dev); +} + +module_init(juno_init); +module_exit(juno_exit); + diff -urN orig/drivers/ssi/ssi_bus.h linux/drivers/ssi/ssi_bus.h --- orig/drivers/ssi/ssi_bus.h Thu Jan 1 01:00:00 1970 +++ linux/drivers/ssi/ssi_bus.h Wed Dec 20 17:11:32 2000 @@ -0,0 +1,21 @@ +#include + +struct ssi_dev; + +struct ssi_bus { + u_char cfglen; + u_char framelen; + u_char clkpol; + u_char proto; + struct ssi_dev *dev; /* current device */ + int (*select)(struct ssi_bus *, struct ssi_dev *); + int (*trans)(struct ssi_bus *, u_int data); + int (*init)(struct ssi_bus *); + void (*exit)(struct ssi_bus *); + char *name; + u_int devices; +}; + +extern int ssi_core_rcv(struct ssi_bus *bus, u_int data); +extern int ssi_register_bus(struct ssi_bus *bus); +extern int ssi_unregister_bus(struct ssi_bus *bus); diff -urN orig/drivers/ssi/ssi_core.c linux/drivers/ssi/ssi_core.c --- orig/drivers/ssi/ssi_core.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/ssi/ssi_core.c Wed Dec 20 17:11:32 2000 @@ -0,0 +1,175 @@ +/* + * linux/drivers/ssi/ssi_core.c + * + * This file provides a common framework to allow multiple SSI devices + * to work together on a single SSI bus. + * + * You can use this in two ways: + * 1. select the device, queue up data, flush the data to the device, + * (optionally) purge the received data, deselect the device. + * 2. select the device, queue up one data word, flush to the device + * read data word, queue up next data word, flush to the device... + * deselect the device. + */ +#include +#include +#include +#include +#include + +#include + +#include "ssi_bus.h" +#include "ssi_dev.h" + +#define DEBUG + +/** + * ssi_core_rcv - pass received SSI data to the device + * @bus: the bus that the data was received from + * @data: the data word that was received + * + * This function is intended to be called by SSI bus device + * drivers to pass received data to the device driver. + */ +int ssi_core_rcv(struct ssi_bus *bus, u_int data) +{ + struct ssi_dev *dev = bus->dev; + + if (dev && dev->rcv) + dev->rcv(dev, data); + + return 0; +} + +/** + * ssi_transmit_data - queue SSI data for later transmission + * @dev: device requesting data to be transmitted + * @data: data word to be transmitted. + * + * Queue one data word of SSI data for later transmission. + */ +int ssi_transmit_data(struct ssi_dev *dev, u_int data) +{ + struct ssi_bus *bus = dev->bus; + + /* + * Make sure that we currently own the bus + */ + if (bus->dev != dev) + BUG(); + + bus->trans(bus, data); + return 0; +} + +/** + * ssi_select_device - select a SSI device for later transactions + * @dev: device to be selected + */ +int ssi_select_device(struct ssi_bus *bus, struct ssi_dev *dev) +{ + int retval; + +#ifdef DEBUG + printk("SSI: selecting device %s on bus %s\n", + dev ? dev->name : "", bus->name); +#endif + + /* + * Select the device if it wasn't already selected. + */ + retval = 0; + if (bus->dev != dev) { + retval = bus->select(bus, dev); + bus->dev = dev; + } + + return retval; +} + +/** + * ssi_register_device - register a SSI device with a SSI bus + * @bus: bus + * @dev: SSI device + */ +int ssi_register_device(struct ssi_bus *bus, struct ssi_dev *dev) +{ + int retval; + + dev->bus = bus; + bus->devices++; + retval = dev->init(dev); + if (retval != 0) { + dev->bus = NULL; + bus->devices--; + } else { +#ifdef DEBUG + printk("SSI: registered new device %s on bus %s\n", dev->name, bus->name); +#endif + } + return retval; +} + +/** + * ssi_unregister_device - unregister a SSI device from a SSI bus + * @dev: SSI device + */ +int ssi_unregister_device(struct ssi_dev *dev) +{ + struct ssi_bus *bus = dev->bus; + + if (bus->dev == dev) + bus->dev = NULL; + + dev->bus = NULL; + bus->devices--; +#ifdef DEBUG + printk("SSI: unregistered device %s on bus %s\n", dev->name, bus->name); +#endif + return 0; +} + +/** + * ssi_register_bus - register a SSI bus driver + * @bus: bus + */ +int ssi_register_bus(struct ssi_bus *bus) +{ + int retval; + + retval = bus->init(bus); + if (retval == 0) { + bus->devices = 0; +#ifdef DEBUG + printk("SSI: registered new bus %s\n", bus->name); +#endif + } + + return retval; +} + +/** + * ssi_unregister_bus - unregister a SSI bus driver + * @bus: bus + */ +int ssi_unregister_bus(struct ssi_bus *bus) +{ + int retval = -EBUSY; + if (bus->devices == 0) { + retval = 0; + } + return retval; +} + +static int __init ssi_init(void) +{ + return 0; +} + +static void __exit ssi_exit(void) +{ +} + +module_init(ssi_init); +module_exit(ssi_exit); diff -urN orig/drivers/ssi/ssi_dev.h linux/drivers/ssi/ssi_dev.h --- orig/drivers/ssi/ssi_dev.h Thu Jan 1 01:00:00 1970 +++ linux/drivers/ssi/ssi_dev.h Wed Dec 20 17:11:33 2000 @@ -0,0 +1,21 @@ +struct ssi_bus; + +#define SSI_SPI 1 +#define SSI_MICROWIRE 2 +#define SSI_TISSF 3 +#define SSI_USAR 4 + +struct ssi_dev { + char *name; + u_int id; + u_int clkfreq; + u_char cfglen; + u_char framelen; + u_char clkpol; + u_char proto; + void (*rcv)(struct ssi_dev *, u_int); + int (*init)(struct ssi_dev *); + struct ssi_bus *bus; +}; + + diff -urN orig/drivers/usb/Config.in linux/drivers/usb/Config.in --- orig/drivers/usb/Config.in Mon Aug 5 13:31:30 2002 +++ linux/drivers/usb/Config.in Mon Aug 5 15:08:21 2002 @@ -4,7 +4,13 @@ mainmenu_option next_comment comment 'USB support' -dep_tristate 'Support for USB' CONFIG_USB $CONFIG_PCI +# ARM SA1111 chips have a non-PCI based "OHCI-compatible" USB host interface. +if [ "$CONFIG_PCI" = "y" -o "$CONFIG_SA1111" = "y" ]; then + tristate 'Support for USB' CONFIG_USB +else + define_bool CONFIG_USB n +fi + if [ "$CONFIG_USB" = "y" -o "$CONFIG_USB" = "m" ]; then bool ' USB verbose debug messages' CONFIG_USB_DEBUG @@ -28,6 +34,7 @@ define_bool CONFIG_USB_UHCI_ALT n fi dep_tristate ' OHCI (Compaq, iMacs, OPTi, SiS, ALi, ...) support' CONFIG_USB_OHCI $CONFIG_USB + dep_tristate ' SA1111 OHCI-compatible host interface support' CONFIG_USB_OHCI_SA1111 $CONFIG_USB comment 'USB Device Class drivers' dep_tristate ' USB Audio support' CONFIG_USB_AUDIO $CONFIG_USB $CONFIG_SOUND diff -urN orig/drivers/usb/Makefile linux/drivers/usb/Makefile --- orig/drivers/usb/Makefile Mon Aug 5 13:31:30 2002 +++ linux/drivers/usb/Makefile Mon Aug 5 15:11:10 2002 @@ -52,7 +52,8 @@ obj-$(CONFIG_USB_UHCI) += usb-uhci.o obj-$(CONFIG_USB_UHCI_ALT) += uhci.o -obj-$(CONFIG_USB_OHCI) += usb-ohci.o +obj-$(CONFIG_USB_OHCI) += usb-ohci.o usb-ohci-pci.o +obj-$(CONFIG_USB_OHCI_SA1111) += usb-ohci.o usb-ohci-sa1111.o ifneq ($(CONFIG_USB_EHCI_HCD),n) usbcore-objs += hcd.o diff -urN orig/drivers/usb/usb-ohci-pci.c linux/drivers/usb/usb-ohci-pci.c --- orig/drivers/usb/usb-ohci-pci.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/usb/usb-ohci-pci.c Mon Aug 5 15:51:02 2002 @@ -0,0 +1,433 @@ +#include +#include +#include +#include +#include +#include +#include /* for in_interrupt() */ +#undef DEBUG +#include + +#include "usb-ohci.h" + +#ifdef CONFIG_PMAC_PBOOK +#include +#include +#include +#ifndef CONFIG_PM +#define CONFIG_PM +#endif +#endif + + +/*-------------------------------------------------------------------------*/ + +/* Increment the module usage count, start the control thread and + * return success. */ + +static struct pci_driver ohci_pci_driver; +extern spinlock_t usb_ed_lock; + +static int __devinit +hc_found_ohci (struct pci_dev *dev, int irq, + void *mem_base, const struct pci_device_id *id) +{ + u8 latency, limit; + ohci_t * ohci; + int ret; + + printk(KERN_INFO __FILE__ ": usb-%s, %s\n", dev->slot_name, dev->name); + + /* bad pci latencies can contribute to overruns */ + pci_read_config_byte (dev, PCI_LATENCY_TIMER, &latency); + if (latency) { + pci_read_config_byte (dev, PCI_MAX_LAT, &limit); + if (limit && limit < latency) { + dbg ("PCI latency reduced to max %d", limit); + pci_write_config_byte (dev, PCI_LATENCY_TIMER, limit); + latency = limit; + } + } + + ret = hc_add_ohci(dev, irq, mem_base, id->driver_data, + &ohci, ohci_pci_driver.name, dev->slot_name); + + if (ret == 0) + ohci->pci_latency = latency; + + return ret; +} + +/*-------------------------------------------------------------------------*/ + +#ifdef CONFIG_PM + +/* controller died; cleanup debris, then restart */ +/* must not be called from interrupt context */ + +static void hc_restart (ohci_t *ohci) +{ + int temp; + int i; + + if (ohci->pci_latency) + pci_write_config_byte (ohci->ohci_dev, PCI_LATENCY_TIMER, ohci->pci_latency); + + ohci->disabled = 1; + ohci->sleeping = 0; + if (ohci->bus->root_hub) + usb_disconnect (&ohci->bus->root_hub); + + /* empty the interrupt branches */ + for (i = 0; i < NUM_INTS; i++) ohci->ohci_int_load[i] = 0; + for (i = 0; i < NUM_INTS; i++) ohci->hcca->int_table[i] = 0; + + /* no EDs to remove */ + ohci->ed_rm_list [0] = NULL; + ohci->ed_rm_list [1] = NULL; + + /* empty control and bulk lists */ + ohci->ed_isotail = NULL; + ohci->ed_controltail = NULL; + ohci->ed_bulktail = NULL; + + if ((temp = hc_reset (ohci)) < 0 || (temp = hc_start (ohci)) < 0) { + err ("can't restart usb-%s, %d", ohci->ohci_dev->slot_name, temp); + } else + dbg ("restart usb-%s completed", ohci->ohci_dev->slot_name); +} + +#endif /* CONFIG_PM */ + +/*-------------------------------------------------------------------------*/ + +/* configured so that an OHCI device is always provided */ +/* always called with process context; sleeping is OK */ + +static int __devinit +ohci_pci_probe (struct pci_dev *dev, const struct pci_device_id *id) +{ + unsigned long mem_resource, mem_len; + void *mem_base; + int status; + + if (pci_enable_device(dev) < 0) + return -ENODEV; + + if (!dev->irq) { + err("found OHCI device with no IRQ assigned. check BIOS settings!"); + pci_disable_device (dev); + return -ENODEV; + } + + /* we read its hardware registers as memory */ + mem_resource = pci_resource_start(dev, 0); + mem_len = pci_resource_len(dev, 0); + if (!request_mem_region (mem_resource, mem_len, ohci_pci_driver.name)) { + dbg ("controller already in use"); + pci_disable_device (dev); + return -EBUSY; + } + + mem_base = ioremap_nocache (mem_resource, mem_len); + if (!mem_base) { + err("Error mapping OHCI memory"); + release_mem_region (mem_resource, mem_len); + pci_disable_device (dev); + return -EFAULT; + } + + /* controller writes into our memory */ + pci_set_master (dev); + + status = hc_found_ohci (dev, dev->irq, mem_base, id); + if (status < 0) { + iounmap (mem_base); + release_mem_region (mem_resource, mem_len); + pci_disable_device (dev); + } + return status; +} + +/*-------------------------------------------------------------------------*/ + +/* may be called from interrupt context [interface spec] */ +/* may be called without controller present */ +/* may be called with controller, bus, and devices active */ + +static void __devexit +ohci_pci_remove (struct pci_dev *dev) +{ + ohci_t *ohci = (ohci_t *) pci_get_drvdata(dev); + + dbg ("remove %s controller usb-%s%s%s", + hcfs2string (ohci->hc_control & OHCI_CTRL_HCFS), + dev->slot_name, + ohci->disabled ? " (disabled)" : "", + in_interrupt () ? " in interrupt" : "" + ); + + hc_remove_ohci(ohci); + + release_mem_region (pci_resource_start (dev, 0), pci_resource_len (dev, 0)); + pci_disable_device (dev); +} + + +#ifdef CONFIG_PM + +/*-------------------------------------------------------------------------*/ + +static int +ohci_pci_suspend (struct pci_dev *dev, u32 state) +{ + ohci_t *ohci = (ohci_t *) pci_get_drvdata(dev); + unsigned long flags; + u16 cmd; + + if ((ohci->hc_control & OHCI_CTRL_HCFS) != OHCI_USB_OPER) { + dbg ("can't suspend usb-%s (state is %s)", dev->slot_name, + hcfs2string (ohci->hc_control & OHCI_CTRL_HCFS)); + return -EIO; + } + + /* act as if usb suspend can always be used */ + info ("USB suspend: usb-%s", dev->slot_name); + ohci->sleeping = 1; + + /* First stop processing */ + spin_lock_irqsave (&usb_ed_lock, flags); + ohci->hc_control &= ~(OHCI_CTRL_PLE|OHCI_CTRL_CLE|OHCI_CTRL_BLE|OHCI_CTRL_IE); + writel (ohci->hc_control, &ohci->regs->control); + writel (OHCI_INTR_SF, &ohci->regs->intrstatus); + (void) readl (&ohci->regs->intrstatus); + spin_unlock_irqrestore (&usb_ed_lock, flags); + + /* Wait a frame or two */ + mdelay(1); + if (!readl (&ohci->regs->intrstatus) & OHCI_INTR_SF) + mdelay (1); + +#ifdef CONFIG_PMAC_PBOOK + if (_machine == _MACH_Pmac) + disable_irq (ohci->irq); + /* else, 2.4 assumes shared irqs -- don't disable */ +#endif + /* Enable remote wakeup */ + writel (readl(&ohci->regs->intrenable) | OHCI_INTR_RD, &ohci->regs->intrenable); + + /* Suspend chip and let things settle down a bit */ + ohci->hc_control = OHCI_USB_SUSPEND; + writel (ohci->hc_control, &ohci->regs->control); + (void) readl (&ohci->regs->control); + mdelay (500); /* No schedule here ! */ + switch (readl (&ohci->regs->control) & OHCI_CTRL_HCFS) { + case OHCI_USB_RESET: + dbg("Bus in reset phase ???"); + break; + case OHCI_USB_RESUME: + dbg("Bus in resume phase ???"); + break; + case OHCI_USB_OPER: + dbg("Bus in operational phase ???"); + break; + case OHCI_USB_SUSPEND: + dbg("Bus suspended"); + break; + } + /* In some rare situations, Apple's OHCI have happily trashed + * memory during sleep. We disable it's bus master bit during + * suspend + */ + pci_read_config_word (dev, PCI_COMMAND, &cmd); + cmd &= ~PCI_COMMAND_MASTER; + pci_write_config_word (dev, PCI_COMMAND, cmd); +#ifdef CONFIG_PMAC_PBOOK + { + struct device_node *of_node; + + /* Disable USB PAD & cell clock */ + of_node = pci_device_to_OF_node (ohci->ohci_dev); + if (of_node) + pmac_call_feature(PMAC_FTR_USB_ENABLE, of_node, 0, 0); + } +#endif + return 0; +} + +/*-------------------------------------------------------------------------*/ + +static int +ohci_pci_resume (struct pci_dev *dev) +{ + ohci_t *ohci = (ohci_t *) pci_get_drvdata(dev); + int temp; + unsigned long flags; + + /* guard against multiple resumes */ + atomic_inc (&ohci->resume_count); + if (atomic_read (&ohci->resume_count) != 1) { + err ("concurrent PCI resumes for usb-%s", dev->slot_name); + atomic_dec (&ohci->resume_count); + return 0; + } + +#ifdef CONFIG_PMAC_PBOOK + { + struct device_node *of_node; + + /* Re-enable USB PAD & cell clock */ + of_node = pci_device_to_OF_node (ohci->ohci_dev); + if (of_node) + pmac_call_feature(PMAC_FTR_USB_ENABLE, of_node, 0, 1); + } +#endif + + /* did we suspend, or were we powered off? */ + ohci->hc_control = readl (&ohci->regs->control); + temp = ohci->hc_control & OHCI_CTRL_HCFS; + +#ifdef DEBUG + /* the registers may look crazy here */ + ohci_dump_status (ohci); +#endif + + /* Re-enable bus mastering */ + pci_set_master(ohci->ohci_dev); + + switch (temp) { + + case OHCI_USB_RESET: // lost power + info ("USB restart: usb-%s", dev->slot_name); + hc_restart (ohci); + break; + + case OHCI_USB_SUSPEND: // host wakeup + case OHCI_USB_RESUME: // remote wakeup + info ("USB continue: usb-%s from %s wakeup", dev->slot_name, + (temp == OHCI_USB_SUSPEND) + ? "host" : "remote"); + ohci->hc_control = OHCI_USB_RESUME; + writel (ohci->hc_control, &ohci->regs->control); + (void) readl (&ohci->regs->control); + mdelay (20); /* no schedule here ! */ + /* Some controllers (lucent) need a longer delay here */ + mdelay (15); + temp = readl (&ohci->regs->control); + temp = ohci->hc_control & OHCI_CTRL_HCFS; + if (temp != OHCI_USB_RESUME) { + err ("controller usb-%s won't resume", dev->slot_name); + ohci->disabled = 1; + return -EIO; + } + + /* Some chips likes being resumed first */ + writel (OHCI_USB_OPER, &ohci->regs->control); + (void) readl (&ohci->regs->control); + mdelay (3); + + /* Then re-enable operations */ + spin_lock_irqsave (&usb_ed_lock, flags); + ohci->disabled = 0; + ohci->sleeping = 0; + ohci->hc_control = OHCI_CONTROL_INIT | OHCI_USB_OPER; + if (!ohci->ed_rm_list[0] && !ohci->ed_rm_list[1]) { + if (ohci->ed_controltail) + ohci->hc_control |= OHCI_CTRL_CLE; + if (ohci->ed_bulktail) + ohci->hc_control |= OHCI_CTRL_BLE; + } + writel (ohci->hc_control, &ohci->regs->control); + writel (OHCI_INTR_SF, &ohci->regs->intrstatus); + writel (OHCI_INTR_SF, &ohci->regs->intrenable); + /* Check for a pending done list */ + writel (OHCI_INTR_WDH, &ohci->regs->intrdisable); + (void) readl (&ohci->regs->intrdisable); + spin_unlock_irqrestore (&usb_ed_lock, flags); +#ifdef CONFIG_PMAC_PBOOK + if (_machine == _MACH_Pmac) + enable_irq (ohci->irq); +#endif + if (ohci->hcca->done_head) + dl_done_list (ohci, dl_reverse_done_list (ohci)); + writel (OHCI_INTR_WDH, &ohci->regs->intrenable); + writel (OHCI_BLF, &ohci->regs->cmdstatus); /* start bulk list */ + writel (OHCI_CLF, &ohci->regs->cmdstatus); /* start Control list */ + break; + + default: + warn ("odd PCI resume for usb-%s", dev->slot_name); + } + + /* controller is operational, extra resumes are harmless */ + atomic_dec (&ohci->resume_count); + + return 0; +} + +#endif /* CONFIG_PM */ + + +/*-------------------------------------------------------------------------*/ + +static const struct pci_device_id __devinitdata ohci_pci_ids [] = { { + + /* + * AMD-756 [Viper] USB has a serious erratum when used with + * lowspeed devices like mice. + */ + vendor: 0x1022, + device: 0x740c, + subvendor: PCI_ANY_ID, + subdevice: PCI_ANY_ID, + + driver_data: OHCI_QUIRK_AMD756, + +} , { + + /* handle any USB OHCI controller */ + class: ((PCI_CLASS_SERIAL_USB << 8) | 0x10), + class_mask: ~0, + + /* no matter who makes it */ + vendor: PCI_ANY_ID, + device: PCI_ANY_ID, + subvendor: PCI_ANY_ID, + subdevice: PCI_ANY_ID, + + }, { /* end: all zeroes */ } +}; + +MODULE_DEVICE_TABLE (pci, ohci_pci_ids); + +static struct pci_driver ohci_pci_driver = { + name: "usb-ohci", + id_table: &ohci_pci_ids [0], + + probe: ohci_pci_probe, + remove: __devexit_p(ohci_pci_remove), + +#ifdef CONFIG_PM + suspend: ohci_pci_suspend, + resume: ohci_pci_resume, +#endif /* PM */ +}; + + +/*-------------------------------------------------------------------------*/ + +static int __init ohci_hcd_init (void) +{ + return pci_module_init (&ohci_pci_driver); +} + +/*-------------------------------------------------------------------------*/ + +static void __exit ohci_hcd_cleanup (void) +{ + pci_unregister_driver (&ohci_pci_driver); +} + +module_init (ohci_hcd_init); +module_exit (ohci_hcd_cleanup); + diff -urN orig/drivers/usb/usb-ohci-sa1111.c linux/drivers/usb/usb-ohci-sa1111.c --- orig/drivers/usb/usb-ohci-sa1111.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/usb/usb-ohci-sa1111.c Sat Feb 1 15:40:29 2003 @@ -0,0 +1,118 @@ +/* + * linux/drivers/usb/usb-ohci-sa1111.c + * + * The outline of this code was taken from Brad Parkers + * original OHCI driver modifications, and reworked into a cleaner form + * by Russell King . + */ +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include + +#include "usb-ohci.h" + +int __devinit +hc_add_ohci(struct pci_dev *dev, int irq, void *membase, unsigned long flags, + ohci_t **ohci, const char *name, const char *slot_name); +extern void hc_remove_ohci(ohci_t *ohci); + +static ohci_t *sa1111_ohci; + +static void __init sa1111_ohci_configure(void) +{ + unsigned int usb_rst = 0; + + if (machine_is_xp860() || + machine_has_neponset() || + machine_is_accelent_sa() || + machine_is_pfs168() || + machine_is_badge4()) + usb_rst = USB_RESET_PWRSENSELOW | USB_RESET_PWRCTRLLOW; + + /* + * Configure the power sense and control lines. Place the USB + * host controller in reset. + */ + USB_RESET = usb_rst | USB_RESET_FORCEIFRESET | USB_RESET_FORCEHCRESET; + + /* + * Now, carefully enable the USB clock, and take + * the USB host controller out of reset. + */ + SKPCR |= SKPCR_UCLKEN; + udelay(11); + USB_RESET = usb_rst; +} + +static int __init sa1111_ohci_init(void) +{ + int ret; + + /* + * Request memory resources. + */ +// if (!request_mem_region(_USB_OHCI_OP_BASE, _USB_EXTENT, "usb-ohci")) +// return -EBUSY; + + sa1111_ohci_configure(); + + /* + * Initialise the generic OHCI driver. + */ + ret = hc_add_ohci(SA1111_FAKE_PCIDEV, NIRQHCIM, + (void *)&USB_OHCI_OP_BASE, 0, &sa1111_ohci, + "usb-ohci", "sa1111"); + +// if (ret) +// release_mem_region(_USB_OHCI_OP_BASE, _USB_EXTENT); + +#ifdef CONFIG_SA1100_BADGE4 + if (machine_is_badge4() && (!ret)) { + /* found the controller, so now power the bus */ + badge4_set_5V(BADGE4_5V_USB, 1); + } +#endif + + return ret; +} + +static void __exit sa1111_ohci_exit(void) +{ + hc_remove_ohci(sa1111_ohci); + + /* + * Put the USB host controller into reset. + */ + USB_RESET |= USB_RESET_FORCEIFRESET | USB_RESET_FORCEHCRESET; + + /* + * Stop the USB clock. + */ + SKPCR &= ~SKPCR_UCLKEN; + + /* + * Release memory resources. + */ +// release_mem_region(_USB_OHCI_OP_BASE, _USB_EXTENT); + +#ifdef CONFIG_SA1100_BADGE4 + if (machine_is_badge4()) { + badge4_set_5V(BADGE4_5V_USB, 0); + } +#endif +} + +module_init(sa1111_ohci_init); +module_exit(sa1111_ohci_exit); diff -urN orig/drivers/usb/usb-ohci.c linux/drivers/usb/usb-ohci.c --- orig/drivers/usb/usb-ohci.c Mon Aug 5 13:31:37 2002 +++ linux/drivers/usb/usb-ohci.c Mon Aug 5 15:51:05 2002 @@ -79,16 +79,6 @@ #include "usb-ohci.h" -#ifdef CONFIG_PMAC_PBOOK -#include -#include -#include -#ifndef CONFIG_PM -#define CONFIG_PM -#endif -#endif - - /* * Version Information */ @@ -96,14 +86,10 @@ #define DRIVER_AUTHOR "Roman Weissgaerber , David Brownell" #define DRIVER_DESC "USB OHCI Host Controller Driver" -/* For initializing controller (mask in an HCFS mode too) */ -#define OHCI_CONTROL_INIT \ - (OHCI_CTRL_CBSR & 0x3) | OHCI_CTRL_IE | OHCI_CTRL_PLE - #define OHCI_UNLINK_TIMEOUT (HZ / 10) static LIST_HEAD (ohci_hcd_list); -static spinlock_t usb_ed_lock = SPIN_LOCK_UNLOCKED; +spinlock_t usb_ed_lock = SPIN_LOCK_UNLOCKED; /*-------------------------------------------------------------------------*/ @@ -439,7 +425,7 @@ static void ohci_dump (ohci_t *controller, int verbose) { - dbg ("OHCI controller usb-%s state", controller->ohci_dev->slot_name); + dbg ("OHCI controller usb-%s state", controller->slot_name); // dumps some of the state we know about ohci_dump_status (controller); @@ -853,7 +839,7 @@ if (ed->state == ED_OPER) { /* driver on that interface didn't unlink an urb */ dbg ("driver usb-%s dev %d ed 0x%x unfreed URB", - ohci->ohci_dev->slot_name, usb_dev->devnum, i); + ohci->slot_name, usb_dev->devnum, i); ep_unlink (ohci, ed); } ep_rm_ed (usb_dev, ed); @@ -898,7 +884,7 @@ } else { /* likely some interface's driver has a refcount bug */ err ("bus %s devnum %d deletion in interrupt", - ohci->ohci_dev->slot_name, usb_dev->devnum); + ohci->slot_name, usb_dev->devnum); BUG (); } } @@ -1304,7 +1290,8 @@ err("internal OHCI error: TD index > length"); return; } - +if (data & (1 << 20)) panic("td_fill: A20 = 1: %08x\n", data); + /* use this td as the next dummy */ td_pt = urb_priv->td [index]; td_pt->hwNextTD = 0; @@ -1815,7 +1802,7 @@ num_ports = roothub_a (ohci) & RH_A_NDP; if (num_ports > MAX_ROOT_PORTS) { err ("bogus NDP=%d for OHCI usb-%s", num_ports, - ohci->ohci_dev->slot_name); + ohci->slot_name); err ("rereads as NDP=%d", readl (&ohci->regs->roothub.a) & RH_A_NDP); /* retry later; "should not happen" */ @@ -2160,7 +2147,7 @@ writel (OHCI_INTR_MIE, &ohci->regs->intrdisable); dbg("USB HC reset_hc usb-%s: ctrl = 0x%x ;", - ohci->ohci_dev->slot_name, + ohci->slot_name, readl (&ohci->regs->control)); /* Reset USB (needed by some controllers) */ @@ -2292,14 +2279,14 @@ ints = OHCI_INTR_WDH; } else if ((ints = (readl (®s->intrstatus) & readl (®s->intrenable))) == 0) { return; - } + } // dbg("Interrupt: %x frame: %x", ints, le16_to_cpu (ohci->hcca->frame_no)); if (ints & OHCI_INTR_UE) { ohci->disabled++; err ("OHCI Unrecoverable Error, controller usb-%s disabled", - ohci->ohci_dev->slot_name); + ohci->slot_name); // e.g. due to PCI Master/Target Abort #ifdef DEBUG @@ -2376,7 +2363,9 @@ ohci->regs = mem_base; ohci->ohci_dev = dev; +#ifdef CONFIG_PCI pci_set_drvdata(dev, ohci); +#endif INIT_LIST_HEAD (&ohci->ohci_hcd_list); list_add (&ohci->ohci_hcd_list, &ohci_hcd_list); @@ -2403,7 +2392,7 @@ static void hc_release_ohci (ohci_t * ohci) { - dbg ("USB HC release ohci usb-%s", ohci->ohci_dev->slot_name); + dbg ("USB HC release ohci usb-%s", ohci->slot_name); /* disconnect all devices */ if (ohci->bus->root_hub) @@ -2416,7 +2405,9 @@ free_irq (ohci->irq, ohci); ohci->irq = -1; } - pci_set_drvdata(ohci->ohci_dev, NULL); +#ifdef CONFIG_PCI + pci_set_drvdata(ohci->ohci_dev, 0); +#endif if (ohci->bus) { if (ohci->bus->busnum) usb_deregister_bus (ohci->bus); @@ -2438,18 +2429,15 @@ /*-------------------------------------------------------------------------*/ -/* Increment the module usage count, start the control thread and - * return success. */ - -static struct pci_driver ohci_pci_driver; - -static int __devinit -hc_found_ohci (struct pci_dev *dev, int irq, - void *mem_base, const struct pci_device_id *id) +/* + * Host bus independent add one OHCI host controller. + */ +int __devinit +hc_add_ohci(struct pci_dev *dev, int irq, void *mem_base, unsigned long flags, + ohci_t **ohci, const char *name, const char *slot_name) { - ohci_t * ohci; - u8 latency, limit; char buf[8], *bufp = buf; + ohci_t * ohci; int ret; #ifndef __sparc__ @@ -2459,34 +2447,20 @@ #endif printk(KERN_INFO __FILE__ ": USB OHCI at membase 0x%lx, IRQ %s\n", (unsigned long) mem_base, bufp); - printk(KERN_INFO __FILE__ ": usb-%s, %s\n", dev->slot_name, dev->name); - + ohci = hc_alloc_ohci (dev, mem_base); if (!ohci) { return -ENOMEM; } + ohci->slot_name = slot_name; if ((ret = ohci_mem_init (ohci)) < 0) { hc_release_ohci (ohci); return ret; } - ohci->flags = id->driver_data; + ohci->flags = flags; if (ohci->flags & OHCI_QUIRK_AMD756) printk (KERN_INFO __FILE__ ": AMD756 erratum 4 workaround\n"); - /* bad pci latencies can contribute to overruns */ - pci_read_config_byte (dev, PCI_LATENCY_TIMER, &latency); - if (latency) { - pci_read_config_byte (dev, PCI_MAX_LAT, &limit); - if (limit && limit < latency) { - dbg ("PCI latency reduced to max %d", limit); - pci_write_config_byte (dev, PCI_LATENCY_TIMER, limit); - ohci->pci_latency = limit; - } else { - /* it might already have been reduced */ - ohci->pci_latency = latency; - } - } - if (hc_reset (ohci) < 0) { hc_release_ohci (ohci); return -ENODEV; @@ -2498,8 +2472,7 @@ usb_register_bus (ohci->bus); - if (request_irq (irq, hc_interrupt, SA_SHIRQ, - ohci_pci_driver.name, ohci) != 0) { + if (request_irq (irq, hc_interrupt, SA_SHIRQ, name, ohci) != 0) { err ("request interrupt %s failed", bufp); hc_release_ohci (ohci); return -EBUSY; @@ -2507,7 +2480,7 @@ ohci->irq = irq; if (hc_start (ohci) < 0) { - err ("can't start usb-%s", dev->slot_name); + err ("can't start usb-%s", ohci->slot_name); hc_release_ohci (ohci); return -EBUSY; } @@ -2518,114 +2491,11 @@ return 0; } -/*-------------------------------------------------------------------------*/ - -#ifdef CONFIG_PM - -/* controller died; cleanup debris, then restart */ -/* must not be called from interrupt context */ - -static void hc_restart (ohci_t *ohci) -{ - int temp; - int i; - - if (ohci->pci_latency) - pci_write_config_byte (ohci->ohci_dev, PCI_LATENCY_TIMER, ohci->pci_latency); - - ohci->disabled = 1; - ohci->sleeping = 0; - if (ohci->bus->root_hub) - usb_disconnect (&ohci->bus->root_hub); - - /* empty the interrupt branches */ - for (i = 0; i < NUM_INTS; i++) ohci->ohci_int_load[i] = 0; - for (i = 0; i < NUM_INTS; i++) ohci->hcca->int_table[i] = 0; - - /* no EDs to remove */ - ohci->ed_rm_list [0] = NULL; - ohci->ed_rm_list [1] = NULL; - - /* empty control and bulk lists */ - ohci->ed_isotail = NULL; - ohci->ed_controltail = NULL; - ohci->ed_bulktail = NULL; - - if ((temp = hc_reset (ohci)) < 0 || (temp = hc_start (ohci)) < 0) { - err ("can't restart usb-%s, %d", ohci->ohci_dev->slot_name, temp); - } else - dbg ("restart usb-%s completed", ohci->ohci_dev->slot_name); -} - -#endif /* CONFIG_PM */ - -/*-------------------------------------------------------------------------*/ - -/* configured so that an OHCI device is always provided */ -/* always called with process context; sleeping is OK */ - -static int __devinit -ohci_pci_probe (struct pci_dev *dev, const struct pci_device_id *id) +/* + * Host bus independent remove one OHCI host controller. + */ +void __devexit hc_remove_ohci(ohci_t *ohci) { - unsigned long mem_resource, mem_len; - void *mem_base; - int status; - - if (pci_enable_device(dev) < 0) - return -ENODEV; - - if (!dev->irq) { - err("found OHCI device with no IRQ assigned. check BIOS settings!"); - pci_disable_device (dev); - return -ENODEV; - } - - /* we read its hardware registers as memory */ - mem_resource = pci_resource_start(dev, 0); - mem_len = pci_resource_len(dev, 0); - if (!request_mem_region (mem_resource, mem_len, ohci_pci_driver.name)) { - dbg ("controller already in use"); - pci_disable_device (dev); - return -EBUSY; - } - - mem_base = ioremap_nocache (mem_resource, mem_len); - if (!mem_base) { - err("Error mapping OHCI memory"); - release_mem_region (mem_resource, mem_len); - pci_disable_device (dev); - return -EFAULT; - } - - /* controller writes into our memory */ - pci_set_master (dev); - - status = hc_found_ohci (dev, dev->irq, mem_base, id); - if (status < 0) { - iounmap (mem_base); - release_mem_region (mem_resource, mem_len); - pci_disable_device (dev); - } - return status; -} - -/*-------------------------------------------------------------------------*/ - -/* may be called from interrupt context [interface spec] */ -/* may be called without controller present */ -/* may be called with controller, bus, and devices active */ - -static void __devexit -ohci_pci_remove (struct pci_dev *dev) -{ - ohci_t *ohci = pci_get_drvdata(dev); - - dbg ("remove %s controller usb-%s%s%s", - hcfs2string (ohci->hc_control & OHCI_CTRL_HCFS), - dev->slot_name, - ohci->disabled ? " (disabled)" : "", - in_interrupt () ? " in interrupt" : "" - ); #ifdef DEBUG ohci_dump (ohci, 1); #endif @@ -2642,269 +2512,7 @@ &ohci->regs->control); hc_release_ohci (ohci); - - release_mem_region (pci_resource_start (dev, 0), pci_resource_len (dev, 0)); - pci_disable_device (dev); -} - - -#ifdef CONFIG_PM - -/*-------------------------------------------------------------------------*/ - -static int -ohci_pci_suspend (struct pci_dev *dev, u32 state) -{ - ohci_t *ohci = pci_get_drvdata(dev); - unsigned long flags; - u16 cmd; - - if ((ohci->hc_control & OHCI_CTRL_HCFS) != OHCI_USB_OPER) { - dbg ("can't suspend usb-%s (state is %s)", dev->slot_name, - hcfs2string (ohci->hc_control & OHCI_CTRL_HCFS)); - return -EIO; - } - - /* act as if usb suspend can always be used */ - info ("USB suspend: usb-%s", dev->slot_name); - ohci->sleeping = 1; - - /* First stop processing */ - spin_lock_irqsave (&usb_ed_lock, flags); - ohci->hc_control &= ~(OHCI_CTRL_PLE|OHCI_CTRL_CLE|OHCI_CTRL_BLE|OHCI_CTRL_IE); - writel (ohci->hc_control, &ohci->regs->control); - writel (OHCI_INTR_SF, &ohci->regs->intrstatus); - (void) readl (&ohci->regs->intrstatus); - spin_unlock_irqrestore (&usb_ed_lock, flags); - - /* Wait a frame or two */ - mdelay(1); - if (!readl (&ohci->regs->intrstatus) & OHCI_INTR_SF) - mdelay (1); - -#ifdef CONFIG_PMAC_PBOOK - if (_machine == _MACH_Pmac) - disable_irq (ohci->irq); - /* else, 2.4 assumes shared irqs -- don't disable */ -#endif - /* Enable remote wakeup */ - writel (readl(&ohci->regs->intrenable) | OHCI_INTR_RD, &ohci->regs->intrenable); - - /* Suspend chip and let things settle down a bit */ - ohci->hc_control = OHCI_USB_SUSPEND; - writel (ohci->hc_control, &ohci->regs->control); - (void) readl (&ohci->regs->control); - mdelay (500); /* No schedule here ! */ - switch (readl (&ohci->regs->control) & OHCI_CTRL_HCFS) { - case OHCI_USB_RESET: - dbg("Bus in reset phase ???"); - break; - case OHCI_USB_RESUME: - dbg("Bus in resume phase ???"); - break; - case OHCI_USB_OPER: - dbg("Bus in operational phase ???"); - break; - case OHCI_USB_SUSPEND: - dbg("Bus suspended"); - break; - } - /* In some rare situations, Apple's OHCI have happily trashed - * memory during sleep. We disable it's bus master bit during - * suspend - */ - pci_read_config_word (dev, PCI_COMMAND, &cmd); - cmd &= ~PCI_COMMAND_MASTER; - pci_write_config_word (dev, PCI_COMMAND, cmd); -#ifdef CONFIG_PMAC_PBOOK - { - struct device_node *of_node; - - /* Disable USB PAD & cell clock */ - of_node = pci_device_to_OF_node (ohci->ohci_dev); - if (of_node) - pmac_call_feature(PMAC_FTR_USB_ENABLE, of_node, 0, 0); - } -#endif - return 0; } - -/*-------------------------------------------------------------------------*/ - -static int -ohci_pci_resume (struct pci_dev *dev) -{ - ohci_t *ohci = pci_get_drvdata(dev); - int temp; - unsigned long flags; - - /* guard against multiple resumes */ - atomic_inc (&ohci->resume_count); - if (atomic_read (&ohci->resume_count) != 1) { - err ("concurrent PCI resumes for usb-%s", dev->slot_name); - atomic_dec (&ohci->resume_count); - return 0; - } - -#ifdef CONFIG_PMAC_PBOOK - { - struct device_node *of_node; - - /* Re-enable USB PAD & cell clock */ - of_node = pci_device_to_OF_node (ohci->ohci_dev); - if (of_node) - pmac_call_feature(PMAC_FTR_USB_ENABLE, of_node, 0, 1); - } -#endif - - /* did we suspend, or were we powered off? */ - ohci->hc_control = readl (&ohci->regs->control); - temp = ohci->hc_control & OHCI_CTRL_HCFS; - -#ifdef DEBUG - /* the registers may look crazy here */ - ohci_dump_status (ohci); -#endif - - /* Re-enable bus mastering */ - pci_set_master(ohci->ohci_dev); - - switch (temp) { - - case OHCI_USB_RESET: // lost power - info ("USB restart: usb-%s", dev->slot_name); - hc_restart (ohci); - break; - - case OHCI_USB_SUSPEND: // host wakeup - case OHCI_USB_RESUME: // remote wakeup - info ("USB continue: usb-%s from %s wakeup", dev->slot_name, - (temp == OHCI_USB_SUSPEND) - ? "host" : "remote"); - ohci->hc_control = OHCI_USB_RESUME; - writel (ohci->hc_control, &ohci->regs->control); - (void) readl (&ohci->regs->control); - mdelay (20); /* no schedule here ! */ - /* Some controllers (lucent) need a longer delay here */ - mdelay (15); - temp = readl (&ohci->regs->control); - temp = ohci->hc_control & OHCI_CTRL_HCFS; - if (temp != OHCI_USB_RESUME) { - err ("controller usb-%s won't resume", dev->slot_name); - ohci->disabled = 1; - return -EIO; - } - - /* Some chips likes being resumed first */ - writel (OHCI_USB_OPER, &ohci->regs->control); - (void) readl (&ohci->regs->control); - mdelay (3); - - /* Then re-enable operations */ - spin_lock_irqsave (&usb_ed_lock, flags); - ohci->disabled = 0; - ohci->sleeping = 0; - ohci->hc_control = OHCI_CONTROL_INIT | OHCI_USB_OPER; - if (!ohci->ed_rm_list[0] && !ohci->ed_rm_list[1]) { - if (ohci->ed_controltail) - ohci->hc_control |= OHCI_CTRL_CLE; - if (ohci->ed_bulktail) - ohci->hc_control |= OHCI_CTRL_BLE; - } - writel (ohci->hc_control, &ohci->regs->control); - writel (OHCI_INTR_SF, &ohci->regs->intrstatus); - writel (OHCI_INTR_SF, &ohci->regs->intrenable); - /* Check for a pending done list */ - writel (OHCI_INTR_WDH, &ohci->regs->intrdisable); - (void) readl (&ohci->regs->intrdisable); - spin_unlock_irqrestore (&usb_ed_lock, flags); -#ifdef CONFIG_PMAC_PBOOK - if (_machine == _MACH_Pmac) - enable_irq (ohci->irq); -#endif - if (ohci->hcca->done_head) - dl_done_list (ohci, dl_reverse_done_list (ohci)); - writel (OHCI_INTR_WDH, &ohci->regs->intrenable); - writel (OHCI_BLF, &ohci->regs->cmdstatus); /* start bulk list */ - writel (OHCI_CLF, &ohci->regs->cmdstatus); /* start Control list */ - break; - - default: - warn ("odd PCI resume for usb-%s", dev->slot_name); - } - - /* controller is operational, extra resumes are harmless */ - atomic_dec (&ohci->resume_count); - - return 0; -} - -#endif /* CONFIG_PM */ - - -/*-------------------------------------------------------------------------*/ - -static const struct pci_device_id __devinitdata ohci_pci_ids [] = { { - - /* - * AMD-756 [Viper] USB has a serious erratum when used with - * lowspeed devices like mice. - */ - vendor: 0x1022, - device: 0x740c, - subvendor: PCI_ANY_ID, - subdevice: PCI_ANY_ID, - - driver_data: OHCI_QUIRK_AMD756, - -} , { - - /* handle any USB OHCI controller */ - class: ((PCI_CLASS_SERIAL_USB << 8) | 0x10), - class_mask: ~0, - - /* no matter who makes it */ - vendor: PCI_ANY_ID, - device: PCI_ANY_ID, - subvendor: PCI_ANY_ID, - subdevice: PCI_ANY_ID, - - }, { /* end: all zeroes */ } -}; - -MODULE_DEVICE_TABLE (pci, ohci_pci_ids); - -static struct pci_driver ohci_pci_driver = { - name: "usb-ohci", - id_table: &ohci_pci_ids [0], - - probe: ohci_pci_probe, - remove: __devexit_p(ohci_pci_remove), - -#ifdef CONFIG_PM - suspend: ohci_pci_suspend, - resume: ohci_pci_resume, -#endif /* PM */ -}; - - -/*-------------------------------------------------------------------------*/ - -static int __init ohci_hcd_init (void) -{ - return pci_module_init (&ohci_pci_driver); -} - -/*-------------------------------------------------------------------------*/ - -static void __exit ohci_hcd_cleanup (void) -{ - pci_unregister_driver (&ohci_pci_driver); -} - -module_init (ohci_hcd_init); -module_exit (ohci_hcd_cleanup); - MODULE_AUTHOR( DRIVER_AUTHOR ); MODULE_DESCRIPTION( DRIVER_DESC ); diff -urN orig/drivers/usb/usb-ohci.h linux/drivers/usb/usb-ohci.h --- orig/drivers/usb/usb-ohci.h Mon Aug 5 13:31:37 2002 +++ linux/drivers/usb/usb-ohci.h Mon Aug 5 13:45:41 2002 @@ -403,6 +403,7 @@ /* PCI device handle, settings, ... */ struct pci_dev *ohci_dev; + const char *slot_name; u8 pci_latency; struct pci_pool *td_cache; struct pci_pool *dev_cache; @@ -448,7 +449,7 @@ #endif #ifndef CONFIG_PCI -# error "usb-ohci currently requires PCI-based controllers" +//# error "usb-ohci currently requires PCI-based controllers" /* to support non-PCI OHCIs, you need custom bus/mem/... glue */ #endif @@ -641,3 +642,6 @@ pci_pool_free (hc->dev_cache, dev, dev->dma); } +/* For initializing controller (mask in an HCFS mode too) */ +#define OHCI_CONTROL_INIT \ + (OHCI_CTRL_CBSR & 0x3) | OHCI_CTRL_IE | OHCI_CTRL_PLE diff -urN orig/drivers/video/Config.in linux/drivers/video/Config.in --- orig/drivers/video/Config.in Mon Aug 5 13:31:38 2002 +++ linux/drivers/video/Config.in Fri Dec 20 17:03:01 2002 @@ -30,13 +30,27 @@ tristate ' Permedia3 support (EXPERIMENTAL)' CONFIG_FB_PM3 fi fi - if [ "$CONFIG_ARCH_ACORN" = "y" ]; then - bool ' Acorn VIDC support' CONFIG_FB_ACORN - fi - dep_tristate ' Cyber2000 support' CONFIG_FB_CYBER2000 $CONFIG_PCI - if [ "$CONFIG_ARCH_SA1100" = "y" ]; then - bool ' SA-1100 LCD support' CONFIG_FB_SA1100 + if [ "$CONFIG_ARM" = "y" ]; then + if [ "$CONFIG_ARCH_ACORN" -o "$CONFIG_ARCH_RISCSTATION" ]; then + bool ' Acorn VIDC support' CONFIG_FB_ACORN + else + define_bool CONFIG_FB_ACORN n + fi + dep_bool ' Anakin LCD support' CONFIG_FB_ANAKIN $CONFIG_ARCH_ANAKIN + dep_bool ' CLPS711X LCD support' CONFIG_FB_CLPS711X $CONFIG_ARCH_CLPS711X + dep_bool ' SA-1100 LCD support' CONFIG_FB_SA1100 $CONFIG_ARCH_SA1100 + if [ "$CONFIG_FB_SA1100" = "y" -a "$CONFIG_SA1100_CERF" = "y" ]; then + choice 'CerfBoard LCD Display Size' \ + "3.8_Color CONFIG_CERF_LCD_38_A \ + 3.8_Mono CONFIG_CERF_LCD_38_B \ + 5.7 CONFIG_CERF_LCD_57_A \ + 7.2 CONFIG_CERF_LCD_72_A" 5.7 + fi + if [ "$CONFIG_FB_SA1100" = "y" -a "$CONFIG_SA1100_CERF_CPLD" = "y" ]; then + bool 'Cerfboard Backlight (CerfPDA)' CONFIG_SA1100_CERF_LCD_BACKLIGHT + fi fi + dep_tristate ' CyberPro 2000/2010/5000 support' CONFIG_FB_CYBER2000 $CONFIG_PCI if [ "$CONFIG_APOLLO" = "y" ]; then define_bool CONFIG_FB_APOLLO y fi @@ -231,7 +245,7 @@ "$CONFIG_FB_MAC" = "y" -o "$CONFIG_FB_RETINAZ3" = "y" -o \ "$CONFIG_FB_VIRGE" = "y" -o "$CONFIG_FB_VIRTUAL" = "y" -o \ "$CONFIG_FB_BWTWO" = "y" -o "$CONFIG_FB_CLGEN" = "y" -o \ - "$CONFIG_FB_TX3912" = "y" ]; then + "$CONFIG_FB_TX3912" = "y" -o "$CONFIG_FB_CLPS711X" = "y" ]; then define_tristate CONFIG_FBCON_MFB y else if [ "$CONFIG_FB_ACORN" = "m" -o "$CONFIG_FB_AMIGA" = "m" -o \ @@ -239,19 +253,19 @@ "$CONFIG_FB_MAC" = "m" -o "$CONFIG_FB_RETINAZ3" = "m" -o \ "$CONFIG_FB_VIRGE" = "m" -o "$CONFIG_FB_VIRTUAL" = "m" -o \ "$CONFIG_FB_BWTWO" = "m" -o "$CONFIG_FB_CLGEN" = "m" -o \ - "$CONFIG_FB_TX3912" = "m" ]; then + "$CONFIG_FB_TX3912" = "m" -o "$CONFIG_FB_CLPS711X" = "m" ]; then define_tristate CONFIG_FBCON_MFB m fi fi if [ "$CONFIG_FB_ACORN" = "y" -o "$CONFIG_FB_MAC" = "y" -o \ "$CONFIG_FB_SA1100" = "y" -o "$CONFIG_FB_VIRTUAL" = "y" -o \ - "$CONFIG_FB_TX3912" = "y" ]; then + "$CONFIG_FB_TX3912" = "y" -o "$CONFIG_FB_CLPS711X" = "y" ]; then define_tristate CONFIG_FBCON_CFB2 y define_tristate CONFIG_FBCON_CFB4 y else if [ "$CONFIG_FB_ACORN" = "m" -o "$CONFIG_FB_MAC" = "m" -o \ "$CONFIG_FB_SA1100" = "m" -o "$CONFIG_FB_VIRTUAL" = "m" -o \ - "$CONFIG_FB_TX3912" = "m" ]; then + "$CONFIG_FB_TX3912" = "m" -o "$CONFIG_FB_CLPS711X" = "m" ]; then define_tristate CONFIG_FBCON_CFB2 m define_tristate CONFIG_FBCON_CFB4 m fi @@ -313,7 +327,7 @@ "$CONFIG_FB_CYBER2000" = "y" -o "$CONFIG_FB_3DFX" = "y" -o \ "$CONFIG_FB_SIS" = "y" -o "$CONFIG_FB_SA1100" = "y" -o \ "$CONFIG_FB_PVR2" = "y" -o "$CONFIG_FB_VOODOO1" = "y" -o \ - "$CONFIG_FB_NEOMAGIC" = "y" ]; then + "$CONFIG_FB_NEOMAGIC" = "y" -o "$CONFIG_FB_ANAKIN" = "y" ]; then define_tristate CONFIG_FBCON_CFB16 y else if [ "$CONFIG_FB_ATARI" = "m" -o "$CONFIG_FB_ATY" = "m" -o \ @@ -463,6 +477,9 @@ fi if [ "$CONFIG_AMIGA" = "y" ]; then define_bool CONFIG_FONT_PEARL_8x8 y + fi + if [ "$CONFIG_ARM" = "y" -a "$CONFIG_ARCH_RISCSTATION" = "y" ]; then + define_bool CONFIG_FONT_ACORN_8x8 y fi if [ "$CONFIG_ARM" = "y" -a "$CONFIG_ARCH_ACORN" = "y" ]; then define_bool CONFIG_FONT_ACORN_8x8 y diff -urN orig/drivers/video/Makefile linux/drivers/video/Makefile --- orig/drivers/video/Makefile Mon Aug 5 13:31:38 2002 +++ linux/drivers/video/Makefile Mon Aug 5 14:39:17 2002 @@ -56,6 +56,7 @@ obj-$(CONFIG_FB_PLATINUM) += platinumfb.o obj-$(CONFIG_FB_VALKYRIE) += valkyriefb.o obj-$(CONFIG_FB_CT65550) += chipsfb.o +obj-$(CONFIG_FB_CLPS711X) += clps711xfb.o obj-$(CONFIG_FB_CYBER) += cyberfb.o obj-$(CONFIG_FB_CYBER2000) += cyber2000fb.o obj-$(CONFIG_FB_SGIVW) += sgivwfb.o @@ -118,6 +119,7 @@ obj-$(CONFIG_FB_E1355) += epson1355fb.o fbgen.o obj-$(CONFIG_FB_PVR2) += pvr2fb.o obj-$(CONFIG_FB_VOODOO1) += sstfb.o +obj-$(CONFIG_FB_ANAKIN) += anakinfb.o # Generic Low Level Drivers diff -urN orig/drivers/video/acornfb.c linux/drivers/video/acornfb.c --- orig/drivers/video/acornfb.c Mon Aug 5 13:31:38 2002 +++ linux/drivers/video/acornfb.c Tue Mar 4 20:10:09 2003 @@ -752,11 +752,12 @@ } #endif #ifdef FBCON_HAS_CFB16 - if (bpp == 16 && regno < 16) { + if (bpp == 16) { int i; - current_par.cmap.cfb16[regno] = - regno | regno << 5 | regno << 10; + if (regno < 16) + current_par.cmap.cfb16[regno] = + regno | regno << 5 | regno << 10; pal.p = 0; vidc_writel(0x10000000); diff -urN orig/drivers/video/anakinfb.c linux/drivers/video/anakinfb.c --- orig/drivers/video/anakinfb.c Thu Jan 1 01:00:00 1970 +++ linux/drivers/video/anakinfb.c Thu Jul 12 23:37:43 2001 @@ -0,0 +1,221 @@ +/* + * linux/drivers/video/anakinfb.c + * + * Copyright (C) 2001 Aleph One Ltd. for Acunia N.V. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + * + * Changelog: + * 23-Apr-2001 TTC Created + */ + +#include +#include +#include +#include +#include +#include + +#include + +#include