/* $XFree86: xc/programs/Xserver/hw/xfree86/drivers/nsc/panel/platform.c,v 1.3tsi Exp $ */ /* * $Workfile: platform.c $ * $Revision$ * * File Contents: This file contains platform dependent functions * which provide interface to that platform. * * * SubModule: Geode FlatPanel library * */ /* * NSC_LIC_ALTERNATIVE_PREAMBLE * * Revision 1.0 * * National Semiconductor Alternative GPL-BSD License * * National Semiconductor Corporation licenses this software * ("Software"): * * Panel Library * * under one of the two following licenses, depending on how the * Software is received by the Licensee. * * If this Software is received as part of the Linux Framebuffer or * other GPL licensed software, then the GPL license designated * NSC_LIC_GPL applies to this Software; in all other circumstances * then the BSD-style license designated NSC_LIC_BSD shall apply. * * END_NSC_LIC_ALTERNATIVE_PREAMBLE */ /* NSC_LIC_BSD * * National Semiconductor Corporation Open Source License for * * Panel Library * * (BSD License with Export Notice) * * Copyright (c) 1999-2001 * National Semiconductor Corporation. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * * Redistributions in binary form must reproduce the above * copyright notice, this list of conditions and the following * disclaimer in the documentation and/or other materials provided * with the distribution. * * * Neither the name of the National Semiconductor Corporation nor * the names of its contributors may be used to endorse or promote * products derived from this software without specific prior * written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "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 * NATIONAL SEMICONDUCTOR CORPORATION OR CONTRIBUTORS 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, * INTELLECTUAL PROPERTY INFRINGEMENT, OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY * OF SUCH DAMAGE. * * EXPORT LAWS: THIS LICENSE ADDS NO RESTRICTIONS TO THE EXPORT LAWS OF * YOUR JURISDICTION. It is licensee's responsibility to comply with * any export regulations applicable in licensee's jurisdiction. Under * CURRENT (2001) U.S. export regulations this software * is eligible for export from the U.S. and can be downloaded by or * otherwise exported or reexported worldwide EXCEPT to U.S. embargoed * destinations which include Cuba, Iraq, Libya, North Korea, Iran, * Syria, Sudan, Afghanistan and any other country to which the U.S. * has embargoed goods and services. * * END_NSC_LIC_BSD */ /* NSC_LIC_GPL * * National Semiconductor Corporation Gnu General Public License for * * Panel Library * * (GPL License with Export Notice) * * Copyright (c) 1999-2001 * National Semiconductor Corporation. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted 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 * * In addition to the terms of the GNU General Public License, neither * the name of the National Semiconductor Corporation nor the names of * its contributors may be used to endorse or promote products derived * from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "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 * NATIONAL SEMICONDUCTOR CORPORATION OR CONTRIBUTORS 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, * INTELLECTUAL PROPERTY INFRINGEMENT, OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY * OF SUCH DAMAGE. See the GNU General Public License for more details. * * EXPORT LAWS: THIS LICENSE ADDS NO RESTRICTIONS TO THE EXPORT LAWS OF * YOUR JURISDICTION. It is licensee's responsibility to comply with * any export regulations applicable in licensee's jurisdiction. Under * CURRENT (2001) U.S. export regulations this software * is eligible for export from the U.S. and can be downloaded by or * otherwise exported or reexported worldwide EXCEPT to U.S. embargoed * destinations which include Cuba, Iraq, Libya, North Korea, Iran, * Syria, Sudan, Afghanistan and any other country to which the U.S. * has embargoed goods and services. * * You should have received a copy of the GNU General Public License * along with this file; if not, write to the Free Software Foundation, * Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * * END_NSC_LIC_GPL */ #define LINUX_ROM_SEGMENT 0x000F #define SEGMENT_LENGTH 0xFFFF #define PAGE_LENGTH 0x1000 #define SYS_BOARD_NAME_LEN 24 #define PLT_READ 0 #define PLT_WRITE 1 #define PLT_WRITE_BYTES 2 #define PLT_READ_BYTES 3 #define PLT_WRITE_WORDS 4 #define PLT_READ_WORDS 5 #define PLT_WRITE_DWORDS 6 #define PLT_READ_DWORDS 7 #define PLT_UNKNOWN ((SYS_BOARD) 0xFFFF) typedef struct { char sys_board_name[SYS_BOARD_NAME_LEN]; SYS_BOARD sys_board; } SYS_BOARD_INFO; static SYS_BOARD_INFO Sys_info; /* * The names in the sys_board_name string must exactly match the names in the * BIOS header. These names are used by FindStringInSeg() to find the names * in the BIOS header space. The BIOS does not use OTHER; it is a dummy value * for program useonly. * */ SYS_BOARD_INFO Sys_board_info_array[] = { {"Marmot", MARMOT_PLATFORM}, {"Unicorn", UNICORN_PLATFORM}, {"Centaurus", CENTAURUS_PLATFORM}, {"Aries", ARIES_PLATFORM}, {"Carmel", CARMEL_PLATFORM}, {"Hyrda", HYDRA_PLATFORM}, {"Dorado", DORADO_PLATFORM}, {"Redcloud", REDCLOUD_PLATFORM}, {"Other", OTHER_PLATFORM} }; #define NUM_SYS_BOARD_TYPES sizeof(Sys_board_info_array)/sizeof(SYS_BOARD_INFO) static int Num_sys_board_type = NUM_SYS_BOARD_TYPES; SYS_BOARD_INFO *Sys_board_array_base = Sys_board_info_array; int FindStringInSeg(unsigned int, char *); static unsigned char get_sys_board_type(SYS_BOARD_INFO *, SYS_BOARD_INFO *); #if defined(linux) && !defined(__KERNEL__) #if !defined(XFree86Server) static void protected_mode_access(unsigned int, unsigned int, unsigned long, unsigned char *); static void setup_pma(); static void close_pma(); static int fd; #endif /* IN_MODULE */ #endif /* __KERNEL__ */ /* Detect the Platform */ int Detect_Platform(void) { #if defined(linux) && !defined(__KERNEL__) #if !defined(XFree86Server) setup_pma(); #endif /* IN_MODULE */ #endif /* __KERNEL__ */ /* See if we can find the board name using Xpressrom */ if (get_sys_board_type(&Sys_info, Sys_board_array_base) == TRUE) { #if 0 if (Sys_info.sys_board == CENTAURUS_PLATFORM) { printk("CENTAURUS Platform Found\n"); } else if (Sys_info.sys_board == DORADO_PLATFORM) { printk("DORADO Platform Found \n"); } else { printk("UNKNOWN Platform Found \n"); } #endif } #if defined(linux) && !defined(__KERNEL__) #if !defined(XFree86Server) close_pma(); #endif /* IN_MODULE */ #endif /* __KERNEL__ */ return (Sys_info.sys_board); } static int Strncmp(char *str1, char *str2, int len) { int i; if ((str1 == 0x0) || (str2 == 0x0) || (len == 0)) return (1); for (i = 0; i < len; i++) { if (*(str1 + i) > *(str2 + i)) { return 1; } else if (*(str1 + i) < *(str2 + i)) { return -1; } } return 0; } static char * Strcpy(char *dst, char *src) { int i; if ((dst == 0x0) || (src == 0x0)) return (0); for (i = 0; src[i] != 0x0; i++) { dst[i] = src[i]; } dst[i] = 0x0; /* NULL termination */ return dst; } static int Strlen(char *str) { int i; if (str == 0x0) return 0; for (i = 0; str[i] != 0x0; i++) ; return i; } /***********************************************************************/ /* Platform Detection Code */ /***********************************************************************/ /************************************************************************ * int FindStringInSeg( unsigned int segment_address, char *string_ptr ) * Returns the offset where the NULL terminated string pointed to by * string_ptr is located in the segment passed in segment_address. * Segment_address must be of the form 0xXXXX (i.e 0xf000 for segment f). * Returns NULL if the string is not found. ************************************************************************ */ int FindStringInSeg(unsigned int segment_address, char *string_ptr) { int string_length = Strlen(string_ptr); char *psegment_buf; unsigned long mem_ptr = (unsigned long)segment_address << 16; char segment_buffer[SEGMENT_LENGTH + 1]; int i, cursor = 0; /* silence compiler */ (void)cursor; (void)mem_ptr; (void)segment_buffer; #if defined(linux) && !defined(XFree86Server) #ifdef __KERNEL__ XpressROMPtr = (unsigned char *)ioremap(mem_ptr, SEGMENT_LENGTH + 1); psegment_buf = (char *)XpressROMPtr; #else /* Fill the segment_buffer with 16 page accesses */ for (cursor = 0; (cursor * PAGE_LENGTH) < SEGMENT_LENGTH; cursor++) { protected_mode_access(PLT_READ_BYTES, PAGE_LENGTH, mem_ptr + (cursor * PAGE_LENGTH), &(segment_buffer[(cursor * PAGE_LENGTH)])); } psegment_buf = segment_buffer; #endif /* __KERNEL__ */ #elif defined (XFree86Server) psegment_buf = (char *)XpressROMPtr; #elif defined(_WIN32) /* Windows */ psegment_buf = XpressROMPtr; #endif /* Now search for the first character of the string_ptr */ for (i = 0; i < SEGMENT_LENGTH + 1; i++) { if (*(psegment_buf + i) == *string_ptr) { /* If we match the first character, do a * string compare. */ if (!Strncmp(string_ptr, (psegment_buf + i), string_length)) { /* They match! */ return (1); } } } /* if we got this far we didn't find anything. Return NULL. */ return (0); } /* end FindStringInSeg() */ /********************************************************************** * TRUE_FALSE get_sys_board_type( SYS_BOARD_INFO *sys_info, * SYS_BOARD_INFO *sys_board_array_base) Checks the system * BIOS area for Xpressrom information. If found, searches the BIOS * area for one of names in the array pointed to by sys_board_array_ptr. * If a match is found, sets the SYS_INFO system_board_name string * and the system_board variable to the board name and returns TRUE. * If Xpressrom or a board is not found, sets the variables to * their default values and returns FALSE. * Uses the global string pointer *xpress_rom_string_ptr. * *********************************************************************** */ static unsigned char get_sys_board_type(SYS_BOARD_INFO * sys_info, SYS_BOARD_INFO * sys_board_array_base) { int index; char *xpress_rom_string_ptr = "XpressStart"; unsigned int segment = LINUX_ROM_SEGMENT; /* See if XpressStart is present in the BIOS area. * If it is, search for a board string. If not, Xpressrom is * not present, set system_board information to UNKNOWN and * return FALSE. */ if (!FindStringInSeg(segment, xpress_rom_string_ptr)) { sys_info->sys_board = PLT_UNKNOWN; Strcpy(sys_info->sys_board_name, "Unknown"); return (FALSE); } else { /* we have Xpressrom, so look for a board */ for (index = 0; index < Num_sys_board_type; index++) { if (!FindStringInSeg(segment, (sys_board_array_base + index)->sys_board_name)) { continue; } else { /* a match!! */ sys_info->sys_board = (sys_board_array_base + index)->sys_board; Strcpy(sys_info->sys_board_name, (sys_board_array_base + index)->sys_board_name); return (TRUE); } } /* end for() */ } /* end else */ /* if we are here we have failed */ sys_info->sys_board = PLT_UNKNOWN; Strcpy(sys_info->sys_board_name, "Unknown"); return (FALSE); } /* end get_sys_board_type() */ #if defined(linux) && !defined(__KERNEL__) #if !defined(XFree86Server) /****************************************************************** * * protected_mode_access( unsigned int mode, unsigned int width, * unsigned long addr, unsigned char* pdata ) * This function provides access to physical memory * at the requested address. * mode is: PLT_READ or PLT_WRITE (accesses a single byte, word * or double word depending on the value of "width". * Only 1, 2 or 4 supported). * PLT_READ_BYTES, PLT_WRITE_BYTES accesses "width" number * of bytes (8 bits) * PLT_READ_WORDS, PLT_WRITE_WORDS accesses "width" number * of words (16 bits) PLT_READ_DWORDS, PLT_WRITE_DWORDS accesses * "width" number of dwords (32 bits) * width is: The size of the access. * For PLT_READ or PLT_WRITE, only 1, 2 and 4 are * supported. For other modes, width is not limited but * will cause paging if the block traverses page boundaries. * addr is: The physical address being accessed * pdata is: A pointer to the data to be read or written into. * NOTE! WORD or DWORD accesses can only be made on * WORD or DWORD boundaries! * ****************************************************************** */ static void protected_mode_access(unsigned int mode, unsigned int width, unsigned long addr, unsigned char *pdata) { #define PMTRASH 0x12345678L unsigned long base; /* The physical page address */ int length = 0x1000; /* the page size is 4k */ unsigned int offset = 0; /* The physical addr offset into page */ unsigned int index = 0; /* Used to read/write from/to a block */ unsigned int chunk = 0; /* The amount to read/wr from THIS block */ unsigned int size = 0; /* Data size shift value (to avoid math) */ static void *ptr; /* pointer to real memory location. */ static unsigned long lastbase = PMTRASH; /* temp storage of previous base used. */ /* type specific buffer pointers */ unsigned char *byte_data = (unsigned char *)pdata; unsigned int *word_data = (unsigned int *)pdata; unsigned long *dword_data = (unsigned long *)pdata; switch (mode) { case PLT_READ_WORDS: case PLT_WRITE_WORDS: size = 1; break; case PLT_READ_DWORDS: case PLT_WRITE_DWORDS: size = 2; } /* Check if we're in the user accessable range */ if (addr < 0xFF000000L) { /* We get physical memory in "pages", defined by the * following "base" address and the "offset" into it. * "base" will be used with mmap to get "ptr", which * points to the memory mapped actual physical memory at * the address pointed-to by "base". * "width" and "chunk" are in units of whatever data * type we're reading. * "length" and "offset" are in units of bytes. * "width" and "chunk" must be adjusted with "<>size" to make into the * proper type units when done. */ base = addr & 0xFFFFF000L; offset = addr & 0x00000FFFL; do { if ((offset + (width << size)) > length) { /* Block being read extends beyond the * page boundary. Adjust things. */ chunk = (length - offset) >> size; /* Figure the chunk size */ width -= chunk; /* Reduce width by the current chunk */ } else { /* Block being read is within the * page boundary. */ chunk = width; width = 0; /* set to zero so we'll exit at the end */ } /* We keep the page around in case we need to * access it again. * This saves us some time if we have consecutive * accesses. */ if (base != lastbase) { /* we haven't mmap'd this address * Have to get a new page. Free the * previous page, if it's valid (ie, not * PMTRASH). If not, unmap it and get the * new page. */ if (lastbase != PMTRASH) munmap(ptr, length); ptr = mmap(NULL, length, PROT_READ | PROT_WRITE, MAP_SHARED, fd, base); if ((int)ptr == -1) { lastbase = PMTRASH; return; /* error */ } } /* Now we're ready to get the data. * It's pure memory access now, no funny * function calls, however we do cast things to get * the right size data. */ /* Scale the offset for the data type size */ index = offset >> size; /* Note that the above line and below lines, * which shift "offset", discard address information * if you happen to be trying to write, for example, * dwords on non-dword boundaries. */ /* Note that cases PLT_READ and PLT_WRITE don't * use "index". They shift "offset" on their own. * This is because in PLT_READ and PLT_WRITE modes, * the information on the size of the data * transaction is in the "width" variable not "size". * We also need separate cases to cast the values * right. */ switch (mode) { case PLT_READ:{ switch (chunk) { case FOUR_BYTES: *(dword_data) = (unsigned long) (*(((unsigned long *)ptr) + (offset >> 2))); break; case TWO_BYTES: *(word_data) = (unsigned int) (*(((unsigned int *)ptr) + (offset >> 1))); break; default: *(byte_data) = (unsigned char) (*(((unsigned char *)ptr) + (offset))); break; } /* end switch() */ break; } /* end case PLT_READ */ case PLT_WRITE:{ switch (chunk) { case FOUR_BYTES: *(((unsigned long *)ptr) + (offset >> 2)) = *dword_data; break; case TWO_BYTES: *(((unsigned int *)ptr) + (offset >> 1)) = *word_data; break; default: *(((unsigned char *)ptr) + (offset)) = *byte_data; break; } /* end switch() */ break; } /* end case PLT_WRITE */ case PLT_READ_BYTES:{ for (; chunk > 0; chunk--) { *(byte_data++) = (unsigned char)(*(((unsigned char *)ptr) + (index++))); } break; } /* end case PLT_READ_BYTES */ case PLT_WRITE_BYTES:{ for (; chunk > 0; chunk--) { *(((unsigned char *)ptr) + (index++)) = *(byte_data++); } break; } /* end case PLT_WRITE_BYTES */ case PLT_READ_WORDS:{ for (; chunk > 0; chunk--) { *(word_data++) = (unsigned int) (*(((unsigned int *)ptr) + (index++))); } break; } /* end case PLT_READ_WORDS */ case PLT_WRITE_WORDS:{ for (; chunk > 0; chunk--) { *(((unsigned int *)ptr) + (index++)) = *(word_data++); } break; } /* end case PLT_WRITE_WORDS */ case PLT_READ_DWORDS:{ for (; chunk > 0; chunk--) { *(dword_data++) = (*(((unsigned long *)ptr) + (index++))); } break; } /* end case PLT_READ_DWORDS */ case PLT_WRITE_DWORDS:{ for (; chunk > 0; chunk--) { *(((unsigned long *)ptr) + (index++)) = *(dword_data++); } break; } /* end case PLT_WRITE_DWORDS */ } /* end switch(mode) */ lastbase = base; /* Save the page we've just processed. */ if (width) { /* If there's still width left to get. */ base += length; /* Increment to the next page. */ offset = 0; /* Set the offset to zero. */ } } while (width); /* While there's still data to get. */ return; } /* end for if addr */ else { printf("PMA error: Unable to read ROM address space\n"); exit(1); } return; } /************************************************************************ * setup_pma() loads the ROM memory access module and initializes * memory access file descriptor (access is handled through a file-like * interface). ************************************************************************ */ static void setup_pma() { fd = open("/dev/mem", 2); /* O_RDWR */ if (fd == -1) { printf("Error: Unable to open /dev/mem !\a\n"); exit(1); } return; } /********************************************************************** * close_pma() cleans up the open memory access devices and file * descriptors. ********************************************************************** */ static void close_pma() { close(fd); return; } #endif /* IN_MODULE */ #endif /* linux && !__KERNEL__ */