Path: csiph.com!usenet.pasdenom.info!gegeweb.org!usenet-fr.net!nerim.net!novso.com!newsfeed.xs4all.nl!newsfeed2.news.xs4all.nl!xs4all!post.news.xs4all.nl!not-for-mail Return-Path: X-Original-To: python-list@python.org Delivered-To: python-list@mail.python.org X-Spam-Status: OK 0.015 X-Spam-Evidence: '*H*': 0.97; '*S*': 0.00; 'interpreter': 0.05; 'arguments': 0.09; 'booth': 0.09; 'optimizing': 0.09; 'python:': 0.09; 'reached.': 0.09; 'subject:while': 0.09; 'python': 0.11; 'def': 0.12; 'block.': 0.16; 'itself,': 0.16; 'jumps': 0.16; 'limit,': 0.16; 'rebound': 0.16; 'situation.': 0.16; 'subject:recursion': 0.16; 'unlikely': 0.16; 'wrote:': 0.18; 'stack': 0.19; '>>>': 0.22; 'adds': 0.24; 'header:In-Reply-To:1': 0.27; 'function': 0.29; 'specifically': 0.29; 'am,': 0.29; "doesn't": 0.30; 'skip:@ 10': 0.30; 'message-id:@mail.gmail.com': 0.30; 'code': 0.31; "skip:' 10": 0.31; 'usually': 0.31; 'depth': 0.31; 'run': 0.32; 'call.': 0.33; 'fri,': 0.33; 'implemented': 0.33; 'position.': 0.33; 'could': 0.34; 'case,': 0.35; 'definition': 0.35; 'but': 0.35; 'received:google.com': 0.35; 'there': 0.35; 'doubt': 0.36; 'possible': 0.36; 'checks': 0.38; 'to:addr:python-list': 0.38; 'fact': 0.38; 'to:addr:python.org': 0.39; 'called': 0.40; 'even': 0.60; 'eventually': 0.60; 'tell': 0.60; 'protection': 0.63; 'benefit': 0.68; 'limit': 0.70; 'increase': 0.74; 'verification': 0.83; 'allowable': 0.84; 'irrelevant': 0.84; 'verifying': 0.84; 'imagine': 0.93; '2013': 0.98 DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20120113; h=mime-version:in-reply-to:references:from:date:message-id:subject:to :content-type; bh=DN1uDDax89DX/A1tI+Yqf1LTo6txfiUZ+qZ1+3uik+M=; b=wwMvODPOnifFgjVWiXo2z+G+dM1Cv5E8sgSgvtaVdsiRbnjnZh8rnMdKoVSNbvlTsA WpxOUkyayjPOa6G4a5RWICa9ShtMUC4G6rDlbvI2eCqqufAUpr0tbxLpRQ0kunx35e3C VKvAhGwb1rn01CmjqHxNqg0SIvX0aT88sQZG+sxa02zhbXkMRFTgvjQMpHVwzos94kjb v+yRjwN+jrfDW1NLIl6v9o+k10LEKpJmLcwHAcvcGihoLvldNL0vbqeT7C0yX/ZI3afj aLOWjhx1GDugtY2X/IFvu2a4+LhL83zcHxn7CdV8vAwpWvuj5iF4cTw9+6hz2Id+gvMC wD7Q== X-Received: by 10.68.163.132 with SMTP id yi4mr5589749pbb.158.1380883328528; Fri, 04 Oct 2013 03:42:08 -0700 (PDT) MIME-Version: 1.0 In-Reply-To: References: <87had0axxy.fsf@dpt-info.u-strasbg.fr> From: Ian Kelly Date: Fri, 4 Oct 2013 04:41:28 -0600 Subject: Re: Tail recursion to while iteration in 2 easy steps To: Python Content-Type: text/plain; charset=ISO-8859-1 X-BeenThere: python-list@python.org X-Mailman-Version: 2.1.15 Precedence: list List-Id: General discussion list for the Python programming language List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , Newsgroups: comp.lang.python Message-ID: Lines: 43 NNTP-Posting-Host: 2001:888:2000:d::a6 X-Trace: 1380883336 news.xs4all.nl 15869 [2001:888:2000:d::a6]:43401 X-Complaints-To: abuse@xs4all.nl Xref: csiph.com comp.lang.python:55467 On Fri, Oct 4, 2013 at 4:16 AM, Duncan Booth wrote: > Neil Cerutti wrote: >> On 2013-10-03, Duncan Booth wrote: >>> It isn't hard to imagine adding a TAIL_CALL opcode to the >>> interpreter that checks whether the function to be called is >>> the same as the current function and if it is just updates the >>> arguments and jumps to the start of the code block. >> >> Tail call optimization doesn't involve verification that the >> function is calling itself; you just have to verfify that the >> call is in tail position. > > You misunderstood me. As usually implemented tail call recursion doesn't > require verifying that the function is calling itself, but in Python the > function could be rebound so a check would be a necessary protection > against this unlikely situation. Consider this Python: > > @some_decorator > def fact(n, acc=1): > if n <= 1: > return acc > return fact(n-1, n*acc) > > Is that tail recursion or not? > > You cannot tell whether or not that is tail recursion unless you know the > definition of 'some_decorator' and even then the answer may vary from run > to run. There is no doubt that it's a tail call. Whether it is recursion is irrelevant to optimizing it. The reason we talk about "tail call recursion" specifically is because the recursive case is the one that makes the optimization worthwhile, not because the recursion is necessary to perform the optimization. It is possible that fact is recursive but that some_decorator adds additional stack frames that are not tail calls and not optimizable. If this were the case, then the recursion would still eventually hit the stack limit, but there would still be benefit in optimizing the "fact" frames, as it would increase the allowable depth before the stack limit is reached. So I see no reason to check the identity of the called function before optimizing the tail call.