1. Technical Field
The present invention relates generally to the automated layout of integrated circuits. In particular, the present invention is directed toward automatic, optimized insertion of buffers into integrated circuit routing trees.
2. Description of Related Art
In recent years, it has become commonplace for integrated circuit designers to build an integrated circuit layout from libraries of reusable high-level modules, sometimes referred to as “macro blocks.” Proprietary macro blocks are often referred to as “intellectual property blocks” (“IP blocks”), to emphasize their relatively intangible, yet proprietary nature. Computerized integrated circuit design tools may be used to store, retrieve, and combine macro blocks into complete integrated circuits. This design philosophy of combining reusable macro blocks to produce a complex integrated circuit is known as “system-on-a-chip” (SoC) design. Designing a “system-on-a-chip” involves designing the interconnections between macro blocks. Despite the apparent simplicity of SoC design, this is often not a trivial task. The reason for this is that the connections themselves are physical components (i.e., wires) with non-ideal properties. Like all electrical conductors, integrated circuit connections suffer from delay and signal loss due to physical properties such as resistance, capacitance, and relativistic limitations on the speed at which electrons are able to travel. In order to ensure that all components in an integrated circuit are properly synchronized to work properly, it is important to take these factors into account when designing interconnections between macro blocks to minimize signal loss and to allow operation within acceptable timing specifications.
Buffer insertion is now widely recognized as a key technology for improving VLSI (Very Large Scale Integration) interconnect performance. For a buffer insertion technique to be effective, however, it must be fully aware of its surrounding blockage constraints while also being efficient enough to quickly process thousands of nets. In the buffer insertion literature, van Ginneken's dynamic programming based algorithm has established itself as a classic in the field. Van Ginneken's algorithm is described in L. P. P. van Ginneken, “Buffer placement in distributed RC-tree networks for minimal Elmore delay,” Proceedings of the IEEE International Symposium on Circuits and Systems, pp. 865-868, 1990, which is hereby incorporated by reference. Van Ginneken's algorithm assumes a Steiner tree routing topology and inserts buffers into the Steiner tree so as to minimize Elmore delay. A Steiner tree is defined as follows: If, in a weighted graph, a subset of the vertices are designated as “terminals,” a Steiner tree is a minimum-weight connected subgraph which includes all of the “terminals.” Thus, for example, a minimum spanning tree of a graph is a special case of a Steiner tree in which all of the vertices in the graph are selected as terminals.
Prior to buffer insertion, several large area chunks may be already occupied by macro or IP blocks for which wires can be routed over the blocks, but buffers cannot be inserted inside the blocks. We call these regions “buffer blockages.” For example,
A number of papers, including H. Zhou, D. F. Wong, I-M. Liu, and A. Aziz, “Simultaneous routing and buffer insertion with restrictions on buffer locations,” Proceedings of the ACM/IEEE Design Automation Conference, pp. 96-99, 1999; A. Jagannathan, S.-W. Hur, and J. Lillis, “A fast algorithm for context-aware buffer insertion,” Proceedings of the ACM/IEEE Design Automation Conference, pp. 368-373, 2000; and M. Lai and D. F. Wong, “Maze routing with buffer insertion and wiresizing,” Proceedings of the ACM/IEEE Design Automation ACM/IEEE Design Automation Conference, pp. 374-378, 2000, propose optimal algorithms for finding a minimum delay buffered path with buffer blockages. In J. Cong and X. Yuan, “Routing tree construction under fixed buffer locations,” Proceedings of the ACM/IEEE Design Automation Conference, pp. 379-384, 2000, Cong and Yuan proposed a dynamic programming algorithm, called RMP, to handle the multi-sink net buffer insertion with location restrictions. RMP is designed for the buffer block methodology for which the number of legal buffer locations is quite limited. The buffer block methodology is described in J. Cong, T. Kong, and D. Z. Pan, “Buffer block planning for interconnect-driven floorplanning,” Proceedings of the IEEE/ACM International Conference on Computer-Aided Design, pp. 358-363, 1999. RMP works on a grid graph that is constructed by adding horizontal and vertical lines through each potential buffer locations to the Hanan grid. It not only explores almost every node on the grid in tree construction but also considers many sink combinations in subsolutions. Consequently, RMP tends to be slow when either the number of net pins or legal buffer locations is large. Nevertheless, RMP generally yields near optimal solutions in term of timing performance. More recently, Tang et al. suggested a graph-based algorithm on a similar problem in X. Tang, R. Tian, H. Xiang, and D. F. Wong, “A new algorithm for routing tree construction with buffer insertion and wire sizing under obstacle constraints,” Proceedings of the IEEE/ACM International Conference on Computer-Aided Design, pp. 49-56, 2001. While more efficient than RMP, it can optimize only the maximum sink delay rather than the minimum slack.
Difficult buffering problems occur not just with large nets but also when sink polarity constraints are present. Alpert et al. developed the “buffer-aware” C-Tree heuristic to be used as a precursor to van Ginneken's algorithm. The C-Tree heuristic is described in C. J. Alpert, G. Gandham, M. Hrkic, J. Hu, A. B. Kahng, J. Lillis, B. Liu, S. T. Quay, S. S. Saptnekar, and A. J. Sullivan, “Buffered Steiner Trees for Difficult Instances,” IEEE Transactions on Computer-Aided Design, vol. 21, no. 1, January 2002, pp. 3-14, which is hereby incorporated by reference. The C-Tree method is not “blockage-aware,” however. To solve this problem, one could first run C-Tree, then invoke the algorithm of C. J. Alpert, G. Gandham, J. Hu, J. L. Neves, S. T. Quay, and S. S. Sapatnekar, “Steiner Tree Optimization for Buffers, Blockages, and Bays,” IEEE Transactions on Computer-Aided Design, vol. 20, no. 4, April 2001, pp. 556-562, hereby incorporated by reference, which performs local re-routing to avoid the blockages without adding too much wiring. Then, one could pass this modified tree to van Ginneken's buffer insertion algorithm. For example, this approach would obtain the buffered solution in FIG. 1E. However, a carefully constructed timing-driven topology can be destroyed by these local topology changes, making the final slack worse than not running local rerouting at all.
Thus, a need exists for a fast and effective technique for performing optimal buffer insertion on multi-sink nets.
The present invention is directed toward a method, computer program product, and data processing system for inserting buffers into integrated circuit routing trees. The present invention dynamically modifies a Steiner tree configuration as needed to derive a maximal slack solution that takes into account blockages such as those presented by IP blocks.
The novel features believed characteristic of the invention are set forth in the appended claims. The invention itself, however, as well as a preferred mode of use, further objectives and advantages thereof, will best be understood by reference to the following detailed description of an illustrative embodiment when read in conjunction with the accompanying drawings, wherein:
With reference now to the figures and in particular with reference to
With reference now to
An operating system runs on processor 302 and is used to coordinate and provide control of various components within data processing system 300 in FIG. 3. The operating system may be a commercially available operating system such as Windows XP, which is available from Microsoft Corporation. An object oriented programming system such as Java may run in conjunction with the operating system and provides calls to the operating system from Java programs or applications executing on data processing system 300. “Java” is a trademark of Sun Microsystems, Inc. Instructions for the operating system, the object-oriented programming system, and applications or programs are located on storage devices, such as hard disk drive 326, and may be loaded into main memory 304 for execution by processor 302.
Those of ordinary skill in the art will appreciate that the hardware in
For example, data processing system 300, if optionally configured as a network computer, may not include SCSI host bus adapter 312, hard disk drive 326, tape drive 328, and CD-ROM 330. In that case, the computer, to be properly called a client computer, includes some type of network communication interface, such as LAN adapter 310, modem 322, or the like. As another example, data processing system 300 may be a stand-alone system configured to be bootable without relying on some type of network communication interface, whether or not data processing system 300 comprises some type of network communication interface. As a further example, data processing system 300 may be a personal digital assistant (PDA), which is configured with ROM and/or flash ROM to provide non-volatile memory for storing operating system files and/or user-generated data.
The depicted example in FIG. 3 and above-described examples are not meant to imply architectural limitations. For example, data processing system 300 also may be a notebook computer or hand held computer in addition to taking the form of a PDA. Data processing system 300 also may be a kiosk or a Web appliance. The processes of the present invention are performed by processor 302 using computer implemented instructions, which may be located in a memory such as, for example, main memory 304, memory 324, or in one or more peripheral devices 326-330.
The present invention is directed toward a method, computer program product, and data processing system for inserting buffers into integrated circuit routing trees. The present invention dynamically modifies a Steiner tree configuration as needed to derive a maximal slack solution that takes into account blockages such as those presented by IP blocks.
Formal Problem Definition
For the Steiner tree construction, let Vinternal represent the set of nodes in the tree other than the source and sinks. The problem we address is formulated as follows:
Problem formulation: Given a net N={υ0, υ1, υ2, . . . , υn} with source υ0, sinks υ1, . . . , υn, load capacitances c(υi) and required arrival time q(υi) for each sink υiεN, a set of rectangles R={r1, r2, . . . rk} representing buffer blockages, and a buffer library B={b1, b2, . . . bm}, find a buffered Steiner tree T(V, E) where V=N∪Vinternal and E spans every node in v such that the required arrival time at the source is maximized.
We adopt the Elmore delay model for interconnect and an RC switch model for gate delays. The Elmore delay metric is described in R. Gupta, B. Tutuianu, and L. T. Pileggi, “The Elmore Delay as a Bound for RC Trees with Generalized Input Signals,” IEEE Transactions on Computer Aided Design of Integrated Circuits and Systems, vol. 16, no. 1, pp. 95-104, January 1997, which is hereby incorporated by reference. We assume that the given a routing tree T(V,E) is a binary tree, i.e., every internal node has no more than two children and that every sink has degree one. Any routing tree can be easily transformed to satisfy both conditions by inserting zero-length edges (i.e., pseudo-edges). Note that the choice of which sub-trees to group together can have an effect on solution quality. Grouping the subtrees together in a non-optimal way generally has limited effect on timing quality, but may waste buffers that have to be inserted for decoupling. Our implementation arbitrarily groups the child nodes.
Since a preferred embodiment of the present invention extends van Ginneken's algorithm to directly handle buffer blockages, we first overview the algorithm to form a basis for the remainder of this detailed description of the preferred embodiment. Van Ginneken's algorithm proceeds bottom-up from the leaf nodes along a given tree topology toward the source node. A set of candidate solutions is computed for each node during this process. A candidate solution at a node υ is characterized by the load capacitance c(υ) seen downstream and the required arrival time q(υ) at node υ. We use a pair s=(c(υ),q(υ)) to specify a buffering solution at υ. For any two candidate solutions s1=(c1(υ),q1(υ)) and s2=(c2(υ),q2(υ)), s1 is dominated by (inferior to) s2 if c1(υ)≧c2(υ) and q1(υ)≦q2(υ). A candidate solution set S(υ)={s1, s2, . . . } is a non-dominating set if no solution in this set is dominated by any other solution in this set. During the bottom-up process of van Ginneken's algorithm, the candidate solutions at leaf node evolve through the following operations:
After a set of candidate solutions are propagated to the source, the solution with the maximum required arrival time is selected for the final solution. For a fixed routing tree, van Ginneken's algorithm can find the optimal solution in O(n2) time if there are n pins in this net.
C. Alpert and A. Devgan, “Wire Segmenting for Improved Buffer Insertion,” Proceedings of the ACM/IEEE Design Automation Conference, pp. 588-593, 1997, hereby incorporated by reference, describes an algorithm for segmenting wires to increase the number of buffer insertion points available to a buffer insertion algorithm. If such a wire segmenting technique is used, n should be the number of candidate insertion points.
Algorithm Strategy
A common strategy to solve a sophisticated problem is divide-and-conquer, i.e., partitioning a complex problem into a set of subproblems in manageable scales. Such partitioning can be performed on either physical or design flow aspects. For example, a large net can be physically clustered into smaller nets as in the aforementioned C-Tree algorithm for computing a Steiner tree. Such partitioning not only speeds up the problem solving process, but also isolates subproblems according to their natures so that scattered targets can be avoided and the optimization can be well focused. Separating the Steiner tree construction from buffer insertion procedure is an example of partitioning the design flow. An initial Steiner tree construction can limit the buffer solution search along an anticipatedly good direction. A directional search is should be more efficient than the simultaneous routing and buffer insertion which is an implicitly brute-force search, even though the search may intelligently prune some unnecessary candidate solutions.
When considering how to incorporate blockage constraints, it is important to address the problem at an appropriate phase in the design flow. Blockage avoidance is more tied the generation of buffering solutions, i.e., it is hard to know how to make a Steiner tree avoid blockages without knowing where buffers are needed. A simultaneous approach is generally less efficient while separate routing and buffer insertion approach may not adequately plan for blockages. However, we can move the partitioning line to the middle of these two approaches, i.e., we can generate a Steiner tree which is allowed to be adjusted during buffer insertion construction according to dynamic requests for buffer blockage avoidance. Our key idea is to explore only a handful of alternative buffer insertion locations for which the tree topology can be modified (as opposed to an approach like buffered P-Tree, described in J. Lillis, C. K. Cheng, and T. Y. Lin, “Simultaneous routing and buffer insertion for high performance interconnect,” Proceedings of the Great Lake Symposium on VLSI, pp. 148-153, 1996, which explores a much larger space). Exploring these alternative locations corresponds to moving a branch node outside a blockage which enables opportunities for decoupling and efficient driving of long paths.
Buffer blockages along paths that do not contain any Steiner nodes (i.e., nodes or branching points in the Steiner tree) can be mitigated relatively easily by allowing them to take multi-bend route without increasing wirelength. This type of solution can be achieved by applying the work in the incorporated C. J. Alpert, G. Gandham, J. Hu, J. L. Neves, S. T. Quay, and S. S. Sapatnekar, “Steiner Tree Optimization for Buffers, Blockages, and Bays,” IEEE Transactions on Computer-Aided Design, vol. 20, no. 4, April 2001, pp. 556-562 to obtain a Steiner tree that has L-shapes and Z-bends that minimize overlap with blockages but no additional wirelength or tree topology adjustment. The difficult buffer blockage problems occur when a Steiner node lies on top of blockage which eliminates opportunities for decoupling non-critical paths and for driving long wires directly. We address this problem by generating alternative candidate solutions within van Ginneken's algorithm by trying an alternate location outside of blockage for the branching Steiner node.
The Basic RIATA Algorithm
Given a Steiner tree, we extend the van Ginneken's algorithm so that the tree topology is adaptively adjusted during the bottom-up candidate solution propagation process, i.e., buffer insertion is not restricted to a fixed topology any more. During the bottom-up propagation process, if a Steiner point does not overlap a buffer blockage, our algorithm proceeds in the same way as the van Ginneken's algorithm. The difference occurs when a Steiner point 400 is within a buffer blockage 402, as depicted in FIG. 4A. To compensate for the inability to have possible buffer insertion candidates near blocked Steiner point 400, we seek alternative unblocked sites nearby to use instead. For the sake of simplicity, the alternative point is searched only between node v 400 and its parent node υp 401. Within bounding box 404 lying between nodes υ 400 and υp 401, we search for an unblocked point that is the closest to υ 400. Other searching schemes will be introduced in the next section. In our example,
Before we propagate the candidate solutions from the children nodes υt 403 and υr 405, we search for the least blocked path to their parent nodes υ 400 and υ′ 407 through the technique presented in the incorporated C. J. Alpert, G. Gandham, J. Hu, J. L. Neves, S. T. Quay, and S. S. Sapatnekar, “Steiner Tree Optimization for Buffers, Blockages, and Bays,” IEEE Transactions on Computer-Aided Design, vol. 20, no. 4, April 2001, pp. 556-562. By carefully choosing the cost, this technique can provide a path connecting a node a and another node b such that the path length is the shortest and the total path length overlapping with buffer blockages is minimized. We will hereafter refer to a subroutine LeastBlockedPath(a, b) to denote the application of this technique to find the least blocked path between child node b and parent node a, followed by an application of the wire segmenting algorithm described in the incorporated C. Alpert and A. Devgan, “Wire Segmenting for Improved Buffer Insertion,” Proceedings of the ACM/IEEE Design Automation Conference, pp. 588-593, 1997. Thus, by calling LeastBlockedPath(υ, υt), LeastBlockedPath(υ, υr), LeastBlockedPath(υ′, υt), and LeastBlockedPath(υ′, υr), candidate solutions containing the least blocked paths between child nodes υt 403 and υr 405, and parent nodes υ 400 and υ′ 407 may be derived.
Next, we propagate candidate solutions from υt 403 and υr 405 through the least blocked paths to both υ 400 and υ′ 407. Note that during this propagation process, more candidate solutions may be generated by inserting buffers at segmenting points along the paths (e.g., buffer 408 in
The Enhanced Algorithm (RIATA+)
In previous section, we introduced the basic RIATA algorithm in which only one alternative point is searched for each Steiner node between itself and its parent node. When a Steiner node and its parent node are both in the same blockage, no alternative point will be found. This is illustrated in the example in
We illustrate this enhanced search scheme through the example in
If parent node υj 602 of υi 600 is a Steiner node in a blockage as in
In a preferred embodiment, the pruning scheme is defined by the following rules: (1) when we propagate candidate solutions from each node of {tilde over (V)}(υi) to each node of {tilde over (V)}(υj), any monotone propagation is allowed; (2) propagation from υi to any node in {tilde over (V)}(υj) is always allowed in order to ensure there is at least one set of solutions propagated to every node in {tilde over (V)}(υj); and (3) any non-monotone propagation from a node other than υi is disallowed. For the example depicted in
Searching alternative Steiner points on four boundaries of the blockage guarantees that alternative points can always be found unless the whole chip area is blocked. Furthermore, this search scheme allows Steiner nodes to be spread out around the blockage if there are multiple Steiner nodes in the same blockage as shown in FIG. 6C. If we consider only one alternative point for each Steiner node, the alternative Steiner nodes may be crowded on single side of the blockage. We allow candidate solutions from υi to be propagated to every node in {tilde over (V)}(υj) in
To implement this heuristic we need to efficiently find the closest unblocked node set {tilde over (V)}(υ) for a node υ. Given a node υ and a set of rectangles R={r1, r2, . . . , rk} representing the buffer blockages, if node υ is within a blockage rεR, we need to find the unblocked points which is the closest to υ on each boundary of r. If there is no overlap between any two buffer blockages, all we need to do is to locate the rectangle r that overlaps υ. If r is defined by bounding coordinates (xto,yto,xhi,yhi) and υ is located at point (xυ,yυ), the unblocked points closest to υ on each boundary of r are (xtυ,yυ), (xυ,yhi), (xhi,yυ) and (xυ,ytυ). If the set of rectangles R is stored as an interval tree, the desired rectangle r can be found in O(k) time in the worst case. Interval trees are described in T. H. Cormen, C. E. Leiserson, R. L. Rivest, and C. Stein, Introduction to Algorithms, 2d. Ed., MIT Press, 2001, pp. 311-316, which is hereby incorporated by reference.
We let UnblockedNodes(υ,R) denote a procedure that finds such an unblocked node set.
We call this enhanced algorithm “RIATA+.” A preferred embodiment of the RIATA+ algorithm is described in
Turning now to the pseudo-code description provided in
On line 4, if the current node υ is a sink (i.e., a leaf node of the Steiner tree), then capacitance c(υ) and required arrival time q(υ) for the sink are returned as a solution. On line 5, since the current node υ is not a sink, it has at least one child (i.e., the left child of υ). FindCandidates is then called recursively to find a candidate solution set for the left child node υt. On line 6, the Propagate subroutine is called to propagate the solution set S({tilde over (V)}(υt) for the candidate node set {tilde over (V)}(υt) for the left child node υt to the candidate node set {tilde over (V)}(υt) for the current node υ to obtain a new solution set St({tilde over (V)}(υ)). If, on line 7, υ only has a left child, the solution set St({tilde over (V)}(υ)) is returned as the solution for υ.
If υ has two children, however (since a binary Steiner tree is assumed), FindCandidates is called recursively on line 8 to find a solution set S({tilde over (V)}(υr)) for the right child node υr. On line 9, this solution set Sr({tilde over (V)}(υr)) is propagated to the current candidate node set {tilde over (V)}(υ) using the Propagate subroutine to obtain a new solution set Sr({tilde over (V)}(υ)). The solution sets St({tilde over (V)}(υ)) and Sr({tilde over (V)}(υ)) are then merged on line 10 using the Merge operation previously described in conjunction with van Ginneken's algorithm to obtain a solution set S({tilde over (V)}(υ)) for the current node υ. Finally, on line 11, this solution set S({tilde over (V)}(υ)) is returned.
On line 0, the routine starts by setting the solution set S({tilde over (V)}(υj)) to the empty set (∅). Lines 1 and 2 establish a doubly nested loop iterating over the Cartesian product of the node sets ({tilde over (V)}(υi)×{tilde over (V)}(υj)). In other words, the doubly nested loop established by lines 1 and 2 iterates over all pairings of an element from {tilde over (V)}(υi) with an element from {tilde over (V)}(υj) Each iteration, the element from {tilde over (V)}(υi) is denoted υi,k and the element from {tilde over (V)}(υj) is denoted υj,l.
Line 3 states that at each iteration of the loop, a determination is made as to whether a propagation from υi,k to υj,l (written as υi,kυj,l) is monotone, according to previously stated definition, or if υi,k is the same node as υi. If either of these condition is true, then lines 4-8 are executed. Otherwise, the loop cycles through another iteration.
Line 4 calls the aforementioned LeastBlockedPath routine to calculate the least blocked path P from υi,k to υj,l according to the algorithm described in the incorporated C. J. Alpert, G. Gandham, J. Hu, J. L. Neves, S. T. Quay, and S. S. Sapatnekar, “Steiner Tree Optimization for Buffers, Blockages, and Bays,” IEEE Transactions on Computer-Aided Design, vol. 20, no. 4, April 2001, pp. 556-562. On line 5, the path P is segmented according to the algorithm described in the incorporated C. Alpert and A. Devgan, “Wire Segmenting for Improved Buffer Insertion,” Proceedings of the ACM/IEEE Design Automation Conference, pp. 588-593, 1997. On line 6, a variable called υp is set to the value of υi,k.
At line 7, a loop is executed that generates a number of candidate solutions, each representing a buffer insertion at a different position (i.e., at a different segment) along the segmented path P. Each of these candidate solutions is added to the final solution set S({tilde over (V)}(υj)) on line 8.
Line 9, which is executed after the doubly nested loop terminates, prunes any inferior solutions from S({tilde over (V)}(υj)). Finally, on line 10, the solution set S({tilde over (V)}(υj)) is returned.
The RIATA+ algorithm is actually a generalization of the basic RIATA algorithm: If we define {tilde over (V)}(υi) to include only υi and its nearest unblocked point between υi and its parent υj, we obtain the basic RIATA algorithm. One of ordinary skill in the art will recognize that by varying the definition of {tilde over (V)}(υi), many variations on the general RIATA+ algorithm are possible without departing from the scope and spirit of the present invention, each with varying levels of solution quality and runtime performance. For example, one can include more alternative Steiner points in {tilde over (V)}(υi) or even allow non-monotone propagations when the net is extremely timing-critical or its size is small.
Complexity
Given a net with n insertion points and m pins, a buffer library B and k rectangles representing blockages, if the maximal candidate solution set size is g and the maximal expanded Steiner node set size is h, then the complexity of our heuristic is O(g·n·|B|·h2+m·k). The term of m·k comes from the operations of searching unblocked alternative Steiner points. The values for the constant h are 2 for the RIATA heuristic and 5 for the RIATA+ heuristic. We may assume that the capacitance value in each candidate solution can take only a polynomially bounded integer, thus the complexity of our heuristic is pseudo-polynomial.
Flowchart
If the current node is a sink (block 908: Yes), then a solution is returned consisting of the capacitance and required arrival time for the current node (block 910). If not (block 908: No), then the routine in
A determination is then made as to whether the current node has a right child in addition to a left child (block 916). If not (block 916: No), the solution set created in block 914 is simply returned (block 918). If so (block 916: Yes), however, the routine in
It is important to note that while the present invention has been described in the context of a fully functioning data processing system, those of ordinary skill in the art will appreciate that the processes of the present invention are capable of being distributed in the form of a computer readable medium of instructions or other functional descriptive material and in a variety of other forms and that the present invention is equally applicable regardless of the particular type of signal bearing media actually used to carry out the distribution. Examples of computer readable media include recordable-type media, such as a floppy disk, a hard disk drive, a RAM, CD-ROMs, DVD-ROMS, and transmission-type media, such as digital and analog communications links, wired or wireless communications links using transmission forms, such as, for example, radio frequency and light wave transmissions. The computer readable media may take the form of coded formats that are decoded for actual use in a particular data processing system. Functional descriptive material is information that imparts functionality to a machine. Functional descriptive material includes, but is not limited to, computer programs, instructions, rules, facts, definitions of computable functions, objects, and data structures.
The description of the present invention has been presented for purposes of illustration and description, and is not intended to be exhaustive or limited to the invention in the form disclosed. Many modifications and variations will be apparent to those of ordinary skill in the art. The embodiment was chosen and described in order to best explain the principles of the invention, the practical application, and to enable others of ordinary skill in the art to understand the invention for various embodiments with various modifications as are suited to the particular use contemplated.
Number | Name | Date | Kind |
---|---|---|---|
5629860 | Jones et al. | May 1997 | A |
6044209 | Alpert et al. | Mar 2000 | A |
6401234 | Alpert et al. | Jun 2002 | B1 |
6467069 | Mehrotra et al. | Oct 2002 | B2 |
6519745 | Srinivas et al. | Feb 2003 | B1 |
6557145 | Boyle et al. | Apr 2003 | B2 |
6591411 | Alpert et al. | Jul 2003 | B2 |
6615401 | Gasanov et al. | Sep 2003 | B1 |
6622291 | Ginetti | Sep 2003 | B1 |
20030212976 | Drumm | Nov 2003 | A1 |
Number | Date | Country | |
---|---|---|---|
20040123261 A1 | Jun 2004 | US |